Data structure – Learning notes integration

Data Structures – Stacks (Study Notes)

Data Structures – Queues (Study Notes)

Data Structures – Linked Lists (Study Notes)

Data Structures – Collections (Study Notes)

| dictionary and hash table data structure (front end call it objects)

Data structures – binary trees and binary search trees

| binary heap data structure

preface

In a Set, our focus is on each value itself. A Set stores elements as values, whereas in a dictionary, it stores data as key, value pairs. Dictionaries are also called maps, symbol tables, or associative arrays.

A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. JavaScript objects are essentially collections of key-value pairs (Hash structures), but traditionally strings can only be used as keys, so ES6 brings Map class and WeakMap class, a weaker version of Map class.

Collections, hash tables, and dictionaries are all data structures used to store unique (non-repeating) values.

Dictionary (Dictionary class)

define

In dictionaries, data is stored in pairs of keys. Dictionaries are also called maps, symbol tables, or associative arrays.

Actual dictionary

An actual dictionary (words and their definitions)

Address book

Code implementation

Dictionaries ideally use strings as key names, and values can be of any data type, so you need a method to convert all key names to strings. Make it easier for dictionary classes to search or retrieve values.

/** * Dictionaries ideally use strings as key names, and values can be of any type (from primitive types such as numbers and strings to complex objects) * This method converts the passed key names to strings *@param {*} The key name passed to item *@returns {string}* /
export function defaultToString(item) {
  if (item === null) { // If key is null
    return 'NULL'; // Returns a NULL string
  }
  if (item === undefined) { // If key is undefined
    return 'UNDEFINED'; // Returns UNDEFINED as a string
  }
  if (typeof item === 'string' || item instanceof String) { // If it is a string, return it directly
    return `${item}`;
  }
  return item.toString(); // Otherwise return it as a string
}
Copy the code

In a dictionary, the values in the dictionary need to store both the original key and value in the dictionary for subsequent extension of the dictionary class.

/** * ValuePair class * To save information, we also need to save the original key and value *@class ValuePair* /
export class ValuePair {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  /** ** returns the key and value as strings *@returns {string}
   * @memberof ValuePair* /
  toString() {
    return ` [#The ${this.key}: The ${this.value}] `; }}Copy the code

With these in hand, you are ready to implement the final dictionary class.

/** * In a Set, our focus is on each value itself. A Set stores elements as [value, value], whereas in a dictionary, it stores data as [key, value] pairs. Dictionaries are also called maps, symbol tables, or associative arrays. A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. The Object of JavaScript is essentially a set of key-value pairs (Hash structure), but traditionally only strings can be used as keys. Therefore, ES2015 brings Map class and WeakMap class, a weakened version of Map class. Collections, hash tables, and dictionaries are all data structures used to store unique (non-repeating) values. * /

/** * Dictionaries ideally use strings as key names, and values can be of any type (from primitive types such as numbers and strings to complex objects) * This method converts the passed key names to strings *@param {*} The key name passed to item *@returns {string}* /
export function defaultToString(item) {
  if (item === null) { // If key is null
    return 'NULL'; // Returns a NULL string
  }
  if (item === undefined) { // If key is undefined
    return 'UNDEFINED'; // Returns UNDEFINED as a string
  }
  if (typeof item === 'string' || item instanceof String) { // If it is a string, return it directly
    return `${item}`;
  }
  return item.toString(); // Otherwise return it as a string
}
/** * ValuePair class * To save information, we also need to save the original key and value *@class ValuePair* /
export class ValuePair {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  /** ** returns the key and value as strings *@returns {string}
   * @memberof ValuePair* /
  toString() {
    return ` [#The ${this.key}: The ${this.value}] `; }}// Implement Dictionary class based on ES2015 Map class design idea
export class Dictionary {
  constructor(toStrFn = defaultToString) {
    this.toStrFn = toStrFn; // You can also pass in custom functions to specify how to convert a key to a string
    this.table = {}; // Use an Object instance to store dictionary elements
  }

  /** * adds a new element to the dictionary. If the key already exists, the existing value will be overwritten by the new value *@param {string} Key Key * of the new item to be added@param {*} Value Value * to which a new entry is to be added@return {boolean} Return true if the addition succeeded, false */ otherwise
  set(key, value) {
    if(key ! =null&& value ! =null) { // If neither key nor value is undefined or null
      const tableKey = this.toStrFn(key); // Get the string representing key
      this.table[tableKey] = new ValuePair(key, value); // Create a new key-value pair and assign it to the key property on the table object (tableKey)
      return true; // Add/overwrite success returns true
    }
    // Otherwise return false
    return false;
  }
  /** * finds a specific numeric value by taking a key value as an argument and returns *@param {*} Key Indicates the key to be obtained *@returns {*} Return valuePair or undefined */
  get(key) {
    const valuePair = this.table[this.toStrFn(key)]; // Get valuePair for the specified key
    return valuePair === null ? undefined : valuePair.value; // If the valuePair object exists, the value is returned, otherwise a undefined value is returned
  }
  /** ** Returns true if a key exists in the dictionary, false * otherwise@param {*} key
   * @returns {boolean}* /
  hasKey(key) {
    return this.table[this.toStrFn(key)] ! =null;
  }
  /** ** removes the data value * from the dictionary by using the key value as an argument@param {*} Key Indicates the key to be deleted *@returns {boolean} Check whether the file is successfully deleted */
  remove(key) {
    if (this.hasKey(key)) { // If the key exists
      delete this.table[this.toStrFn(key)]; // Delete the specified key from the table object
      return true; // Returns true on success
    }
    // Key does not exist, return false
    return false;
  }
  /** ** returns all values contained in the dictionary as an array *@returns {array}* /
  values() {
    return this.keyValues().map(valuePair= > valuePair.value);
  }
  /** ** returns * as an array of all the keys contained in the dictionary@returns {array}* /
  keys() {
    return this.keyValues().map(valuePair= > valuePair.key);
  }
  /** ** returns * for all [key, value] pairs in the dictionary@returns {array}* /
  keyValues() {
    return Object.values(this.table);
  }
  /** * iterates through all key-value pairs in the dictionary. CallbackFn takes two arguments: key and value. * This method can be aborted when the callback returns false (similar to the every method in the Array class). *@param {*} callbackFn* /
  forEach(callbackFn) {
    const valuePairs = this.keyValues(); // Get all the key-value pairs of the dictionary
    for (let i = 0; i < valuePairs.length; i++) { // walk through the dictionary
      const result = callbackFn(valuePairs[i].key, valuePairs[i].value); // Executes the callbackFn function passed to the forEach method as an argument
      if (result === false) { // If the callback returns false
        break; // Interrupts the forEach method, breaking the for loop iterating through valuePairs}}}/** * returns true if size equals zero, false otherwise@returns* /
  isEmpty() {
    return this.size() === 0;
  }
  /** ** returns the number of values the dictionary contains. Similar to the length property of an array *@returns {Number}* /
  size() {
    return Object.keys(this.table).length;
  }
  /** * delete all values */ in this dictionary
  clear() {
    this.table = {};
  }
  /** * outputs the dictionary value * as a string@returns {string}* /
  toString() {
    if (this.isEmpty()) { // Check if the dictionary is empty
      return ' '; // Returns an empty string if it is empty
    }
    // If the dictionary is not empty
    const valuePairs = this.keyValues(); // Get all the key-value pairs of the dictionary
    let objString = `${valuePairs[0].toString()}`; // Use the first element in the dictionary as the initial value of the string
    for (let i = 1; i < valuePairs.length; i++) { // Walk through all the key pairs of the dictionary
      // Add a comma (,) and the next element
      objString = `${objString}.${valuePairs[i].toString()}`;
    }
    // Prints the dictionary values as strings
    returnobjString; }}Copy the code

Experience the Dictionary class

Create an instance of the dictionary class and pass in three key-value pairs

const dictionary = new Dictionary();
dictionary.set('Gandalf'.'[email protected]');
dictionary.set('John'.'[email protected]');
dictionary.set('Tyrion'.'[email protected]');
Copy the code

Find if Gandalf is in the dictionary class instance

console.log(dictionary.hasKey('Gandalf')); / / return true
Copy the code

Gets the size of the dictionary class instance

console.log(dictionary.size()); / / return 3
Copy the code

Gets all keys of the dictionary class instance

console.log(dictionary.keys()); // ["Gandalf", "John", "Tyrion"] 
Copy the code

Gets all values of the dictionary class instance

console.log(dictionary.values()); // ["[email protected]", "[email protected]", "[email protected]"] 
Copy the code

Gets Tyrion in the dictionary class instance

console.log(dictionary.get('Tyrion')); // [email protected] 
Copy the code

Delete John from the dictionary class instance and see if it has been deleted.

dictionary.remove('John');
console.log(dictionary.keyValues()); // [{key: "Gandalf", value: "[email protected]"}, {key: "Tyrion", value:"[email protected]"}]
Copy the code

The forEach loop iterates over instances of dictionary classes

dictionary.forEach((k, v) = > {
  console.log('forEach: '.`key: ${k}, value: ${v}`);
});
 
// forEach: key: Gandalf, value: [email protected]
// forEach: key: Tyrion, value: [email protected]
Copy the code

HashTable (also known as HashTable, HashMap)

define

A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. The Object of JavaScript is essentially a set of key-value pairs (Hash structure), but traditionally only strings can be used as keys. Therefore, ES2015 brings Map class and WeakMap class, a weakened version of Map class.

Hash tables are also used in many practical scenarios:

  • Because it is an implementation of a dictionary, it can be used as an associative array
  • JavaScript objects are essentially collections of key-value pairs. Inside the JavaScript language, Hash tables are used to represent each Object.
  • For indexing in databases (such as MySQL, Microsoft SQL Server, Oracle, and so on), hash tables can be used to hold keys and references to records in a table

Why hash tables

A hash table is a data structure that is accessed directly based on key-value pairs.

That is, it accesses records by mapping key values to a location in the table to speed up lookups. The algorithm used in this mapping is called a hash algorithm, and the array of records is called a hash table.

Hash table a few basic concepts

Loading factor

Loading factor: A =n/m Where N is the number of keys and m is the size of the hash table.

The fill factor represents the degree to which the hash table is filled.

  • The larger the loading factor is, the more elements are filled, the more space utilization is high, and the greater the possibility of hash conflicts.
  • The smaller the loading factor is, the fewer elements are filled and the less possibility of hash conflicts occurs, but the space utilization is low.

The right balance needs to be found between the possibility of conflict and the utilization of space.

Hash algorithm

The purpose of the hash algorithm is to find a value in the data structure as quickly as possible. A hash algorithm is a function used to convert a piece of data to a fixed length number.

The two hashing algorithms used in the notes:

/** * The hash function loseloseHashCode * converts the passed key to a hash value and returns it, or returns * if the passed key is already a hash value@returns {number} Returns the hash value */
  loseloseHashCode = (key) = > {
    if (typeof key === 'number') { // Check if key is a number
      return key; // If yes, we return it directly
    }
    // Key is not a number
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 0; // Configure a hash variable to store the hash sum
    for (let i = 0; i < tableKey.length; i++) { / / traverse the key
      hash += tableKey.charCodeAt(i); // The ASCII value for each character retrieved from the ASCII table is added to the hash variable
    }
    return hash % 37; // To get a smaller value, we use the hash value and an arbitrary number as the remainder of the division (%) (to avoid the risk of the operands exceeding the maximum representation range of the numeric variable)
  }
  /** * hash function djb2HashCode */
  djb2HashCode(key) {
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 5381; // The initial hash variable with a prime value (most implementations use 5381)
    for (let i = 0; i < tableKey.length; i++) { // Then iterate over the key argument
    hash = (hash * 33) + tableKey.charCodeAt(i); // Multiply hash by 33 (used as a magic number -[magic number used directly in programming]) and add it to the ASCII value of the currently iterated character
    }
    return hash % 1013; // Finally, the remainder of the sum divided by another random prime number is returned
    // This is not the best hash function, but it is one of the most respected by the community.
    // There are also hash functions for numeric keys, a series of implementations can be found at http://t.cn/Eqg1yb0.
   } 
