线索化二叉树(先序,中序,后序)+线索化二叉树的遍历【java详解】

news2024/12/26 0:14:01

目录

线索化二叉树的基本介绍:

 举个栗子:

二叉树的中序线索化:

创建HeroNode类,表示节点信息:

编写中序线索化方法代码:

中序线索化遍历代码:

测试代码:

测试结果:

 二叉树的先序线索化:

编写先序线索化代码:

 先序线索化遍历代码:

测试结果:

二叉树前序和中序线索化完整代码:

二叉树的后续线索化:

定义一个PostTreadBinaryTree 来表示节点信息:

编写createBinaryTree方法来实现二叉树的创建:

编写后续线索化方法代码: 

后序线索化遍历:

代码详解:

 测试代码:

测试结果:

最后,后序线索化的详细代码,有需要可以自己查看:

小结:


写在之前:在学习线索化二叉树之前,要对二叉树的前中后序遍历有一定的了解,如果还不会的小伙伴可以看看我的这篇博客:二叉树-------前,中,后序遍历 + 前,中,后序查找+删除节点 (java详解)-CSDN博客

由于网上以及书上对线索化二叉树的前序和后序及其遍历介绍的较少,这里统一整理了一下,希望对各位有所帮助。欢迎点赞加关注支持博主~

------------------------------------------------------------------------------------------------------------------------------- 

线索化二叉树的基本介绍:

  • n个结点的二叉链表中含有n+1  【公式 2n-(n-1)=n+1】 个空指针域。利用二叉链表中的空指针域,存放指向该结点在某种遍历次序下的前驱和后继结点的指针(这种附加的指针称为"线索")
  • 这种加上了线索的二叉链表称为线索链表,相应的二叉树称为线索二叉树(Threaded BinaryTree)。根据线索性质的不同,线索二叉树可分为前序线索二叉树、中序线索二叉树和后序线索二叉树三种
  • 一个结点的前一个结点,称为前驱结点
  • 一个结点的后一个结点,称为后继结点

 举个栗子:

将一颗普通的二叉树如下图所示(它是一个含有6个节点的二叉树,按照上述公式计算其空指针域为7【n+1】):

中序线索化后,得到如图所示的二叉树(中序线索化后):

 这实际上是一个将二叉树的指针域充分利用的过程(即充分利用各个节点的左右指针),那么,上述的过程究竟是如何实现的呢?看到这里,你想必还是一脸懵圈,别急,然我们再举个栗子(以中序线索化为例):

我们先将上述数字转化为一个数组,当然这个数组不是随便写的,是通过中序遍历得到的,这个数组是:8,3,10,1,14,6  ,接着我们只需要结合原图,将左右指针为空的节点进行连线,具体如下:

说明:数字8左右指针都为空,理应联两条线,但根据其中序遍历的结果,它的左边为空,所以上述图中数字8只连了一根线,对应到3节点。同理数字10连两根线对应3,1;数字14连2根线,对应1,6;6连一根线,对应空(null),实际在图中表示为不连线---》连线的根数是根据节点的左右指针个数决定的,如数字8,节点左右指针为空,连两条线(对应左右指针的方向),而数字6右指针为空,所以连一条线

实际线索化后的二叉树(通常null省略罢了 ):

 通过上面的一些描述,想必你对二叉树的线索化有了一定理解,那么我们接着开始进入正题:

二叉树的中序线索化:

说明:当线索化二叉树后,Node节点的属性left和right有如下情况:

1).left指向的是左子树,也可能是指向前驱节点,比如①节点left指向左子树,二⑩节点的left指向的就是前驱节点

2).right指向的是右子树,也可能是指向后继节点,比如①节点right指向左=右子树,二⑩节点的right指向的就是后继节点

同时,为了方便表示线索后的二叉树,我们这样定义:
1. 如果leftType == 0 表示指向的是左子树, 如果 1 则表示指向前驱结点
2. 如果rightType == 0 表示指向是右子树, 如果 1表示指向后继结点

