Today we will start with the second data type, the chained table, and again we will implement a chained table in the most primitive way, by requesting memory and managing it ourselves.
01, 01, Definitions
What is a chained table? Chained table in the physical storage structure for non-sequential and non-continuous, so the physical storage location of the data elements of the chained table is random, dynamic allocation; and in the logical structure of the performance of the linear structure of the characteristics of the structure, that is, the elements of an element one after another element strung together like a line .
nodal: One of the chain table elements is also called a node, a node mainly contains a data field and a pointer field, where the data field mainly stores the data elements, while the pointer field mainly stores the address of the next node storage location.
header pointer: A plain pointer to the first node position of a linked table, and always pointing to the first node position, to facilitate later use of the linked table.
head node: Usually indicates the first node of a chain table, and the data field inside the node is empty, so it is also called null node, and its role is mainly used to solve some special problems, so it can also be omitted.
primordial node: Since the data field of the head node is empty, the first node of the chain table whose data field is not empty is called the primitive node, which is just a name and has no real meaning.
0202. Classification
There are two ways to categorize chained tables, one can be classified as static and dynamic chained tables, and the other can be classified as unidirectional, bidirectional and circular chained tables.
A single linked table has only one direction, and each node contains a data field and a pointer field to the next node.
A bi-directional linked table has two directions, i.e., each node contains a data field as well as a pointer field pointing to both the previous node and the next node.
A circular chain table means that the first and last nodes of the chain table are connected, i.e., the pointer field of the last node points to the first node. Circular chain table is also divided into unidirectional circular chain table and bidirectional circular chain table, the principle is the same.
03, 03. Realization
Together we will use the most primitive way to complete the implementation of a chained table by requesting memory space and maintaining it ourselves.
1. Definition of ADT
Let's start by defining the ADT (single linked table) of a chained table.
ADT LinkedList{
Data object: D is a non-empty collection of elements, D = {a1, a2, ... , an}, where ai denotes an element i.e. a node, a node stores data and a pointer to the next node.
Data Relationships: Nodes in D are connected by pointers and each node contains a pointer to the next node.
Basic operations: [
Init(n) : Initialize an empty linked table, i.e. declare a head pointer, or a head node if necessary.
Length: return the length of the chain table.
HeadNode: return the head node.
Find(v): return the node corresponding to data field v.
Update(n,v): update the data domain of node n.
InsertAfter(n,v): add a new node with data field v after node.
Remove(n): remove node n.
Destroy(): destroy the chain table.
]
}
With the chained table ADT defined, we can now begin to implement a chained table with a data field of type string on our own.
2. Define the class
First we need to define the node, which contains two fields one is to store data, one is to store the pointer, the code is as follows.
public struct MyselfLinkedListNode
{
//data domain
public string Data { get; set; }
//pointer field (computing)
public IntPtr Next { get; set; }
}
Then define the linked list implementation class MyselfLinkedList, which is used to implement the operations related to linked lists.
Because we manage memory directly, we need a pointer field that maintains memory;
Because we are getting the length of the chain table directly, we need a field to store the length of the chain table;
So our MyselfLinkedList class initially looks like this:
public sealed class MyselfLinkedList : IDisposable
{
//Request a pointer to the start of memory
private IntPtr _head;
//Length of chain table
private int _length;
}
3、Initialization Init
Initializing a structure does a few things.
a. Allocate memory space;
b. What header pointer;
c.Create the head node;
d.Maintain the chain list length attribute;
The specific implementation code is as follows:
// Initialize the linked list, declare the header pointer, and create the header node
public MyselfLinkedListNode Init()
{
// Calculate the size of the node
var size = (typeof(MyselfLinkedListNode));
// Allocate the specified number of bytes of memory space
_head = (size);
// Create the head node
var node = new MyselfLinkedListNode
{
Data = null,
Next =
};
// Write the node instance to the allocated memory
(node, _head, false);
// Add 1 to the length of the chain table
_length++;
// Return the head node
return node.
}
4. Get the length of the chain table Length
It's easier to just return the private field for the length of the chain table.
//Length of chain table
public int Length
{
get
{
return _length;
}
}
5、Get the head node HeadNode
Getting the header node is mainly to facilitate data processing, you can read the memory address directly through the header pointer to get. The specific code is as follows:
//head node
public MyselfLinkedListNode? HeadNode
{
get
{
if (_head == )
{
return null;
}
return GetNode(_head);
}
}
//Get node
private MyselfLinkedListNode GetNode(IntPtr pointer)
{
// Example of reading from allocated memory
return <MyselfLinkedListNode>(pointer);
}
Similarly we can define a tail node attribute that can be easily used, the principles are similar, so I won't go into it here.
6, in the specified node after the insertion of nodes InsertAfter
Through the previous understanding of the structure of the chain table, to add a new node between two more nodes, just need to cut the line between the two, that is, the former node's pointer field needs to be redirected to the new node, and the new node's pointer field to point to the latter node, and other remain unchanged, as follows:
Business logic is clear, let's sort out the code logic, to realize this function we need a few steps:
a.Get a pointer to the specified node;
b.Create a new node;
c. Realign the specified node and the new node pointer field;
d. Update the specified node and the new node pointer adjusted data into memory;
e.Update the chain table length attribute;
The specific realizations are as follows:
// Insert a new node after the specified node
public MyselfLinkedListNode InsertAfter(MyselfLinkedListNode node, string value)
{
//Get the corresponding pointer to the node you want to take
var pointer = GetPointer(node); //If the pointer is not null, then the value is not null.
//If the pointer is not null, then process it.
if (pointer ! = )
{
// Create a node with the new value
var (newPointer, newNode) = CreateNode(value);
// Point the next node pointer of the new node to the next node of the specified node
= ;
// point the next node pointer of the specified node to the new node
= newPointer;
// Update the modified node
(newNode, newPointer, false); ;/Update the modified node.
(node, pointer, false);
// add 1 to the length of the chain table
_length++;
return newNode.
}
return default; }
}
// Get the pointer to the node
private IntPtr GetPointer(MyselfLinkedListNode node)
{
// Start looking at the head pointer
var currentPointer = _head;
// Stop searching if current pointer is null
while (currentPointer ! = )
{
//Get the node corresponding to the current pointer
var currentNode = GetNode(currentPointer);
//If the current node's data field and pointer field are the same as the node to be found, then the current node pointer is returned.
if ( == && == )
{
return currentPointer; }
}
//Otherwise find the next node
currentPointer = ;
}
return ;
}
//Create the node
private (IntPtr Pointer, MyselfLinkedListNode Node) CreateNode(string value)
{
// Calculate the size
var size = (typeof(MyselfLinkedListNode));
//Allocate the specified number of bytes of memory space
var pointer = (size);
// Create the instance and set the value
var node = new MyselfLinkedListNode
{
Data = value,
Next =
Next = value, Next = value, Next = value, Next = value, Next = value }; }
// Write the instance to the allocated memory
(node, pointer, false); //Returns the node pointer and node memory.
// Return the node pointer and node
return (pointer, node);
}
Here only a node inserted after the specified node, we can also achieve the specified node before the insertion of the first element node before the insertion of the tail node after the addition, are available, interested in the realization of their own try.
7. Find nodes according to data fields Find
In the chain table is not friendly to find, because find a value, you need to find from the head of the chain table one by one backward, the realization of the logic to the uncomplicated, the specific implementation of the code is as follows:
//Find the node according to the data
public MyselfLinkedListNode Find(string value)
{
//Find from the head pointer
var pointer = _head;
//If the current pointer is null then stop searching
while (pointer ! = )
{
//Get the node corresponding to the current pointer
var node = GetNode(pointer); //If the data field of the current node is the same as the value to be found then return the current node.
//If the data field of the current node is the same as the value to be found, then the current node is returned.
if ( == value)
{
return node; }
}
//Otherwise find the next node
pointer = ;
}
return default; }
}
8, update the specified node data field Update
The logic of this method is also relatively simple, you just need to find the node pointer, then update the node, and finally write the updated data to memory.
//Update node data
public void Update(MyselfLinkedListNode node, string value)
{
//Get the corresponding pointer to the node
var pointer = GetPointer(node); //When the pointer is not empty, then update the node data.
//When the pointer is not null, then update the node data
if (pointer ! = )
{
//Modify the data
= value; //Write the data to the allocated memory to complete the data update.
// Write the data to the allocated memory to complete the data update
(node, pointer, false);
}
}
9. Remove the specified node Remove
If you want to remove a node, you need to remove the connection between the specified node and the node before and after it, and then establish the connection between the two nodes before and after it, and you need to manually free the memory of the deleted node. As shown in the figure below.
The specific code implementation is as follows:
//Remove node
public void Remove(MyselfLinkedListNode node)
{
//Start searching from the head pointer
var currentPointer = _head;
//Get the current node
var currentNode = GetNode(_head);
// Find the pointer corresponding to the node
var pointer = GetPointer(node); //Find the pointer to the node.
while (true)
{
if ( == )
{
// Return if pointer is null
return; }
}
else if ( == pointer)
{
// Point the next node corresponding to the previous node of the node to be deleted to the next node of the node to be deleted.
= ;
// Manually free the memory corresponding to the deleted node
(pointer);
//Update the previous node of the node to be deleted
(currentNode, currentPointer, false); ;//Manually free the memory corresponding to the deleted node.
//Subtract 1 from the length of the chain table.
currentPointer, false); //chain table length minus 1
break;
}
break; }
{
//find the next node
currentPointer = ;
currentNode = GetNode(currentPointer);
}
}
}
10. Destruction of chained tables Destroy
Destruction of the chain table is mainly used because it is our own manual management of memory, after use to clean up in a timely manner, placed in memory leaks and other unforeseen circumstances. Code is also very simple, the cycle of each node memory can be released, the following code:
//Destroy the chained table
public void Destroy()
{
var pointer = _head;
while (pointer != )
{
var value = GetNode(pointer);
(pointer);
_length--;
pointer = ;
}
_head = ;
}
11、Release memory Dispose
Because we implement the IDisposable interface, all we need to do is implement the Dispose method and just call the Destroy method above in the Dispose method to destroy the chain table.
classifier for sums of money: The test method code as well as the sample source code have been uploaded to the code repository for those who are interested./hugogoos/Planner