Copy the code

Hash conflict

If multiple elements have the same hash value calculated by the hash algorithm, the data with the same hash value will be overwritten if the data with the same hash value is saved. The purpose of using a data structure to store data is obviously not to lose it, but to store it all in some way. Therefore, there are several ways to handle hash conflicts routinely:

  • The separation of link
  • Linear probe
  • Double hashing

The first two hash conflict resolution methods are described later

Base hash

Code implementation

/** * In a Set, our focus is on each value itself. A Set stores elements as [value, value], whereas in a dictionary, it stores data as [key, value] pairs. Dictionaries are also called maps, symbol tables, or associative arrays. A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. The Object of JavaScript is essentially a set of key-value pairs (Hash structure), but traditionally only strings can be used as keys. Therefore, ES2015 brings Map class and WeakMap class, a weakened version of Map class. Collections, hash tables, and dictionaries are all data structures used to store unique (non-repeating) values. * /

/** * The ideal state for a hash table (a hash implementation of the Dictionary class) is a string as the key name, and the value can be any type (from primitive types such as numbers and strings to complex objects) * This method converts the passed key name to a string *@param {*} The key name passed to item *@returns {string}* /
export function defaultToString(item) {
  if (item === null) { // If key is null
    return 'NULL'; // Returns a NULL string
  }
  if (item === undefined) { // If key is undefined
    return 'UNDEFINED'; // Returns UNDEFINED as a string
  }
  if (typeof item === 'string' || item instanceof String) { // If it is a string, return it directly
    return `${item}`;
  }
  return item.toString(); // Otherwise return it as a string
}
/** * ValuePair class * To save information, we also need to save the original key and value *@class ValuePair* /
export class ValuePair {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  /** ** returns the key and value as strings *@returns {string}
   * @memberof ValuePair* /
  toString() {
    return ` [#The ${this.key}: The ${this.value}] `; }}// Implement Dictionary class based on ES2015 Map class
export class Dictionary {
  constructor(toStrFn = defaultToString) {
    this.toStrFn = toStrFn; // You can also pass in custom functions to specify how to convert a key to a string
    this.table = {}; // Use an Object instance to store dictionary elements
  }
  /** * The hash function * converts the passed key to a hash value and returns it, or returns * if the passed key is already a hash value@returns {number} Returns the hash value */
  loseloseHashCode = (key) = > {
    if (typeof key === 'number') { // Check if key is a number
      return key; // If yes, we return it directly
    }
    // Key is not a number
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 0; // Configure a hash variable to store the hash sum
    for (let i = 0; i < tableKey.length; i++) { / / traverse the key
      hash += tableKey.charCodeAt(i); // The ASCII value for each character retrieved from the ASCII table is added to the hash variable
    }
    return hash % 37; // To get a smaller value, we use the hash value and an arbitrary number as the remainder of the division (%) (to avoid the risk of the operands exceeding the maximum representation range of the numeric variable)
  }
  /** * Converts the passed key to a hash code using the loseloseHashCode method@returns {number} Returns the hash value */
  hashCode(key) {
    return this.loseloseHashCode(key);
  }

  /** * Add a new item to the hash table (it can also update the hash table) * The put method is logically similar to the set method in the Dictionary class. We could also name it set, but most programming languages use the PUT method in the HashTable data structure, so we follow the same naming pattern. *@param {string} Key Key * of the new item to be added@param {*} Value Value * to which a new entry is to be added@return {boolean} Return true if the addition succeeded, false */ otherwise
  put(key, value) {
    if(key ! =null&& value ! =null) { // Check whether key and value are valid
      const position = this.hashCode(key); // Get the hash code corresponding to the key value
      this.table[position] = new ValuePair(key, value); // Create a new key-value pair and assign it to the key property (position) on the table object.
      return true; // Add/overwrite success returns true
    }
    // Return false if it is illegal
    return false;
  }
  /** * returns the specified value * retrieved based on the key value@param {*} Key Indicates the key to be obtained *@returns {*} Return valuePair or undefined */
  get(key) {
    const valuePair = this.table[this.hashCode(key)]; // Get hashCode based on the specified key value, and valuePair based on the specified key value
    return valuePair === null ? undefined : valuePair.value; // If the valuePair object exists, the value is returned, otherwise a undefined value is returned
  }
  /** ** removes the value * from the hash table based on the key@param {*} Key Indicates the key to be deleted *@returns {boolean} Check whether the file is successfully deleted */
  remove(key) {
    // To remove a value from the HashTable, you first need to know where the value is
    const hash = this.hashCode(key); // Use the hashCode function to get the hash
    const valuePair = this.table[hash]; // valuePair is obtained at the hash location
    if(valuePair ! = =null) { // If valuePair is not null or undefined
      delete this.table[hash]; // Delete the specified hash from the table object
      return true; // Returns true on success
    }
    // Otherwise return false
    return false;
  }
  /** * get the entire hash table *@returns* /
  getTable() {
    return this.table;
  }
  /** * returns true if size equals zero, false otherwise@returns* /
  isEmpty() {
    return this.size() === 0;
  }
  /** ** returns the number of values the dictionary contains. Similar to the length property of an array *@returns {Number}* /
  size() {
    return Object.keys(this.table).length;
  }
  /** * delete all values */ in this dictionary
  clear() {
    this.table = {};
  }
  /** * outputs the dictionary value * as a string@returns {string}* /
  toString() {
    if (this.isEmpty()) { // Check if the dictionary is empty
      return ' '; // Returns an empty string if it is empty
    }
    // If the dictionary is not empty
    const keys = Object.keys(this.table); // Get all the keys of the hash table
    let objString = ` {${keys[0]}= >The ${this.table[keys[0]].toString()}} `; // Use the first element of the hash table as the initial value of the string
    for (let i = 1; i < keys.length; i++) { // Walk through all the key pairs of the dictionary
      // Add the next element to the string
      objString = `${objString}, {${keys[i]}= >The ${this.table[keys[i]].toString()}} `;
    }
    // Prints the dictionary values as strings
    returnobjString; }}Copy the code

Use the base hash table instance

Create a hash table and pass in three key-value pairs

const hash = new HashTable();
hash.put('Gandalf'.'[email protected]');
hash.put('John'.'[email protected]');
hash.put('Tyrion'.'[email protected]');
console.log(hash.hashCode('Gandalf') + ' - Gandalf');
console.log(hash.hashCode('John') + ' - John');
console.log(hash.hashCode('Tyrion') + ' - Tyrion');

/** 19 - Gandalf 29 - John 16 - Tyrion */
Copy the code