创建HeroNode类,表示节点信息:


    //先创建HeroNode 结点
    class HeroNode {
        private int no;
        private String name;
        private HeroNode left; //默认null
        private HeroNode right; //默认null
        //说明
        //1. 如果leftType == 0 表示指向的是左子树, 如果 1 则表示指向前驱结点
        //2. 如果rightType == 0 表示指向是右子树, 如果 1表示指向后继结点
        private int leftType;
        private int rightType;


        public int getLeftType() {
            return leftType;
        }

        public void setLeftType(int leftType) {
            this.leftType = leftType;
        }

        public int getRightType() {
            return rightType;
        }

        public void setRightType(int rightType) {
            this.rightType = rightType;
        }

        public HeroNode(int no, String name) {
            this.no = no;
            this.name = name;
        }

        public int getNo() {
            return no;
        }

        public void setNo(int no) {
            this.no = no;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public HeroNode getLeft() {
            return left;
        }

        public void setLeft(HeroNode left) {
            this.left = left;
        }

        public HeroNode getRight() {
            return right;
        }

        public void setRight(HeroNode right) {
            this.right = right;
        }

        @Override
        public String toString() {
            return "HeroNode [no=" + no + ", name=" + name + "]";
        }
    }*/

通过上面的中序线索化构造的栗子,我们可以按照先

(一)先线索化左子树---》(二)线索化当前结点---》(三)在线索化右子树

编写中序线索化方法代码:

public void threadedNodes(HeroNode node) {

        //如果node==null, 不能线索化
        if (node == null) {
            return;
        }

        //(一)先线索化左子树
        threadedNodes(node.getLeft());
        //(二)线索化当前结点

        //处理当前结点的前驱结点
        //以8结点来理解
        //8结点的.left = null , 8结点的.leftType = 1
        if (node.getLeft() == null) {
            //让当前结点的左指针指向前驱结点
            node.setLeft(pre);
            //修改当前结点的左指针的类型,指向前驱结点
            node.setLeftType(1);
        }

        //处理后继结点
        if (pre != null && pre.getRight() == null) {
            //让前驱结点的右指针指向当前结点
            pre.setRight(node);
            //修改前驱结点的右指针类型
            pre.setRightType(1);
        }
        //!!! 每处理一个结点后,让当前结点是下一个结点的前驱结点
        pre = node;

        //(三)在线索化右子树
        threadedNodes(node.getRight());

    }

中序线索化遍历代码:

/遍历线索化二叉树的方法
    public void threadedList() {
        //定义一个变量,存储当前遍历的结点,从root开始
        HeroNode node = root;
        while (node != null) {
            //循环的找到leftType == 1的结点,第一个找到就是8结点
            //后面随着遍历而变化,因为当leftType==1时,说明该结点是按照线索化
            //处理后的有效结点
            while (node.getLeftType() == 0) {
                node = node.getLeft();
            }

            //打印当前这个结点
            System.out.println(node);
            //如果当前结点的右指针指向的是后继结点,就一直输出
            while (node.getRightType() == 1) {
                //获取到当前结点的后继结点
                node = node.getRight();
                System.out.println(node);
            }
            //替换这个遍历的结点
            node = node.getRight();

        }
    }

测试代码:

public class ThreadedBinaryTreeDemo {
    public static void main(String[] args) {
        //测试一把中序线索二叉树的功能
        HeroNode root = new HeroNode(1, "tom");
        HeroNode node2 = new HeroNode(3, "jack");
        HeroNode node3 = new HeroNode(6, "smith");
        HeroNode node4 = new HeroNode(8, "mary");
        HeroNode node5 = new HeroNode(10, "king");
        HeroNode node6 = new HeroNode(14, "dim");
        HeroNode node7 = new HeroNode(1, "xiao");


        //二叉树,后面我们要递归创建, 现在简单处理使用手动创建
        root.setLeft(node2);
        root.setRight(node3);
        node2.setLeft(node4);
        node2.setRight(node5);
        node3.setLeft(node6);

        //测试中序线索化
        ThreadedBinaryTree threadedBinaryTree = new ThreadedBinaryTree();
        threadedBinaryTree.setRoot(root);
        threadedBinaryTree.threadedNodes();

        //测试: 以10号节点测试
        HeroNode leftNode = node5.getLeft();
        HeroNode rightNode = node5.getRight();
        System.out.println("10号结点的前驱结点是 ="  + leftNode); //3
        System.out.println("10号结点的后继结点是="  + rightNode); //1

        //当线索化二叉树后,能在使用原来的遍历方法
        //threadedBinaryTree.infixOrder();
        System.out.println("使用线索化的方式遍历 线索化二叉树");
        threadedBinaryTree.threadedList(); // 8, 3, 10, 1, 14, 6

    }
}

 

接着我们测试一下就以节点⑩为例(如上图),线索化后其前驱节点为③,其后继节点为①

同时,遍历的结果与中序遍历保持一致,即 8,3,10,1,14,6

测试结果:

 二叉树的先序线索化:

按照中序线索化的方法,我给出线索化的图解:

 

 

HeroNode类与上面基本一致,不做过多说明

按照  线索化当前节点---》线索化左子树---》线索化右子树  过程进行

编写先序线索化代码:

  //前序线索化
    public void pretreadedNodes(HeroNode node){
        if(node == null){
            return ;
        }

        if(node.getLeft() == null){
            node.setLeft(pre);
            node.setLeftType(1);
        }
        if(pre != null && pre.getRight() == null){
            pre.setRight(node);
            pre.setRightType(1);
        }
        pre = node;

        if(node.getLeftType() == 0) {
            pretreadedNodes(node.getLeft());
        }
        if(node.getRightType() == 0) {
            pretreadedNodes(node.getRight());
        }
    }

 先序线索化遍历代码:

//遍历前序线索化二叉树
    public void pretreadedList(){
        HeroNode node = root;
        while(node != null){
            while(node.getLeftType() == 0){
                System.out.println(node);
                node = node.getLeft();
            }
            System.out.println(node);
            while(node.getRightType() == 1){
                node = node.getRight();
                System.out.println(node);
            }
            node = node.getRight();
        }
    }

 

接着我们测试一下就以节点⑩为例(如上图),线索化后其前驱节点为⑧,其后继节点为⑥

同时,先序线索化遍历的结果与先序遍历保持一致

测试结果:

二叉树前序和中序线索化完整代码:

//测试
public class ThreadedBinaryTreeDemo {

    public static void main(String[] args){
        HeroNode root = new HeroNode(1, "tom");
        HeroNode node2 = new HeroNode(3, "jack");
        HeroNode node3 = new HeroNode(6, "smith");
        HeroNode node4 = new HeroNode(8, "mary");
        HeroNode node5 = new HeroNode(10, "king");
        HeroNode node6 = new HeroNode(14, "dim");

        root.setLeft(node2);
        root.setRight(node3);
        node2.setLeft(node4);
        node2.setRight(node5);
        node3.setLeft(node6);

        TreadedBinaryTree treadedBinaryTree = new TreadedBinaryTree(root);
        //treadedBinaryTree.intreadedNodes();
        treadedBinaryTree.pretreadedNodes();

        HeroNode leftNode = node5.getLeft();
        HeroNode rightNode = node5.getRight();
        System.out.println("10号节点的前驱节点是:"+leftNode);
        System.out.println("10号节点的后继节点是:"+rightNode);

        System.out.println("使用线索化二叉树的遍历:");
        treadedBinaryTree.pretreadedList();
        //treadedBinaryTree.intreadedList();
        

    }
}

class TreadedBinaryTree{
   private HeroNode root;
    private HeroNode pre = null;

    public TreadedBinaryTree(HeroNode root) {
        this.root = root;
    }

    public void posttreadedNodes(){
        this.posttreadedNodes(root);
    }

    public void intreadedNodes(){
        this.intreadedNodes(root);
    }

    public void pretreadedNodes(){
        this.pretreadedNodes(root);
    }

    //遍历中序线索化二叉树
   public void intreadedList(){
        HeroNode node =root;
        while(node != null){
            while(node.getLeftType() == 0){
                node = node.getLeft();
            }
            System.out.println(node);

            while(node.getRightType() == 1){
                node = node.getRight();
                System.out.println(node);
            }
            node = node.getRight();
        }
   }
    //遍历前序线索化二叉树
    public void pretreadedList(){
        HeroNode node = root;
        while(node != null){
            while(node.getLeftType() == 0){
                System.out.println(node);
                node = node.getLeft();
            }
            System.out.println(node);
            while(node.getRightType() == 1){
                node = node.getRight();
                System.out.println(node);
            }
            node = node.getRight();
        }
    }
   //中序线索化
    public void intreadedNodes(HeroNode node){
        if(node == null){
            return ;
        }
        intreadedNodes(node.getLeft());

        if(node.getLeft() == null){
            node.setLeft(pre);
            node.setLeftType(1);
        }
        if(pre != null && pre.getRight() == null){
            pre.setRight(node);
            pre.setRightType(1);
        }
        pre = node;

        intreadedNodes(node.getRight());
    }

    //前序线索化
    public void pretreadedNodes(HeroNode node){
        if(node == null){
            return ;
        }

        if(node.getLeft() == null){
            node.setLeft(pre);
            node.setLeftType(1);
        }
        if(pre != null && pre.getRight() == null){
            pre.setRight(node);
            pre.setRightType(1);
        }
        pre = node;

        if(node.getLeftType() == 0) {
            pretreadedNodes(node.getLeft());
        }
        if(node.getRightType() == 0) {
            pretreadedNodes(node.getRight());
        }
    }
    //后续线索化二叉树
    public void posttreadedNodes(HeroNode node){
        if(node == null){
            return ;
        }

            posttreadedNodes(node.getLeft());

            posttreadedNodes(node.getRight());


        if(node.getLeft() == null){
            node.setLeft(pre);
            node.setLeftType(1);
        }
        if(pre != null && pre.getRight() == null){
            pre.setRight(node);
            pre.setRightType(1);
        }
        pre = node;

    }



}

class HeroNode{
    private int no;
    private String name;
    private HeroNode left;//默认null
    private HeroNode right;//默认null

    //说明
    //1. 如果leftType == 0 表示指向的是左子树, 如果 1 则表示指向前驱结点
    //2. 如果rightType == 0 表示指向是右子树, 如果 1表示指向后继结点
    private int leftType;
    private int rightType;


    public HeroNode() {
    }

    public HeroNode(int no, String name) {
        this.no = no;
        this.name = name;
    }

    /**
     * 获取
     * @return no
     */
    public int getNo() {
        return no;
    }

    /**
     * 设置
     * @param no
     */
    public void setNo(int no) {
        this.no = no;
    }

    /**
     * 获取
     * @return name
     */
    public String getName() {
        return name;
    }

    /**
     * 设置
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }

    /**
     * 获取
     * @return left
     */
    public HeroNode getLeft() {
        return left;
    }

    /**
     * 设置
     * @param left
     */
    public void setLeft(HeroNode left) {
        this.left = left;
    }

    /**
     * 获取
     * @return right
     */
    public HeroNode getRight() {
        return right;
    }

    /**
     * 设置
     * @param right
     */
    public void setRight(HeroNode right) {
        this.right = right;
    }

    /**
     * 获取
     * @return leftType
     */
    public int getLeftType() {
        return leftType;
    }

    /**
     * 设置
     * @param leftType
     */
    public void setLeftType(int leftType) {
        this.leftType = leftType;
    }

    /**
     * 获取
     * @return rightType
     */
    public int getRightType() {
        return rightType;
    }

    /**
     * 设置
     * @param rightType
     */
    public void setRightType(int rightType) {
        this.rightType = rightType;
    }

    public String toString() {
        return "HeroNode{no = " + no + ", name = " + name + "}";
    }
}

二叉树的后续线索化:

(后序相对前序与中序复杂,单独拎出来讲---》主要是遍历过程复杂)

按照上述中序线索化的过程,我给出后续线索化的图解:

 

定义一个PostTreadBinaryTree 来表示节点信息:

public class PostTreadBinaryTree {

    private Node preNode;   //线索化时记录前一个节点

    //节点存储结构
    static class Node {
        String data;        //数据域
        Node left;          //左指针域
        Node right;         //右指针域
        Node parent;        //父节点的指针(为了后序线索化使用)
        boolean isLeftThread = false;   //左指针域类型  false:指向子节点、true:前驱或后继线索
        boolean isRightThread = false;  //右指针域类型  false:指向子节点、true:前驱或后继线索

        Node(String data) {
            this.data = data;
        }
    }

编写createBinaryTree方法来实现二叉树的创建:

 static Node createBinaryTree(String[] array, int index) {
        Node node = null;

        if(index < array.length) {
            node = new Node(array[index]);
            node.left = createBinaryTree(array, index * 2 + 1);
            node.right = createBinaryTree(array, index * 2 + 2);

            //记录节点的父节点(后序线索化遍历时使用)
            if(node.left != null) {
                node.left.parent = node;
            }

            if(node.right != null) {
                node.right.parent = node;
            }
        }

        return node;
    }

编写后续线索化方法代码: 

void postThreadOrder(Node node) {
        if(node == null) {
            return;
        }

        //处理左子树
        postThreadOrder(node.left);
        //处理右子树
        postThreadOrder(node.right);

        //左指针为空,将左指针指向前驱节点
        if(node.left == null) {
            node.left = preNode;
            node.isLeftThread = true;
        }

        //前一个节点的后继节点指向当前节点
        if(preNode != null && preNode.right == null) {
            preNode.right = node;
            preNode.isRightThread = true;
        }
        preNode = node;
    }

后序线索化遍历:

后序遍历线索化二叉树是一种对二叉树进行线索化的方法,使得在遍历二叉树时可以更高效地找到前驱和后继节点。下面是后序遍历线索化二叉树的思路:

  1. 首先,定义一个辅助指针pre,用于记录当前节点的前驱节点。
  2. 对于二叉树的每个节点,按照后序遍历的顺序进行处理。
  3. 如果当前节点的左子树不为空,就将当前节点的左子树线索化。具体步骤如下:
    • 找到当前节点的左子树的最右节点,即左子树中最后一个被访问的节点。
    • 将该最右节点的右指针指向当前节点,并将其线索化标记设置为1,表示指向后继节点。
  4. 如果当前节点的右子树不为空,就将当前节点的右子树线索化。具体步骤如下:
    • 找到当前节点的右子树的最左节点,即右子树中第一个被访问的节点。
    • 将该最左节点的左指针指向当前节点,并将其线索化标记设置为1,表示指向前驱节点。
  5. 更新pre指针为当前节点,以便在下一次迭代时使用。
  6. 重复步骤2-5,直到遍历完整个二叉树。

这样,通过后序遍历线索化二叉树,我们可以在不使用递归或栈的情况下,高效地找到任意节点的前驱和后继节点。

代码详解:

 void postThreadList(Node root) {
        //1、找后序遍历方式开始的节点
        Node node = root;
        while(node != null && !node.isLeftThread) {
            node = node.left;
        }

        Node preNode = null;
        while(node != null) {
            //右节点是线索
            if(node.isRightThread) {
                System.out.print(node.data + ", ");
                preNode = node;
                node = node.right;

            } else {
                //如果上个处理的节点是当前节点的右节点
                if(node.right == preNode) {
                    System.out.print(node.data + ", ");
                    if(node == root) {
                        return;
                    }

                    preNode = node;
                    node = node.parent;

                } else {    //如果从左节点的进入则找到有子树的最左节点
                    node = node.right;
                    while(node != null && !node.isLeftThread) {
                        node = node.left;
                    }
                }
            }
        }
    }

 测试代码:

 public static void main(String[] args) {
        String[] array = {"1","3","6","8","10","14"};
        Node root = createBinaryTree(array, 0);

        PostTreadBinaryTree tree = new PostTreadBinaryTree();
        tree.postThreadOrder(root);
        System.out.println("后序按后继节点遍历线索二叉树结果:");
        tree.postThreadList(root);
    }

该测试结果(遍历测试结果)与后序遍历的结果 一致

测试结果:

最后,后序线索化的详细代码,有需要可以自己查看:

//后序线索化
public class PostTreadBinaryTree {

    private Node preNode;   //线索化时记录前一个节点

    //节点存储结构
    static class Node {
        String data;        //数据域
        Node left;          //左指针域
        Node right;         //右指针域
        Node parent;        //父节点的指针(为了后序线索化使用)
        boolean isLeftThread = false;   //左指针域类型  false:指向子节点、true:前驱或后继线索
        boolean isRightThread = false;  //右指针域类型  false:指向子节点、true:前驱或后继线索

        Node(String data) {
            this.data = data;
        }
    }

    /**
     * 通过数组构造一个二叉树(完全二叉树)
     * @param array
     * @param index
     * @return
     */
    static Node createBinaryTree(String[] array, int index) {
        Node node = null;

        if(index < array.length) {
            node = new Node(array[index]);
            node.left = createBinaryTree(array, index * 2 + 1);
            node.right = createBinaryTree(array, index * 2 + 2);

            //记录节点的父节点(后序线索化遍历时使用)
            if(node.left != null) {
                node.left.parent = node;
            }

            if(node.right != null) {
                node.right.parent = node;
            }
        }

        return node;
    }

    /**
     * 后序线索化二叉树
     * @param node  节点
     */
    void postThreadOrder(Node node) {
        if(node == null) {
            return;
        }

        //处理左子树
        postThreadOrder(node.left);
        //处理右子树
        postThreadOrder(node.right);

        //左指针为空,将左指针指向前驱节点
        if(node.left == null) {
            node.left = preNode;
            node.isLeftThread = true;
        }

        //前一个节点的后继节点指向当前节点
        if(preNode != null && preNode.right == null) {
            preNode.right = node;
            preNode.isRightThread = true;
        }
        preNode = node;
    }

    /**
     * 后续遍历线索二叉树,按照后继方式遍历(思路:后序遍历开始节点是最左节点)
     * @param
     */
    void postThreadList(Node root) {
        //1、找后序遍历方式开始的节点
        Node node = root;
        while(node != null && !node.isLeftThread) {
            node = node.left;
        }

        Node preNode = null;
        while(node != null) {
            //右节点是线索
            if(node.isRightThread) {
                System.out.print(node.data + ", ");
                preNode = node;
                node = node.right;

            } else {
                //如果上个处理的节点是当前节点的右节点
                if(node.right == preNode) {
                    System.out.print(node.data + ", ");
                    if(node == root) {
                        return;
                    }

                    preNode = node;
                    node = node.parent;

                } else {    //如果从左节点的进入则找到有子树的最左节点
                    node = node.right;
                    while(node != null && !node.isLeftThread) {
                        node = node.left;
                    }
                }
            }
        }
    }

    public static void main(String[] args) {
        String[] array = {"1","3","6","8","10","14"};
        Node root = createBinaryTree(array, 0);

        PostTreadBinaryTree tree = new PostTreadBinaryTree();
        tree.postThreadOrder(root);
        System.out.println("后序按后继节点遍历线索二叉树结果:");
        tree.postThreadList(root);
    }
}

小结:

1. 前序线索化二叉树遍历相对最容易理解,实现起来也比较简单。由于前序遍历的顺序是:根左右,所以从根节点开始,沿着左子树进行处理,当子节点的left指针类型是线索时,说明到了最左子节点,然后处理子节点的right指针指向的节点,可能是右子树,也可能是后继节点,无论是哪种类型继续按照上面的方式(先沿着左子树处理,找到子树的最左子节点,然后处理right指针指向),以此类推,直到节点的right指针为空,说明是最后一个,遍历完成。 
2. 中序线索化二叉树的网上相关介绍最多。中序遍历的顺序是:左根右,因此第一个节点一定是最左子节点,先找到最左子节点,依次沿着right指针指向进行处理(无论是指向子节点还是指向后继节点),直到节点的right指针为空,说明是最后一个,遍历完成。 
3. 后序遍历线索化二叉树最为复杂,通用的二叉树数节点存储结构不能够满足后序线索化,因此我们扩展了节点的数据结构,增加了父节点的指针。后序的遍历顺序是:左右根,先找到最左子节点,沿着right后继指针处理,当right不是后继指针时,并且上一个处理节点是当前节点的右节点,则处理当前节点的右子树,遍历终止条件是:当前节点是root节点,并且上一个处理的节点是root的right节点。

小结转载自:后序线索化二叉树(Java版)_线索二叉树的后序遍历-CSDN博客
 

博客到这里也是结束了,制作不易,喜欢的小伙伴可以点赞加关注支持下博主,这对我真的很重要~~

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1452834.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

OpenAI发布Sora技术报告深度解读!真的太强了!

&#x1f60e; 作者介绍&#xff1a;我是程序员洲洲&#xff0c;一个热爱写作的非著名程序员。CSDN全栈优质领域创作者、华为云博客社区云享专家、阿里云博客社区专家博主、前后端开发、人工智能研究生。公粽号&#xff1a;洲与AI。 &#x1f388; 本文专栏&#xff1a;本文收录…

NodeJS背后的人:Express

NodeJS背后的人&#xff1a;Express 本篇文章&#xff0c;学习记录于&#xff1a;尚硅谷&#x1f3a2; 文章简单学习总结&#xff1a;如有错误 大佬 &#x1f449;点. 前置知识&#xff1a;需要掌握了解&#xff1a; JavaScript基础语法 、Node.JS环境API 、前端工程\模块化 …

代码随想录算法训练营第53天 | 121.买卖股票的最佳时机 + 122.买卖股票的最佳时机II

今日任务 121. 买卖股票的最佳时机 122.买卖股票的最佳时机II 121.买卖股票的最佳时机 - Easy 题目链接&#xff1a;力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第…

Rust 语言学习杂谈 (end) (各种工作中遇到的疑难杂症)

1.在运行 “cargo build --release” 的时候&#xff0c;到底发生了什么&#xff1f; 源 (GPT4.0) : 当我们运行 cargo build --release 命令时&#xff0c;实际上在进行一系列复杂的步骤来编译和构建 Rust 项目的发布版本。这个过程大致可以分解为以下几个步骤&#xff1a;…

Java - SPI机制

本文参考&#xff1a;SPI机制 SPI&#xff08;Service Provider Interface&#xff09;&#xff0c;是JDK内置的一种服务提供发现机制&#xff0c;可以用来启动框架扩展和替换组件&#xff0c;主要是被框架的开发人员使用&#xff0c;比如 java.sql.Driver接口&#xff0c;其他…

【C语言】实现队列

目录 &#xff08;一&#xff09;队列 &#xff08;二&#xff09;头文件 &#xff08;三&#xff09; 功能实现 &#xff08;1&#xff09;初始化 &#xff08;2&#xff09; 销毁队列 &#xff08;3&#xff09; 入队 &#xff08;4&#xff09;出队 &#xff08;5&a…

【论文精读】GPT1

摘要 如何从大量未标注文本中获取词级别的信息有两个主要挑战&#xff0c;使用何种优化目标能有效地学习文本表示&#xff0c;如何有效地将学习到的表示迁移到目标任务。针对这些问题&#xff0c;本文提出一种无监督预训练和有监督微调的组合的半监督方法&#xff0c;具体为&am…

Vue3+Ant-Design-Vue:报错Cannot read properties of null (reading ‘isCE‘)

问题描述 在使用Ant-Design-Vue内置的Table表格组件&#xff0c;实现expand展开行功能时&#xff0c;报错&#xff1a;Uncaught TypeError: Cannot read properties of null (reading ‘isCE‘) 。 报错信息图示&#xff1a; 在GitHub上找到如下描述&#xff0c; 解决方案 网上…

【maya 入门笔记】基本视图和拓扑

1. 界面布局 先看基本窗口布局&#xff0c;基本窗口情况如下&#xff1a; 就基本窗口布局的情况来看&#xff0c;某种意义上跟blender更像一点&#xff08;与3ds max相比&#xff09;。 那么有朋友就说了&#xff0c;玛格基&#xff0c;那blender最下面的时间轴哪里去了&…

书生浦语大模型实战营-课程笔记(4)

微调分为两种&#xff0c;增量预训练和指令跟随。 指令跟随微调&#xff1a; 1.只对答案计算Loss 2.训练时数据为一问一答的形式&#xff08;input和output&#xff09; 增量预训练&#xff1a; 只需要output的数据进行训练 xtuner:微调框架 操作部分的笔记参考git上的文档…

杨中科 ASP.NET DI综合案例

综合案例1 需求说明 1、目的:演示DI的能力; 2、有配置服务、日志服务&#xff0c;然后再开发一个邮件发送器服务。可以通过配置服务来从文件、环境变量、数据库等地方读取配置&#xff0c;可以通过日志服务来将程序运行过程中的日志信息写入文件、控制台、数据库等。 3、说明…

python 基础知识点(蓝桥杯python科目个人复习计划43)

今日复习内容&#xff1a;做点真题和继续复习动态规划 例题1&#xff1a;三国游戏 &#xff08;用的知识点是贪心和排序&#xff09; 题目描述&#xff1a; 小蓝正在玩一款游戏&#xff0c;游戏中魏&#xff08;X&#xff09;&#xff0c;蜀&#xff08;Y&#xff09;&…

HAL/LL/STD STM32 U8g2库 +I2C SSD1306/sh1106 WouoUI磁贴案例

HAL/LL/STD STM32 U8g2库 I2C SSD1306/sh1106 WouoUI磁贴案例 &#x1f4cd;基于STM32F103C8T6 LL库驱动版本&#xff1a;https://gitee.com/chcsx/platform-test/tree/master/MDK-ARM&#x1f3ac;视频演示&#xff1a; WouoUI移植磁贴案例&#xff0c;新增确认弹窗 &#x1f…

《剑指Offer》笔记题解思路技巧优化 Java版本——新版leetcode_Part_3

《剑指Offer》笔记&题解&思路&技巧&优化_Part_3 &#x1f60d;&#x1f60d;&#x1f60d; 相知&#x1f64c;&#x1f64c;&#x1f64c; 相识&#x1f622;&#x1f622;&#x1f622; 开始刷题1. LCR 138. 有效数字——表示数值的字符串2. LCR 139. 训练计划…

【数据结构】17 二叉树的建立

二叉树的建立 由于树是非线性结构&#xff0c;创建一颗二叉树必须首先确定树中结点的输入顺序&#xff0c;常用方法是先序创建和层序创建。 层序创建所用的节点输入序列是按数的从上至下从左到右的顺序形成的各层的空结点输入数值0。在构造二叉树过程中需要一个队列暂时存储各…

AI数据中心网络架构需求:400/800G光模块

随着AI技术和相关应用的不断发展&#xff0c;大模型、大数据和AI计算能力在AI发展中的重要性日益凸显。大模型和数据集构成AI研究的软件基础&#xff0c;而AI算力是关键的基础设施。在本文中&#xff0c;我们将探讨AI发展对数据中心网络架构的影响。 Fat-Tree数据中心网络架构…

《白话C++》第10章 STL和boost,Page67~70 std::auto_ptr

std::auto_ptr可以不经意间转移裸指针控制权 std::auto_ptr持有裸指针的控制权&#xff0c;却可以随随便便看似不经意地转移给另一个auto_ptr: #include <iostream> #include <memory>using namespace std;struct S {int a;void SetA(int a){this->a a;}~S()…

跟着pink老师前端入门教程(JavaScript)-day02

三、变量 &#xff08;一&#xff09;变量概述 1、什么是变量 白话&#xff1a;变量就是一个装东西的盒子 通俗&#xff1a;变量是用于存放数据的容器&#xff0c;通过变量名获取数据&#xff0c;甚至数据可以修改 2、变量在内存中的存储 本质&#xff1a;变量是程序在内存…

记录:零基础小白初学云计算 第一天

一、认识【rootlocalhost ~】# root:当前登录用户的用户名 localhost&#xff1a;主机名 ~&#xff1a;当前用户的家目录 #&#xff1a;超级用户的命令提示符 基础命令 ifup ens33&#xff1a;启动网卡 ip a&#xff1a;查看IP地址 远程连接端口默认 &#xff1a;22 二…

WordPress主题YIA移动端文章页的面包屑不显示怎么办?

平时我们一般都会在文章页导航菜单下方显示面包屑&#xff0c;类似于“当前位置&#xff1a;boke112百科 WordPress 正文”。平时用浏览器调试站点的时候&#xff0c;在Edge浏览器的“切换设备仿真”中&#xff0c;不管是选择什么设备都会显示面包屑。具体如下图所示&#xf…