Ordinary tree
In real life, trees grow upward, but in the legend of a data structure, trees usually grow upside down because they look better from top to bottom.
An ordinary tree looks like this:
From this structure of trees we can conclude the following concepts
Some basic concepts of tree structure
Parent: If a node has a next-level node, it is called the next-level parent. Node 3 in the legend is the parent of node 8.
Child: If a node has a parent node, it is called the child of the parent node. Node 2 in the legend is the child of node 1.
Root: The topmost node of a tree, which has no parents and only children. Node 1 in the legend is the root node.
Leaf node: A leaf node is at the end of a branch of the tree and has no children. The 5, 6, 7, 8, 9, 10 nodes in the legend are all leaf nodes, because they don’t have children anymore.
Subtree: The children of a node can be thought of as a subtree. Node 3 in the legend is the child of the root node, and the tree with node 3 as the root is called the subtree of root 1.
The number of layers of the tree: the first layer is calculated from the root node, and the second layer and the third layer are successively calculated. Look at the legend.
Depth: The depth of a tree is the maximum number of layers. For example, if a tree has four layers, its depth is four.
Degree: number of subtrees. As the legend shows, the root node has three subtrees (2, 3, 4), so the degree of the root node is 3. If there are three subtrees of node 2 (5, 6, 7), then the degree of node 2 is 3.
The nature of the tree
- A non-empty tree has one and only one root node
- You can access the rest of the nodes from one node, and there is only one path from one node to the other
- A parent can have multiple children, except for the root, and all other nodes have one or only one parent
Binary tree
Concept: Each node has at most two children. A tree that satisfies this condition is called a binary tree.
There will be only left subtrees and only right subtrees and so on
According to different forms, it can be divided into the following types of binary trees:
Perfect Binary Tree
A binary tree is called full or perfect if it has a depth of K and 2k−12^ k-12K −1 nodes, that is, all leaves at this depth are full.
Complete Binary Tree
A binary tree is called a complete binary tree if it has a depth of K and is a full binary tree from layer 1 to layer K-1, with nodes at layer K concentrated on the left and not completely filled.
So don’t waste your time trying to figure out the difference between a full binary tree and a complete binary tree, because if you’re perfect you’re perfect, if you’re full you’re full, they’re all in their own right, and don’t let anybody confuse you by saying full is a special case or something.
Because of its special form, complete binary tree can be used when sorting or searching the data stored in it, which is relatively high efficiency. Therefore, some data are often converted into complete binary tree and then processed, which will be introduced in detail after learning.
Full Binary Tree
The English names of Full binary tree, complete binary tree and Perfect binary tree are different, but the Chinese translation is a little difficult to distinguish. Full corresponds to Perfect, which means Perfect, and Perfect corresponds to Full, which means that the children of each parent node are Full.
A binary tree is called perfect if all non-leaf nodes have a degree of 2.
Obviously, a full binary tree is a perfect binary tree, but a full binary tree has its own home, it doesn’t come here.
Properties of binary trees
Because of the simple morphology of binary trees, we can summarize some general properties of binary trees:
-
Assuming the depth of the binary tree is K, the k-layer has at most 2k−12^{k-1}2k−1 nodes. For example, a binary tree with a depth of 4 has at most 23−1=42^{3-1} = 423−1=4 nodes at the third level.
-
The maximum number of nodes in layer K is twice that in layer K-1. For example, the third layer has four nodes, and the fourth layer has at most eight nodes, twice as many as the previous layer.
-
Assuming that the depth of a binary tree is K, it will have at most 2k−12^ K-12K −1 nodes. For example, a binary tree with a depth of 4 has a maximum of 24−1=152^ 4-1 = 1524−1=15 nodes.
-
Let n0N_0N0 be the number of leaf nodes with degree 0, n1N_1n1 be the number of leaf nodes with degree 1, and n2N_2n2 be the number of leaf nodes with degree 2 on any binary tree (the whole tree or a certain subtree). So n0N_0n0 is 1 more than N2N_2n2, so n0=n2+ 1N_0 = N_2 + 1N0 =n2+1. For example: a full binary tree of depth 4, then we can get n0=8,n1=0,n2= 7N_0 =8, N_1 =0, n_2 = 7N0 =8,n1=0,n2=7.
-
If NNN is set as the total number of nodes in the binary tree, we can get: n= N0 + N1 + N2n = N_0 + N_1 + N_2N = N0 + N1 +n2. In the example above, n=8+0+7=15n =8+0+7=15n =8+0+7=15. We can also see that if the total number of nodes is NNN, then the number of edges is n− 1n-1n −1, which in the above example is 8−1=7, 8-1 = 78−1=7.
-
And since the node of degree 2 has two edges under it, the node of degree 1 has one edge, and the leaf node has no edge, then we can get: N −1=n1+2∗ n2n-1 = N_1 +2 * N_2n −1= N1 +2∗n2, N0 +n1+n2=n1+2∗n2+1 N0 = N2 + 1N_0 + N_1 + N_2 = N_1 +2 * N_2 +1 N_0 = N_2 + 1N0 + N1 +n2=n1+2∗n2+1 In other words: the number of leaves is 1 more than the number of nodes with degree 2. In the example above, n0=8,n2= 7N_0 =8, N_2 = 7N0 =8,n2=7, conform to this formula.
The generalized table representation of a binary tree
The generalized representation, in a nutshell, looks like this: LS=(a1,a2… LS, an) = (a_1, a_2,… LS, a_n) = (a1, a2,… ,an), we use a generalized table to represent a tree like this:
Root (left,right)root(left, right)root(left,right) So are:
Root indicates only the root node.
Root (left) indicates only the left node and the right node is empty.
Root (, right) indicates that the left node is empty and only the right node is present.
Root (left, right) indicates both left and right nodes.
For example, 9(8(7, 6), 5(4, 3)) can represent a binary tree with 9 as the root node
What does it do?
Suppose you have a Huffman tree, and the binary data encoded by Huffman, and you pass the data to someone else, you also need to pass the code table to someone else, otherwise they can’t decode it. At this point, you can use the form of generalized table to pass Huffman tree to the other party, you can transform into a Huffman tree according to the generalized table, so you can decode.
Binary tree structure
So with the theory behind us let’s see how we can create a binary tree.
By definition, a binary tree consists of three parts: a data field, a left node, and a right node
/** * binary tree structure *@param {*} Data Node data */
function BinaryTree(data) {
this.data = data;
this.lchild = null;
this.rchild = null;
}
Copy the code
Manually create a binary tree for later use
Create a binary tree by hand so that you can get a sense of what its structure looks like. Or you could write a function to generate it.
const root = new BinaryTree(1);
root.lchild = new BinaryTree(2);
root.rchild = new BinaryTree(3);
root.lchild.lchild = new BinaryTree(4);
root.lchild.rchild = new BinaryTree(5);
root.rchild.lchild = new BinaryTree(6);
root.rchild.rchild = new BinaryTree(7);
Copy the code
Once created, it will look like this:
Traversal of binary trees
Since this article is the foundation, let’s learn how to traverse binary trees first.
There are three generally accepted ways of traversing binary trees, but you can also make up your own traversal methods, such as odd and even traversal (traversing odd-numbered nodes first and then even-numbered ones, although I’m just thinking about that). The purpose of saying this sentence is to explain that don’t limit your thinking to death.
Note: the order of first, middle and last refers to the parent node, that is, the processing order of the nodes currently traversed.
This paper uses recursion to traverse binary trees.
The first sequence traversal
Preemptive traversal means that when you walk through the current node, you print the current node, then the left node, and then the right node.
So for the generalized table: 1(2,3), the output is: 1, 2,3
/** * sequential traversal of binary tree *@param {*} * / node tree
function preorder(node) {
// When traversing the current node, output the current node first
console.log(node.data);
// Then enter the left subtree
if(node.lchild ! = =null) {
preorder(node.lchild);
}
// Finally enter the right subtree
if(node.rchild ! = =null) { preorder(node.rchild); }}// Use the root created above
// Output: 1, 2, 4, 5, 3, 6, 7
preorder(root);
Copy the code
In the sequence traversal
Middle-order traversal says that when traversing the current node, output the left node first, then the current node, and finally the right node.
So for the generalized table: 1(2,3), the output is: 2, 1, 3
/** * middle order traversal of binary tree *@param {*} * / node tree
function inorder(node) {
// Output the left node first
if(node.lchild ! = =null) {
inorder(node.lchild);
}
// Output the current node
console.log(node.data);
// Outputs the right node
if(node.rchild ! = =null) { inorder(node.rchild); }}// Output: 4 2 5 1 6 3 7
inorder(root);
Copy the code
After the sequence traversal
Backward traversal means that when you walk through the current node, you output the left node first, then the right node, and then the current node.
So for the generalized table: 1(2,3), the output is: 2,3, 1
/** * subsequent traversal of the binary tree *@param {*} * / node tree
function postorder(node) {
// Output the left node first
if(node.lchild ! = =null) {
inorder(node.lchild);
}
// Output the right node
if(node.rchild ! = =null) {
inorder(node.rchild);
}
// Outputs the current node
console.log(node.data);
}
// Output: 4 2 5 6 3 7 1
postorder(root);
Copy the code
summary
Have you noticed that the code for the first, middle and last traversal of a binary tree is mostly the same, except that the output order is very different when you move the output position.
If you are still confused by this section, I suggest you think about it in your head and draw on a piece of paper.