A partial method to test a hash table

console.log(hash.get('Gandalf')); // [email protected]
console.log(hash.get('Loiane')); // undefined 

hash.remove('Gandalf');
console.log(hash.get('Gandalf')); // undefined
Copy the code

Tests hash conflicts for hash tables

/** * sometimes, some keys will have the same hash value. When different values correspond to the same position in a hash table, we call it a conflict. For example, let's look at the following code to see what the output looks like. * /
const hash1 = new HashTable();
hash.put('Ygritte'.'[email protected]');
hash.put('Jonathan'.'[email protected]');
hash.put('Jamie'.'[email protected]');
hash.put('Jack'.'[email protected]');
hash.put('Jasmine'.'[email protected]');
hash.put('Jake'.'[email protected]');
hash.put('Nathan'.'[email protected]');
hash.put('Athelstan'.'[email protected]');
hash.put('Sue'.'[email protected]');
hash.put('Aethelwulf'.'[email protected]');
hash.put('Sargeras'.'[email protected]');

/** 4 - Ygritte 5 - Jonathan 5 - Jamie 7 - Jack 8 - Jasmine 9 - Jake 10 - Nathan 7 - Athelstan 5 - Sue 5 - Aethelwulf 10  - Sargeras */

// Actual output

console.log(hashTable.toString())

/**
{4 => [#Ygritte: ygritte@email.com]}
{5 => [#Aethelwulf: aethelwulf@email.com]}
{7 => [#Athelstan: athelstan@email.com]}
{8 => [#Jasmine: jasmine@email.com]}
{9 => [#Jake: jake@email.com]}
{10 => [#Sargeras: sargeras@email.com]}
 */

/** Jonathan, Jamie, Sue and Aethelwulf have the same hash value, i.e. 5. Since Aethelwulf is the last to be added, it will be the element that occupies position 5 in the HashTable instance. First Jonathan will take the spot, then Jamie will cover it, Sue will cover it again, and finally Aethelwulf will cover it again. The same is true for other elements in conflict. The purpose of using a data structure to store data is obviously not to lose it, but to store it all in some way. So deal with it when it happens. There are several ways to handle collisions: split chaining, linear probing, and double hashing. * /
Copy the code

Use split chaining to resolve hash conflicts

introduce

Create a linked list for each position in the hash table and store the elements in it. It is the simplest way to resolve hash conflicts, but requires additional storage space beyond the hash instance.

Code implementation

import {
  LinkedList
} from '/LinkedList/app.js';
/** * In a Set, our focus is on each value itself. A Set stores elements as [value, value], whereas in a dictionary, it stores data as [key, value] pairs. Dictionaries are also called maps, symbol tables, or associative arrays. A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. The Object of JavaScript is essentially a set of key-value pairs (Hash structure), but traditionally only strings can be used as keys. Therefore, ES2015 brings Map class and WeakMap class, a weakened version of Map class. Collections, hash tables, and dictionaries are all data structures used to store unique (non-repeating) values. * /

/** * The ideal state for a hash table (a hash implementation of the Dictionary class) is a string as the key name, and the value can be any type (from primitive types such as numbers and strings to complex objects) * This method converts the passed key name to a string *@param {*} The key name passed to item *@returns {string}* /
export function defaultToString(item) {
  if (item === null) { // If key is null
    return 'NULL'; // Returns a NULL string
  }
  if (item === undefined) { // If key is undefined
    return 'UNDEFINED'; // Returns UNDEFINED as a string
  }
  if (typeof item === 'string' || item instanceof String) { // If it is a string, return it directly
    return `${item}`;
  }
  return item.toString(); // Otherwise return it as a string
}
/** * ValuePair class * To save information, we also need to save the original key and value *@class ValuePair* /
export class ValuePair {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  /** ** returns the key and value as strings *@returns {string}
   * @memberof ValuePair* /
  toString() {
    return ` [#The ${this.key}: The ${this.value}] `; }}// Implement Dictionary class based on ES2015 Map class
export class HashTableSeparateChaining {
  constructor(toStrFn = defaultToString) {
    this.toStrFn = toStrFn; // You can also pass in custom functions to specify how to convert a key to a string
    this.table = {}; // Use an Object instance to store dictionary elements
  }
  /** * The hash function * converts the passed key to a hash value and returns it, or returns * if the passed key is already a hash value@returns {number} Returns the hash value */
  loseloseHashCode = (key) = > {
    if (typeof key === 'number') { // Check if key is a number
      return key; // If yes, we return it directly
    }
    // Key is not a number
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 0; // Configure a hash variable to store the hash sum
    for (let i = 0; i < tableKey.length; i++) { / / traverse the key
      hash += tableKey.charCodeAt(i); // The ASCII value for each character retrieved from the ASCII table is added to the hash variable
    }
    return hash % 37; // To get a smaller value, we use the hash value and an arbitrary number as the remainder of the division (%) (to avoid the risk of the operands exceeding the maximum representation range of the numeric variable)
  }
  /** * Converts the passed key to a hash code using the loseloseHashCode method@returns {number} Returns the hash value */
  hashCode(key) {
    return this.loseloseHashCode(key);
  }

  /** * Add a new item to the hash table * The put method is logically similar to the set method in the Dictionary class. We could also name it set, but most programming languages use the PUT method in the HashTable data structure, so we follow the same naming pattern. *@param {string} Key Key * of the new item to be added@param {*} Value Value * to which a new entry is to be added@return {boolean} Return true if the addition succeeded, false */ otherwise
  put(key, value) {
    if(key ! =null&& value ! =null) { // Check whether key and value are valid
      const position = this.hashCode(key); // Get the hash code corresponding to the key value
      if (this.table[position] === null) { // Verify that the location to add the new element is already occupied
        this.table[position] = new LinkedList(); // If it is the first time we add an element to that location, we will initialize an instance of the list at that location
      }
      this.table[position].push(new ValuePair(key, value)); // Add a ValuePair instance (key and value) to the list
      return true; // Returns true on success
    }
    // Return false if it is illegal
    return false;
  }
  /** * returns the specified value * retrieved based on the key value@param {*} Key Indicates the key to be obtained *@returns {*} Return valuePair or undefined */
  get(key) {
    const position = this.hashCode(key); // Get the hash code corresponding to the key value
    const linkedList = this.table[position]; // Get the list instance corresponding to the key value position
    // If the corresponding list instance exists and it is not an empty list
    if(linkedList ! = =null && !linkedList.isEmpty()) {
      let current = linkedList.getHead(); // Get a reference to the list header
      while(current ! =null) { // Iterate from the top of the list to the bottom until the end. Current. Next will be null, ending the iteration
        if (current.element.key === key) { // The Element property holds an instance of ValuePair if the key is the same as the one passed in
          return current.element.value; // Find the value to retrieve, return the value of value
        }
        // If not, iterate over the list and visit the next nodecurrent = current.next; }}// If not, undefined is returned to indicate that the value was not found in the HashTable instance
    return undefined;
    /** * Another way to implement the algorithm is as follows: In addition to searching for the key inside the GET method, you can instantiate the LinkedList in the PUT method, passing a custom equalsFn to the constructor of the LinkedList and using it only to compare elements' key attributes (that is, ValuePair instances). Remember that by default LinkedList uses the === operator to compare its element instances, that is, references to ValuePair instances. In this case, in the get method, we use the indexOf method to search for the target key, and if a position greater than or equal to zero is returned, the element exists in the list. With this location, we can use the getElementAt method to get ValuePair instances from the linked list. * /
  }
  /** ** removes the value * from the hash table based on the key@param {*} Key Indicates the key to be deleted *@returns {boolean} Check whether the file is successfully deleted */
  remove(key) {
    const position = this.hashCode(key); // Get the hash code corresponding to the key value
    const linkedList = this.table[position]; // Get the list instance corresponding to the key value position
    // If the corresponding list instance exists and it is not an empty list
    if(linkedList ! =null && !linkedList.isEmpty()) {
      let current = linkedList.getHead(); // Get a reference to the list header
      while(current ! =null) { // Iterate from the top of the list to the bottom until the end. Current. Next will be null, ending the iteration
        if (current.element.key === key) { // The Element property holds an instance of ValuePair if the key is the same as the one passed in
          linkedList.remove(current.element); // Use the remove method to remove it from the list
          if (linkedList.isEmpty()) { // If the list is empty (there are no more elements in the list)
            delete this.table[position]; // Use the delete operator to remove this position from the hash table, so that when searching for an element, the position can be skipped
          }
          return true; // Returns true to indicate that the element has been removed
        }
        current = current.next; // If it is not the element we are looking for, we continue iterating over the next element as in the get method}}return false; // Returning false indicates that the element does not exist in the hash table
  }
  /** * get the entire hash table *@returns* /
  getTable() {
    return this.table;
  }
  /** * returns true if size equals zero, false otherwise@returns* /
  isEmpty() {
    return this.size() === 0;
  }
  /** ** returns the number of values the dictionary contains. Similar to the length property of an array *@returns {Number}* /
  size() {
    return Object.keys(this.table).length;
  }
  /** * delete all values */ in this dictionary
  clear() {
    this.table = {};
  }
  /** * outputs the dictionary value * as a string@returns {string}* /
  toString() {
    if (this.isEmpty()) { // Check if the dictionary is empty
      return ' '; // Returns an empty string if it is empty
    }
    // If the dictionary is not empty
    const keys = Object.keys(this.table); // Get all the keys of the hash table
    let objString = ` {${keys[0]}= >The ${this.table[keys[0]].toString()}} `; // Use the first element of the hash table as the initial value of the string
    for (let i = 1; i < keys.length; i++) { // Walk through all the key pairs of the dictionary
      // Add the next element to the string
      objString = `${objString}, {${keys[i]}= >The ${this.table[keys[i]].toString()}} `;
    }
    // Prints the dictionary values as strings
    returnobjString; }}Copy the code

Use linear probe to resolve hash conflicts

introduce

Linear probe is called linear because it handles collisions by storing elements directly in a table, rather than in a separate data structure. When you want to add a new element to a position in the table, if the position of the hash value is already occupied, you try to find the next position of the hash value. If the position of the hash value is still occupied, you try to find the next position of the hash value, and so on until you find a free position and fill the new value.

Using linear probe to remove elements will result in an exception in the hash table, so the removed elements need to be handled in one of the following ways:

  • Soft delete
  • After deletion, check and move the key-value pair elements affected by the deletion side effect

Linear probe (soft deletion method)

introduce

Using a delete flag to indicate that the key-value pair has been deleted does not actually remove it from the hash table, but as the number of deletions increases, the hash table becomes redundant with many useless soft delete flags, gradually reducing the efficiency of the hash table.

Code implementation
/** * In a Set, our focus is on each value itself. A Set stores elements as [value, value], whereas in a dictionary, it stores data as [key, value] pairs. Dictionaries are also called maps, symbol tables, or associative arrays. A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. The Object of JavaScript is essentially a set of key-value pairs (Hash structure), but traditionally only strings can be used as keys. Therefore, ES2015 brings Map class and WeakMap class, a weakened version of Map class. Collections, hash tables, and dictionaries are all data structures used to store unique (non-repeating) values. * /

/** * The ideal state for a hash table (a hash implementation of the Dictionary class) is a string as the key name, and the value can be any type (from primitive types such as numbers and strings to complex objects) * This method converts the passed key name to a string *@param {*} The key name passed to item *@returns {string}* /
export function defaultToString(item) {
  if (item === null) { // If key is null
    return 'NULL'; // Returns a NULL string
  }
  if (item === undefined) { // If key is undefined
    return 'UNDEFINED'; // Returns UNDEFINED as a string
  }
  if (typeof item === 'string' || item instanceof String) { // If it is a string, return it directly
    return `${item}`;
  }
  return item.toString(); // Otherwise return it as a string
}
/** * ValuePair class * To save information, we also need to save the original key and value *@class ValuePair* /
export class ValuePair {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  /** ** returns the key and value as strings *@returns {string}
   * @memberof ValuePair* /
  toString() {
    return ` [#The ${this.key}: The ${this.value}] `; }}// Implement Dictionary class based on ES2015 Map class
export class HashTableLinearProbingLazy {
  constructor(toStrFn = defaultToString) {
    this.toStrFn = toStrFn; // You can also pass in custom functions to specify how to convert a key to a string
    this.table = {}; // Use an Object instance to store dictionary elements
  }
  /** * The hash function * converts the passed key to a hash value and returns it, or returns * if the passed key is already a hash value@returns {number} Returns the hash value */
  loseloseHashCode = (key) = > {
    if (typeof key === 'number') { // Check if key is a number
      return key; // If yes, we return it directly
    }
    // Key is not a number
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 0; // Configure a hash variable to store the hash sum
    for (let i = 0; i < tableKey.length; i++) { / / traverse the key
      hash += tableKey.charCodeAt(i); // The ASCII value for each character retrieved from the ASCII table is added to the hash variable
    }
    return hash % 37; // To get a smaller value, we use the hash value and an arbitrary number as the remainder of the division (%) (to avoid the risk of the operands exceeding the maximum representation range of the numeric variable)
  }
  /** * hash function djb2HashCode */
  djb2HashCode(key) {
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 5381; // The initial hash variable with a prime value (most implementations use 5381)
    for (let i = 0; i < tableKey.length; i++) { // Then iterate over the key argument
    hash = (hash * 33) + tableKey.charCodeAt(i); // Multiply hash by 33 (used as a magic number -[magic number used directly in programming]) and add it to the ASCII value of the currently iterated character
    }
    return hash % 1013; // Finally, the remainder of the sum divided by another random prime number is returned
    // This is not the best hash function, but it is one of the most respected by the community.
    // There are also hash functions for numeric keys, a series of implementations can be found at http://t.cn/Eqg1yb0.
   } 
  /** * Converts the passed key to a hash code using the loseloseHashCode method@returns {number} Returns the hash value */
  hashCode(key) {
    return this.loseloseHashCode(key);
  }

  /** * Add a new item to the hash table * The put method is logically similar to the set method in the Dictionary class. We could also name it set, but most programming languages use the PUT method in the HashTable data structure, so we follow the same naming pattern. *@param {string} Key Key * of the new item to be added@param {*} Value Value * to which a new entry is to be added@return {boolean} Return true if the addition succeeded, false */ otherwise
  put(key, value) {
    if(key ! =null&& value ! =null) { // Check whether key and value are valid
      const position = this.hashCode(key); // Get the hash code corresponding to the key value
      if (this.table[position] == null| | -this.table[position] ! =null && this.table[position].isDeleted)) {
        // The position where the new element is to be added is not occupied or the element at that position is deleted
        this.table[position] = new ValuePair(key, value); // Add a new element at this location
      } else {
        // If the position is already occupied, we need to find the next position that is not occupied (position is undefined or null)
        let index = position + 1; // Next to the current location
        while (this.table[index] ! = =null&&!this.table[position].isDeleted) { // Verify that the location is occupied or deleted
          index++; // If it is occupied and the value is not in the deleted state, continue incrementing index until an unoccupied position is found
        }
        // After the appropriate location is found, assign the value to that location
        this.table[index] = new ValuePair(key, value);
      }
      return true; // Returns true on success
    }
    // Return false if it is illegal
    return false;
  }
  /** * returns the specified value * retrieved based on the key value@param {*} Key Indicates the key to be obtained *@returns {*} Return valuePair or undefined */
  get(key) {
    const position = this.hashCode(key); // Get the hash code corresponding to the key value
    if (this.table[position] ! =null) { // If the key exists
      if (this.table[position].key === key  && !this.table[position].isDeleted) { // The key in the current position is equal to the key passed in and is not deleted
        return this.table[position].value; // Directly returns the hash table item at that location
      }
      let index = position + 1; // If not, proceed to the next position in the hash table
      while (this.table[index] ! =null && (this.table[index].key ! == key ||this.table[index].isDeleted)) { // Elements in the hash table are searched in ascending order of position until we find the element we are looking for or an empty position
        if (this.table[index].key === key && this.table[index].isDeleted) { // If the key found matches the one we passed in, but the state is deleted
          return undefined; // Indicates that the value you are looking for has been removed from the hash table, so you can return undefined
        }
        index++; // Continue incrementing
      }
      When exiting from the while loop, verify that the element's key is the one we are looking for
      if (
        this.table[index] ! =null
        && this.table[index].key === key
        && !this.table[index].isDeleted
      ) {
        return this.table[position].value; // If so, return its value}}// If the key does not exist, the value is not in the hash table, so undefined can be returned
    return undefined;
  }
  /** ** removes the value * from the hash table based on the key@param {*} Key Indicates the key to be deleted *@returns {boolean} Check whether the file is successfully deleted */
  remove(key) {
    const position = this.hashCode(key); // Get the hash code corresponding to the key value
    if (this.table[position] ! =null) { // If the key exists
      if (this.table[position].key === key  && !this.table[position].isDeleted) { // The key in the current position is equal to the key passed in and is not deleted
        this.table[position].isDeleted = true; // delete the hash item at that location
        return true; // Successful deletion is displayed
      }
      let index = position + 1; // If not, proceed to the next position in the hash table
      while (
        this.table[index] ! =null
        && (this.table[index].key ! == key ||this.table[index].isDeleted)
      ) { // Elements in the hash table are searched in ascending order of position until we find the element we are looking for or an empty position
        index++; // Continue incrementing
      }
      When exiting from the while loop, verify that the element's key is the one we are looking for
      if (
        this.table[index] ! =null
        && this.table[index].key === key
        && !this.table[index].isDeleted
      ) {
        this.table[position].isDeleted = true; // delete the hash item at that location
        return true; // Successful deletion is displayed}}return false;
  }
  / * * * * *@param {*} Key Indicates the deleted key *@param {*} RemovedPosition Position where the key is deleted */
  verifyRemoveSideEffect(key, removedPosition) {
    const hash = this.hashCode(key); // Get the hash value of the deleted key
    let index = removedPosition + 1; // Iterate over the hash table from the next position
    while (this.table[index] ! =null) { // End the loop until an empty position is found. (At this point, all elements that meet the criteria have been processed into the appropriate position, no need to move)
      const posHash = this.hashCode(this.table[index].key); // Computes the hash value of the element at the current position
      if (posHash <= hash || posHash <= removedPosition) { // If the hash value of the current element is less than or equal to the original hash value or the hash value of the current element is less than or equal to the removedPosition (i.e. the hash value of the last removed key)
        this.table[removedPosition] = this.table[index]; // indicates that we need to move the current element to removedPosition
        delete this.table[index]; // Once the move is complete, we can delete the current element (since it has already been copied to the removedPosition location)
        removedPosition = index; // You also need to update the removedPosition to the current index} index++; }}/** * get the entire hash table *@returns* /
  getTable() {
    return this.table;
  }
  /** * returns true if size equals zero, false otherwise@returns* /
  isEmpty() {
    return this.size() === 0;
  }
  /** ** returns the number of values the dictionary contains. Similar to the length property of an array *@returns {Number}* /
  size() {
    return Object.keys(this.table).length;
  }
  /** * delete all values */ in this dictionary
  clear() {
    this.table = {};
  }
  /** * outputs the dictionary value * as a string@returns {string}* /
  toString() {
    if (this.isEmpty()) { // Check if the dictionary is empty
      return ' '; // Returns an empty string if it is empty
    }
    // If the dictionary is not empty
    const keys = Object.keys(this.table); // Get all the keys of the hash table
    let objString = ` {${keys[0]}= >The ${this.table[keys[0]].toString()}} `; // Use the first element of the hash table as the initial value of the string
    for (let i = 1; i < keys.length; i++) { // Walk through all the key pairs of the dictionary
      // Add the next element to the string
      objString = `${objString}, {${keys[i]}= >The ${this.table[keys[i]].toString()}} `;
    }
    // Prints the dictionary values as strings
    returnobjString; }}Copy the code

Linear probe (move delete side effects of key pair elements)

introduce

To actually remove an element from a hash table, and then determine whether any of its subsequent elements need to be moved to the empty space after the deletion, as the size of the hash table increases, there is a cost of moving elements similar to that of an array.

Code implementation
import {
  LinkedList
} from '/LinkedList/app.js';
/** * In a Set, our focus is on each value itself. A Set stores elements as [value, value], whereas in a dictionary, it stores data as [key, value] pairs. Dictionaries are also called maps, symbol tables, or associative arrays. A HashTable (also known as a HashTable class or a HashMap class) is a HashTable implementation of a dictionary. The Object of JavaScript is essentially a set of key-value pairs (Hash structure), but traditionally only strings can be used as keys. Therefore, ES2015 brings Map class and WeakMap class, a weakened version of Map class. Collections, hash tables, and dictionaries are all data structures used to store unique (non-repeating) values. * /

/** * The ideal state for a hash table (a hash implementation of the Dictionary class) is a string as the key name, and the value can be any type (from primitive types such as numbers and strings to complex objects) * This method converts the passed key name to a string *@param {*} The key name passed to item *@returns {string}* /
export function defaultToString(item) {
  if (item === null) { // If key is null
    return 'NULL'; // Returns a NULL string
  }
  if (item === undefined) { // If key is undefined
    return 'UNDEFINED'; // Returns UNDEFINED as a string
  }
  if (typeof item === 'string' || item instanceof String) { // If it is a string, return it directly
    return `${item}`;
  }
  return item.toString(); // Otherwise return it as a string
}
/** * ValuePair class * To save information, we also need to save the original key and value *@class ValuePair* /
export class ValuePair {
  constructor(key, value) {
    this.key = key;
    this.value = value;
  }
  /** ** returns the key and value as strings *@returns {string}
   * @memberof ValuePair* /
  toString() {
    return ` [#The ${this.key}: The ${this.value}] `; }}// Implement Dictionary class based on ES2015 Map class
export class HashTableSeparateChaining {
  constructor(toStrFn = defaultToString) {
    this.toStrFn = toStrFn; // You can also pass in custom functions to specify how to convert a key to a string
    this.table = {}; // Use an Object instance to store dictionary elements
  }
  /** * The hash function * converts the passed key to a hash value and returns it, or returns * if the passed key is already a hash value@returns {number} Returns the hash value */
  loseloseHashCode = (key) = > {
    if (typeof key === 'number') { // Check if key is a number
      return key; // If yes, we return it directly
    }
    // Key is not a number
    const tableKey = this.toStrFn(key); // Convert the key to a string
    let hash = 0; // Configure a hash variable to store the hash sum
    for (let i = 0; i < tableKey.length; i++) { / / traverse the key
      hash += tableKey.charCodeAt(i); // The ASCII value for each character retrieved from the ASCII table is added to the hash variable
    }
    return hash % 37; // To get a smaller value, we use the hash value and an arbitrary number as the remainder of the division (%) (to avoid the risk of the operands exceeding the maximum representation range of the numeric variable)
  }
  /** * Converts the passed key to a hash code using the loseloseHashCode method@returns {number} Returns the hash value */
  hashCode(key) {
    return this.loseloseHashCode(key);
  }

  /** * Add a new item to the hash table * The put method is logically similar to the set method in the Dictionary class. We could also name it set, but most programming languages use the PUT method in the HashTable data structure, so we follow the same naming pattern. *@param {string} Key Key * of the new item to be added@param {*} Value Value * to which a new entry is to be added@return {boolean} Return true if the addition succeeded, false */ otherwise
  put(key, value) {
    if(key ! =null&& value ! =null) { // Check whether key and value are valid
      const position = this.hashCode(key); // Get the hash code corresponding to the key value
      if (this.table[position] === null) { // Verify that the location to add the new element is already occupied
        this.table[position] = new LinkedList(); // If it is the first time we add an element to that location, we will initialize an instance of the list at that location
      }
      this.table[position].push(new ValuePair(key, value)); // Add a ValuePair instance (key and value) to the list
      return true; // Returns true on success
    }
    // Return false if it is illegal
    return false;
  }
  /** * returns the specified value * retrieved based on the key value@param {*} Key Indicates the key to be obtained *@returns {*} Return valuePair or undefined */
  get(key) {
    const position = this.hashCode(key); // Get the hash code corresponding to the key value
    const linkedList = this.table[position]; // Get the list instance corresponding to the key value position
    // If the corresponding list instance exists and it is not an empty list
    if(linkedList ! = =null && !linkedList.isEmpty()) {
      let current = linkedList.getHead(); // Get a reference to the list header
      while(current ! =null) { // Iterate from the top of the list to the bottom until the end. Current. Next will be null, ending the iteration
        if (current.element.key === key) { // The Element property holds an instance of ValuePair if the key is the same as the one passed in
          return current.element.value; // Find the value to retrieve, return the value of value
        }
        // If not, iterate over the list and visit the next nodecurrent = current.next; }}// If not, undefined is returned to indicate that the value was not found in the HashTable instance
    return undefined;
    /** * Another way to implement the algorithm is as follows: In addition to searching for the key inside the GET method, you can instantiate the LinkedList in the PUT method, passing a custom equalsFn to the constructor of the LinkedList and using it only to compare elements' key attributes (that is, ValuePair instances). Remember that by default LinkedList uses the === operator to compare its element instances, that is, references to ValuePair instances. In this case, in the get method, we use the indexOf method to search for the target key, and if a position greater than or equal to zero is returned, the element exists in the list. With this location, we can use the getElementAt method to get ValuePair instances from the linked list. * /
  }
  /** ** removes the value * from the hash table based on the key@param {*} Key Indicates the key to be deleted *@returns {boolean} Check whether the file is successfully deleted */
  remove(key) {
    const position = this.hashCode(key); // Get the hash code corresponding to the key value
    const linkedList = this.table[position]; // Get the list instance corresponding to the key value position
    // If the corresponding list instance exists and it is not an empty list
    if(linkedList ! =null && !linkedList.isEmpty()) {
      let current = linkedList.getHead(); // Get a reference to the list header
      while(current ! =null) { // Iterate from the top of the list to the bottom until the end. Current. Next will be null, ending the iteration
        if (current.element.key === key) { // The Element property holds an instance of ValuePair if the key is the same as the one passed in
          linkedList.remove(current.element); // Use the remove method to remove it from the list
          if (linkedList.isEmpty()) { // If the list is empty (there are no more elements in the list)
            delete this.table[position]; // Use the delete operator to remove this position from the hash table, so that when searching for an element, the position can be skipped
          }
          return true; // Returns true to indicate that the element has been removed
        }
        current = current.next; // If it is not the element we are looking for, we continue iterating over the next element as in the get method}}return false; // Returning false indicates that the element does not exist in the hash table
  }
  /** * get the entire hash table *@returns* /
  getTable() {
    return this.table;
  }
  /** * returns true if size equals zero, false otherwise@returns* /
  isEmpty() {
    return this.size() === 0;
  }
  /** ** returns the number of values the dictionary contains. Similar to the length property of an array *@returns {Number}* /
  size() {
    return Object.keys(this.table).length;
  }
  /** * delete all values */ in this dictionary
  clear() {
    this.table = {};
  }
  /** * outputs the dictionary value * as a string@returns {string}* /
  toString() {
    if (this.isEmpty()) { // Check if the dictionary is empty
      return ' '; // Returns an empty string if it is empty
    }
    // If the dictionary is not empty
    const keys = Object.keys(this.table); // Get all the keys of the hash table
    let objString = ` {${keys[0]}= >The ${this.table[keys[0]].toString()}} `; // Use the first element of the hash table as the initial value of the string
    for (let i = 1; i < keys.length; i++) { // Walk through all the key pairs of the dictionary
      // Add the next element to the string
      objString = `${objString}, {${keys[i]}= >The ${this.table[keys[i]].toString()}} `;
    }
    // Prints the dictionary values as strings
    returnobjString; }}Copy the code

ES6 Map class

ECMAScript 2015 adds the Map class. We can develop our Dictionary class based on ES2015’s Map class.

const map = new Map(a); map.set('Gandalf'.'[email protected]');
map.set('John'.'[email protected]');
map.set('Tyrion'.'[email protected]');
console.log(map.has('Gandalf')); // true
console.log(map.size); / / 3
console.log(map.keys()); // output {"Gandalf", "John", "Tyrion"}
console.log(map.values()); // Output {"[email protected]", "[email protected]",
"[email protected]"}
console.log(map.get('Tyrion')); // [email protected]
map.delete('John'); // Delete the map element
map.clear(); // Resets the map data structure, as we did in the Dictionary class.
Copy the code

Unlike the dictionary classes we developed earlier, ES2015 Map’s values and keys methods both return iterators (as mentioned in Chapter 3) rather than arrays of values or keys.

ES6 WeakMap class and WeakSet class

WeakSet and WeakMap are weaker versions of Set and Map

The main differences are:

  • WeakSet or WeakMap class does not have entries, keys, values and other methods;
  • You can only use objects as keys.
const map = new WeakMap(a);const ob1 = { name: 'Gandalf' };
const ob2 = { name: 'John' };
const ob3 = { name: 'Tyrion' };
map.set(ob1, '[email protected]');
map.set(ob2, '[email protected]');
map.set(ob3, '[email protected]');
console.log(map.has(ob1)); // true
console.log(map.get(ob3)); // [email protected]
map.delete(ob2);  
Copy the code

Because WeakSet or WeakMap classes do not have strongly referenced keys, it can improve the performance of JavaScript garbage collection, and because there are no iterator methods, it is impossible to extract values without knowing the keys.

(You can use WeakMap class to encapsulate private properties of ES6 class)