二叉搜索树的基础操作

news2025/1/16 5:45:46

如果对于二叉搜索树不是太清楚,为什么要使用二叉搜索树?作者推荐:二叉搜索树的初步认识_加瓦不加班的博客-CSDN博客

定义节点

static class BSTNode {
    int key; // 若希望任意类型作为 key, 则后续可以将其设计为 Comparable 接口
    Object value;
    BSTNode left;
    BSTNode right;

    public BSTNode(int key) {
        this.key = key;
        this.value = key;
    }

    public BSTNode(int key, Object value) {
        this.key = key;
        this.value = value;
    }

    public BSTNode(int key, Object value, BSTNode left, BSTNode right) {
        this.key = key;
        this.value = value;
        this.left = left;
        this.right = right;
    }
}

查询

参照图:

递归实现

//解题思路:从根节点出发 比根节点大的向右找 比根节点小的向左找  如果相等则返回
public Object get(int key) {
    return doGet(root, key);
}
//查询方法:递归实现
private Object doGet(BSTNode node, int key) {
    if (node == null) {
        return null; // 没找到
    }
    if (key < node.key) {
        return doGet(node.left, key); // 向左找
    } else if (node.key < key) {
        return doGet(node.right, key); // 向右找
    } else {
        return node.value; // 找到了
    }
}

非递归实现

public Object get(int key) {
    BSTNode node = root;
    while (node != null) {
        if (key < node.key) {
            node = node.left;
        } else if (node.key < key) {
            node = node.right;
        } else {
            return node.value;
        }
    }
    return null;
}

Comparable

随便给个泛型T就能参与大小比较吗?不能,所以我们要对T进行限制,让它能够参与大小比较,那么就需要实现一个接口: Comparable 接口

如果希望让除 int 外更多的类型能够作为 key,一种方式是 key 必须实现 Comparable 接口。

代码实现:

//<T extends Comparable<T>>:将来我的泛型T就有个上限了,必须是Comparable的子类型,那么T就再也不是任意类型
public class BSTTree2<T extends Comparable<T>> {
    static class BSTNode<T> {
        T key; // 若希望任意类型作为 key, 则后续可以将其设计为 Comparable 接口
        Object value;
        BSTNode<T> left;
        BSTNode<T> right;

        public BSTNode(T key) {
            this.key = key;
            this.value = key;
        }

        public BSTNode(T key, Object value) {
            this.key = key;
            this.value = value;
        }

        public BSTNode(T key, Object value, BSTNode<T> left, BSTNode<T> right) {
            this.key = key;
            this.value = value;
            this.left = left;
            this.right = right;
        }
    }

    //定义根节点
    BSTNode<T> root;

    public Object get(T key) {
        return doGet(root, key);
    }

    //注意:key.compareTo(p.key)的方法比较
    //-1: key p.key
    //0: key =p.key
    //1: key p.key
    //递归的实现
    private Object doGet(BSTNode<T> node, T key) {
        if (node == null) {
            return null;
        }
        int result = node.key.compareTo(key);
        if (result > 0) {
            return doGet(node.left, key);
        } else if (result < 0) {
            return doGet(node.right, key);
        } else {
            return node.value;
        }
    }
    //非递归的实现
    //注意:key.compareTo(p.key)的方法比较
    //-1: key p.key
    //0: key =p.key
    //1: key p.key
    //非递归的实现
    public Object get(T key){
        BSTNode<T> p=root;
        while (p!=null){
            int result = key.compareTo(p.key);
            if(result<0){
                p=p.left;
            }else if(result>0){
                p=p.right;
            }else {
                return p.value;
            }
        }
        return null;
    }

}

还有一种做法不要求 key 实现 Comparable 接口,而是在构造 Tree 时把比较规则作为 Comparator 传入,将来比较 key 大小时都调用此 Comparator 进行比较,这种做法可以参考 Java 中的 java.util.TreeMap

当然我们也可以实现像Map一样的格式,给value也加个泛型,然后我们将key的泛型修改一下名字,就更像Map<K,V>:

public class BSTTree2<K extends Comparable<K>, V> {
static class BSTNode<K, V> {
    K key;
    V value;
    BSTNode<K, V> left;
    BSTNode<K, V> right;

    public BSTNode(K key) {
        this.key = key;
    }

    public BSTNode(K key, V value) {
        this.key = key;
        this.value = value;
    }

    public BSTNode(K key, V value, BSTNode<K, V> left, BSTNode<K, V> right) {
        this.key = key;
        this.value = value;
        this.left = left;
        this.right = right;
    }
}

BSTNode<K, V> root;

public V get(K key) {
    BSTNode<K, V> p = root;
    while (p != null) {
        /*
            -1 key < p.key
            0 key == p.key
            1 key > p.key
         */
        int result = key.compareTo(p.key);
        if (result < 0) {
            p = p.left;
        } else if (result > 0) {
            p = p.right;
        } else {
            return p.value;
        }
    }
    return null;
	}
}

但是后面,我们为了更加使得我们小白能够更好理解代码,我们还是以int key,object value来实现后续的代码。

查询最小值

参照图:

递归实现

思路:从根节点开始,一直往左走,一直走到最后的节点没有左孩子就停止

//思路:从根节点开始,一直往左走,一直走到最后的节点没有左孩子就停止
public Object min() {
    return doMin(root);
}

public Object doMin(BSTNode node) {
    if (node == null) {//当给的节点为Null 我们就不需要去找
        return null;
    }
    // 左边已走到头
    if (node.left == null) { //最小的节点
        return node.value;
    }
    return doMin(node.left);
}

非递归实现

public Object min() {
    if (root == null) {
        return null;
    }
    BSTNode p = root;
    // 左边未走到头
    while (p.left != null) {
        p = p.left;
    }
    return p.value;
}

查询最大值

参考:其实现方法与《最小》的操作基本上是一样的

递归实现

public Object max() {
    return doMax(root);
}

public Object doMax(BSTNode node) {
    if (node == null) {
        return null;
    }
    // 右边已走到头
    if (node.left == null) { 
        return node.value;
    }
    return doMin(node.right);
}

非递归实现

public Object max() {
    if (root == null) {
        return null;
    }
    BSTNode p = root;
    // 右边未走到头
    while (p.right != null) {
        p = p.right;
    }
    return p.value;
}

新增操作

参考图:

解题思路:

//1.key在该二叉树中有 那就更新key所对应的value值
//2.key在该二叉树中没有 那就新增key与所对应的value值

新增前:

新增后:

图示解析:以上述图为例,新增前,从根节点开始找,找到8时,9大于8,那么就要要继续向右找,但是此时前进不了了,因为8已经是叶子节点,走到头还没有找到9,那么就新增9连通值一起创建出来,做为8的右孩子新增进去。

递归实现

public void put(int key, Object value) {
    root = doPut(root, key, value);
}

private BSTNode doPut(BSTNode node, int key, Object value) {
    if (node == null) {
        return new BSTNode(key, value);
    }
    if (key < node.key) {
        node.left = doPut(node.left, key, value);
    } else if (node.key < key) {
        node.right = doPut(node.right, key, value);
    } else {
        node.value = value;
    }
    return node;
}
  • 若找到 key,走 else 更新找到节点的值

  • 若没找到 key,走第一个 if,创建并返回新节点

    • 返回的新节点,作为上次递归时 node 的左孩子或右孩子

    • 缺点是,会有很多不必要的赋值操作

非递归实现

public void put(int key, Object value) {
    BSTNode node = root;
    BSTNode parent = null;
    while (node != null) {
        parent = node;// 4 7 8
        if (key < node.key) {
            node = node.left;
        } else if (node.key < key) {
            node = node.right;//4->7 7->8 8->null
        } else {
            // 1. key 存在则更新
            node.value = value;
            return;
        }
    }
    // 2. key 不存在则新增
    if (parent == null) {
        //当我二叉树是null,那么parent初始就是Null 那么新增的key就是根节点
        root = new BSTNode(key, value);
    } else if (key < parent.key) {
        parent.left = new BSTNode(key, value);
    } else {
        parent.right = new BSTNode(key, value);
    }
}

查询节点的前驱后继

什么叫屈曲与后继?

答:

一个节点的前驱(前任)节点是指比它小的节点中,最大的那个

一个节点的后继(后任)节点是指比它大的节点中,最小的那个

1 2 3 4 5 6 7 8

例如上图中

  • 1 没有前驱,后继是 2

  • 2 前驱是 1,后继是 3

  • 3 前驱是 2,后继是 4

  • ...

查询节点前驱

简单的办法是中序遍历,即可获得排序结果,此时很容易找到前驱后继

二叉树的中序遍历就是升序的结果

要效率更高,需要研究一下规律,找前驱分成 2 种情况:

  1. 节点有左子树,此时前驱节点就是左子树中的最大值, 图中属于这种情况的有

    • 2 的前驱是1

    • 4 的前驱是 3

    • 6 的前驱是 5

    • 7 的前驱是 6

    个人理解:比如:(4的前驱节点有 1 2 3,其中最大值就是3)

  2. 节点没有左子树,若离它最近的祖先自从左而来,此祖先即为前驱,如

    • 3 的祖先 2 自左而来,前驱 2

    • 5 的祖先 4 自左而来,前驱 4

    • 8 的祖先 7 自左而来,前驱 7

    • 1 没有这样的祖先,前驱 null

个人理解:比如:(5的祖先节点有 6 7 4,其中以5为参考点,右边6和7都是比5大的,左边4是比5小的,从左而来的祖先4即为前驱)

比如:(3的祖先节点有 2 4,其中以3为参考点,右边4是比3大的,左边2是比3小的,从左而来的祖先2即为前驱)

// 情况2 - 有祖先自左而来
//对于情况2,我们如何知道哪些节点是要找节点的祖先,又是如何知道这些祖先节点哪些是自从左而来的呢?
//以5节点为例,那我要找到5这个节点的过程中,它必然已经经历过一些节点了,从根节点4开始找,5比4大就向右找,7比5大就向左找,6比5大就像左找,找到了


//在从根节点开始4 7 6是不是都是5的祖先?是的  其实循环的每步都是在经历他这些祖先节点,好,现在知道怎么去获取祖先节点。


//那我们怎么去进一步判断 它这个祖先是左边来还是从右边来呢?
//答:你看4到7是不是向右走,那么以5为参考点,那么4是不是在左边?那么7到6、6到5都是向左走,但是以5为参考点,那么7到6、6到5都是向右走
//所以只要我们看到这种向右走的代码if (p.key < key) {p = p.right;}那就表示祖先是自左而来
//而且我们每次循环更新都是最新也就是最近的自左而来的祖先节点

在predecessor方法之前,我们对于Max方法进行简单的修改,因为我们上面写的Max方法仅仅只是针对于root根节点来查询最大:

   //非递归实现  针对于root的max方法
    public Object max() {
     return   max(root);
//        if (root == null) {
//            return null;
//        }
//        BSTNode p = root;
//        // 右边未走到头
//        while (p.right != null) {
//            p = p.right;
//        }
//        return p.value;
    }
    

    //通用的Max方法
    private Object max(BSTNode node){
        if (node == null) {
            return null;
        }
        BSTNode p = node;
        // 右边未走到头
        while (p.right != null) {
            p = p.right;
        }
        return p.value;
    }

然后我们就开始书写查询节点的前驱代码:

public Object predecessor(int key) {
    BSTNode ancestorFromLeft = null;
    BSTNode p = root;
    //查找用户给的Key在二叉树中是否有
    while (p != null) {
        if (key < p.key) {
            p = p.left;
        } else if (p.key < key) {
            ancestorFromLeft = p;
            p = p.right;
        } else {
            break;
        }
    }
    //key没找到,说明也就没有前任节点
    if (p == null) {
        return null;
    }
    // 情况1 - 有左孩子
    if (p.left != null) {
        return max(p.left);
    }
    // 情况2 - 有祖先自左而来
    //对于情况2,我们如何知道哪些节点是要找节点的祖先,又是如何知道这些祖先节点哪些是自从左而来的呢?
    //以5节点为例,那我要找到5这个节点的过程中,它必然已经经历过一些节点了,从根节点4开始找,5比4大就向右找,7比5大就向左找,6比5大就像左找,找到了
    //在从根节点开始4 7 6是不是都是5的祖先?是的  其实循环的每步都是在经历他这些祖先节点,好,现在知道怎么去获取祖先节点。
    //那我们怎么去进一步判断 它这个祖先是左边来还是从右边来呢?
    //答:你看4到7是不是向右走,那么以5为参考点,那么4是不是在左边?那么7到6、6到5都是向左走,但是以5为参考点,那么7到6、6到5都是向右走
    //所以只要我们看到这种向右走的代码if (p.key < key) {p = p.right;}那就表示祖先是自左而来
    //而且我们每次循环更新都是最新也就是最近的自左而来的祖先节点
           
    return ancestorFromLeft != null ? ancestorFromLeft.value : null;
}

查询节点后继

找后继也分成 2 种情况 与找前任的代码相类似

  1. 节点有右子树,此时后继节点即为右子树的最小值,如

    • 2 的后继 3

    • 3 的后继 4

    • 5 的后继 6

    • 7 的后继 8

  2. 节点没有右子树,若离它最近的祖先自从右而来,此祖先即为后继,如

    • 1 的祖先 2 自右而来,后继 2

    • 4 的祖先 5 自右而来,后继 5

    • 6 的祖先 7 自右而来,后继 7

    • 8 没有这样的祖先,后继 null

在successor方法之前,我们对于Min方法进行简单的修改,因为我们上面写的Min方法仅仅只是针对于root根节点来查询最小:

    //非递归实现
    public Object min() {
        return min(root);
//        if (root == null) {
//            return null;
//        }
//        BSTNode p = root;
//        // 左边未走到头
//        while (p.left != null) {
//            p = p.left;
//        }
//        return p.value;
    }
    
    private Object min(BSTNode node){
        if (node == null) {
            return null;
        }
        BSTNode p = node;
        // 左边未走到头
        while (p.left != null) {
            p = p.left;
        }
        return p.value;
    }

然后我们就开始书写查询节点的后继代码:

public Object successor(int key) {
    BSTNode ancestorFromRight = null;
    BSTNode p = root;
    while (p != null) {
        if (key < p.key) {
            ancestorFromRight = p;
            p = p.left;
        } else if (p.key < key) {
            p = p.right;
        } else {
            break;
        }
    }

    if (p == null) {
        return null;
    }
    // 情况1 - 有右孩子
    if (p.right != null) {
        return min(p.right);
    }
    // 情况2 - 有祖先自右而来
    return ancestorFromRight != null ? ancestorFromRight.value : null;
}

删除操作

要删除某节点(称为 D),必须先找到被删除节点的父节点,这里称为 Parent

1.删除节点没有左孩子,将右孩子托孤给 Parent

2.删除节点没有右孩子,将左孩子托孤给 Parent

3.删除节点左右孩子都没有,已经被涵盖在情况1、情况2当中,把null托孤给Parent

4.删除节点左右孩子都有,可以将它的后继节点(称为S)托孤给Parent,设S的父亲为SP,又分两种情况:

               1.SP 就是被删除节点,此时 D 与 S 紧邻,只需将 S 托孤给 Parent

               2.SP 不是被删除节点,此时 D 与 S 不相邻,此时需要将 S 的后代托孤给 SP,再将 S 托孤给 Parent

非递归实现

/**
 * <h3>根据关键字删除</h3>
 *
 * @param key 关键字
 * @return 被删除关键字对应值
 */
public Object delete(int key) {
    BSTNode p = root;
    BSTNode parent = null;//记录待删除节点的父亲
    while (p != null) {
        if (key < p.key) {
            parent = p;
            p = p.left;
        } else if (p.key < key) {
            parent = p;
            p = p.right;
        } else {
            break;
        }
    }
    if (p == null) {
        return null;
    }
    // 删除操作
    if (p.left == null) {
        shift(parent, p, p.right); // 情况1
    } else if (p.right == null) {
        shift(parent, p, p.left); // 情况2
    } else {
        // 情况4:被删除节点是左、右都有子节点,所以找后继节点:节点有右子树,此时后继节点即为右于树的最小值
        // 4.1 被删除节点找后继
        BSTNode s = p.right;//后继节点
        BSTNode sParent = p; // 后继父亲
        while (s.left != null) {  //当循环结束,后继节点即为s
            sParent = s;
            s = s.left;
        }
        // 4.2 删除和后继不相邻, 处理后继的后事
        if (sParent != p) {                
            shift(sParent, s, s.right); // 不可能有左孩子:因为节点有右子树,此时后继节点即为右于树的最小值
            s.right = p.right;//顶上去的后继节点的右孩子==被删除节点的右孩子
        }
        // 4.3 后继取代被删除节点
        shift(parent, p, s);
        s.left = p.left;//shift方法只改变了父类节点的左右孩子的指向,而没有改变你后继节点的做左右孩子的指向
    }
    return p.value;
}

/**
 * 托孤方法
 *
 * @param parent  被删除节点的父亲
 * @param deleted 被删除节点
 * @param child   被顶上去的节点
 */
// 只考虑让 n1父亲的左或右孩子指向 n2, n1自己的左或右孩子并未在方法内改变
private void shift(BSTNode parent, BSTNode deleted, BSTNode child) {
    //情况讨论:当被删除节点就是根节点,而根节点是没有父亲的
    if (parent == null) {
        root = child;//你们被删除节点的子节点就成根节点
    } else if (deleted == parent.left) {
        parent.left = child;
    } else {
        parent.right = child;
    }
}

递归实现

  /**
     * <h3>根据关键字删除</h3>
     *
     * @param key 关键字
     * @return 被删除关键字对应值  是删除单个节点,当该节点被删除,则它的子节点将会与该节点的父类节点进行连接
     */

    public Object remove(int key) {
        ArrayList<Object> result = new ArrayList<>(); // 保存被删除节点的值
        root = doRemove(root, key, result);
        return result.isEmpty() ? null : result.get(0);
    }


        /*
              4
             / \
            2   6
           /     \
          1       7

        node 起点
        返回值 删剩下的孩子(找到) 或 null(没找到)
     */
    //递归实现删除操作  BSTNode node:我要删除时,从哪个节点开始删除  node是起点
    private BSTNode doRemove(BSTNode node, int key, ArrayList<Object> result) {
        //1.没有找到的情况:
        if (node == null) {
            return null;
        }
        //2.找到的情况:
        if (key < node.key) {//向左查找
            node.left = doRemove(node.left, key, result);
            return node;
        }
        if (node.key < key) {//向右查找
            node.right = doRemove(node.right, key, result);
            return node;
        }
        result.add(node.value);
        if (node.left == null) { // 情况1 - 只有右孩子
            return node.right;
        }
        if (node.right == null) { // 情况2 - 只有左孩子
            return node.left;
        }
        //s:后继节点
        BSTNode s = node.right; // 情况3 - 有两个孩子
        while (s.left != null) {
            s = s.left;
        }
        //while循环结束以后,找到了后继节点
        s.right = doRemove(node.right, s.key, new ArrayList<>());
        s.left = node.left;
        return s;
    }

对于第四种情况进行代码解析

当D与S紧邻:

当D与S不是紧邻:

对于remove方法的解析:

说明

  1. ArrayList<Object> result 用来保存被删除节点的值

  2. 第二、第三个 if 对应没找到的情况,继续递归查找和删除,注意后续的 doDelete 返回值代表删剩下的,因此需要更新

  3. 最后一个 return 对应删除节点只有一个孩子的情况,返回那个不为空的孩子,待删节点自己因没有返回而被删除

  4. 第四个 if 对应删除节点有两个孩子的情况,此时需要找到后继节点,并在待删除节点的右子树中删掉后继节点,最后用后继节点替代掉待删除节点返回,别忘了改变后继节点的左右指针

范围查询

下面三种题型的核心解题思路:

我们利用中序遍历的特性:遍历出来的都是升序的结果来进行范围查询

找小的

    /*
                 4
               /   \
              2     6
             / \   / \
            1   3 5   7
     */
//找 > key 的所有 value
//解题思路:我们利用中序遍历的特性:遍历出来的都是升序的结果
public List<Object> less(int key) { //当我们输入的是6
    //result:将符合条件的加入到result中
    ArrayList<Object> result = new ArrayList<>();
    //中序遍历过程:
    BSTNode p = root;
    LinkedList<BSTNode> stack = new LinkedList<>();
    while (p != null || !stack.isEmpty()) {
        if (p != null) {
            stack.push(p);
            p = p.left;
        } else {
            BSTNode pop = stack.pop();
            if (pop.key < key) {
                result.add(pop.value);
            } else {
                //当我们遇到比key大的分支时,该分支的子分支就没必要多此一举的进行判断,直接跳出
                //比如:当key=6,那么我们6右节点就不需要多此一举的去判断,而是直接跳出
                break;
            }
            p = pop.right;
        }
    }
    return result;
}

找大的

方法与《找小的》操作类似

第一种方法:

    /*
                 4
               /   \
              2     6
             / \   / \
            1   3 5   7
     */
//解题思路:我们利用中序遍历的特性:遍历出来的都是升序的结果
public List<Object> greater(int key) {
    ArrayList<Object> result = new ArrayList<>();
    BSTNode p = root;
    LinkedList<BSTNode> stack = new LinkedList<>();
    while (p != null || !stack.isEmpty()) {
        if (p != null) {
            stack.push(p);
            p = p.left;
        } else {
            BSTNode pop = stack.pop();
            if (pop.key > key) {//在这里,我们遍历>key的就不需要break,让它执行到子节点为Null
                result.add(pop.value);
            }
            p = pop.right;
        }
    }
    return result;
}

但这样效率不高,可以用 RNL 遍历

什么是RNL 遍历?

答:

注:

前三中就是我们之前所讲的前、中、后序遍历 N:值 L:左 R:右

  • Pre-order, NLR

  • In-order, LNR

  • Post-order, LRN

以下三种遍历与上三种遍历的区别:上三种是先左后右,下三种是先右后左

  • Reverse pre-order(反向前序遍历), NRL

  • Reverse in-order(中向前序遍历), RNL

  • Reverse post-order(反向后序遍历), RLN

第二种方法:

public List<Object> greater(int key) {
    ArrayList<Object> result = new ArrayList<>();
    BSTNode p = root;
    //RNL 遍历:得到的是降序的结果
    LinkedList<BSTNode> stack = new LinkedList<>();
    while (p != null || !stack.isEmpty()) {
        if (p != null) {
            stack.push(p);
            p = p.right;
        } else {
            BSTNode pop = stack.pop();
            if (pop.key > key) {
                result.add(pop.value);
            } else {
                break;
            }
            p = pop.left;
        }
    }
    return result;
}

找之间

方法与《找小的》操作类似

public List<Object> between(int key1, int key2) {
    ArrayList<Object> result = new ArrayList<>();
    BSTNode p = root;
    LinkedList<BSTNode> stack = new LinkedList<>();
    while (p != null || !stack.isEmpty()) {
        if (p != null) {
            stack.push(p);
            p = p.left;
        } else {
            BSTNode pop = stack.pop();
            if (pop.key >= key1 && pop.key <= key2) {
                result.add(pop.value);
            } else if (pop.key > key2) {
                break;
            }
            p = pop.right;
        }
    }
    return result;
}

小结

优点:

  1. 如果每个节点的左子树和右子树的大小差距不超过一,可以保证搜索操作的时间复杂度是 O(log n),效率高。

  2. 插入、删除结点等操作也比较容易实现,效率也比较高。

  3. 对于有序数据的查询和处理,二叉查找树非常适用,可以使用中序遍历得到有序序列。

缺点:

  1. 如果输入的数据是有序或者近似有序的,就会出现极度不平衡的情况,可能导致搜索效率下降,时间复杂度退化成O(n)。

  2. 对于频繁地插入、删除操作,需要维护平衡二叉查找树,例如红黑树、AVL 树等,否则搜索效率也会下降。

  3. 对于存在大量重复数据的情况,需要做相应的处理,否则会导致树的深度增加,搜索效率下降。

  4. 对于结点过多的情况,由于树的空间开销较大,可能导致内存消耗过大,不适合对内存要求高的场景。

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

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

相关文章

查看本机Arp缓存,以及清除arp缓存

查看Arp缓存目录 Windows 系统使用 winR&#xff0c;输入cmd 在命令窗口输入 arp -a 删除Arp缓存目录 在命令窗口输入 arp -d * 查看主机路由表

js中 for、forEach、for...in、for...of循环的区别和使用

前端面试题库 &#xff08;面试必备&#xff09; 推荐&#xff1a;★★★★★ 地址&#xff1a;前端面试题库 js中 for、forEach、for...in、for...of循环的区别 我们全部以以下数组举例 var arr [1,2,3,4,5];for循环 for(语句 1; 语句 2; 语句 3){} &#xf…

ThreeJS-3D教学八-OBJLoader模型加载+动画

先看效果图&#xff1a; 本篇给大家介绍下模型加载的知识&#xff0c;用到了OBJLoader对应的模型&#xff0c;为了增加趣味性&#xff0c;花了些时间&#xff0c;利用new THREE.Points获取到模型上的点&#xff0c;做了一个动画效果&#xff0c;其实就是操作 Y轴上的点&#x…

[已解决]java-sun.security.validator.ValidatorException: PKIX path building failed

找了好多文章,终于找到个解决办法! 报错详情内容 解决办法 第一种&#xff08;适用于本人解决办法&#xff09;&#xff1a; httpclient-4.5.jar 定时发送http包&#xff0c;忽然有一天报错&#xff0c;http证书变更引起的。 之前的代码 try {CloseableHttpClient httpCli…

多测师肖sir_高级金牌讲师_python之基本使用003

python之基本使用 一、基础使用 1、python中的打印方式 格式&#xff1a;print&#xff08;打印内容&#xff09; 案例&#xff1a;print&#xff08;12&#xff09; 注意点&#xff1a; 打印的对象中&#xff1a;数值可以直接引用&#xff0c;字母或中文要加上引号&#xff08…

Eclipse导入项目之后中文注释乱码

1 问题 Eclipse导入项目之后中文注释乱码。原因&#xff1a;中文乱码的原因是因为编码的关系 2 解决方法 记事本打开查看编码方式 修改eclipse编码方式 在Eclipse中更改文件的编码方式可以通过以下步骤进行&#xff1a; 打开Eclipse&#xff0c;并导航到要更改编码方式的…

量化交易全流程(七)

本节目录 资金分配 实盘交易 vn.py框架 我将重点介绍资金分配的基础模型和实现。当然&#xff0c;这里介绍的模型是最基础的模型&#xff0c;现实实践中往往并不能直接使用。因为后续我将加入机器学习和深度学习在量化交易领域中的应用。 现代 / 均值——方差资产组合理论…

高级工技能等级认定理论部分 看了就过关

4 理论一_职业道德 4.1职业道德基本知识 4.1.1练习 4.2职业守则 4.2.1练习1 4.2.1练习2 5 理论二 _基础知识 5.1 法律责任 5.1.1 练习1 5.1.2 练习2 5.1.3 练习3 5.2 基础知识 5.2.1 练习1 5.2.2 练习2 5.2.3 练习3 5.2.4 练习4 6 理论三_网络与信息安全防护 6.1 网络相…

部署企业级ChatGPT,将AI整合进工作

引言 3月份AI应用大爆发催生了国内大量需求。 然而&#xff0c;所有的需求都不可避免得遇到很多非技术性的问题&#xff1a; 部署开源模型的成本巨大&#xff0c;且效果成谜&#xff0c;65B的模型推理应用最少需要130G显存&#xff0c;而微调训练则需要额外添加8倍的资源。 …

基于YOLO的BIM对象检测

我在此过程中使用的 BIM 数据集取自澳大利亚卫生设施指南。 该数据集包含一组房间数据表和房间布局表&#xff0c;旨在提供典型房间类型的合规示例&#xff0c;并减少规划和设计这些房间时“重新发明轮子”的需要。 推荐&#xff1a;用 NSDT编辑器 快速搭建可编程3D场景 1、合…

23.5 Bootstrap 框架6

1. 表单布局 部分表单类名介绍: * 1. form-label: 表单标签样式类, 用于定义表单的标签样式. * 2. form-control: 表单控件样式类, 用于定义输入框, 文本域等表单元素的样式.表单元素<input>, <textarea>, <select>在使用.form-control类的情况下, 宽度都是…

取消激光雷达/不升级Orin,小鹏P5改款背后的行业「痛点」

9月25日&#xff0c;小鹏汽车正式发布了旗下改款车型—2024款小鹏P5&#xff08;15.69-17.49万元&#xff09;。车型精简、降本增效&#xff08;减配&#xff09;成为新亮点。 在配置变化方面&#xff0c;智驾成为牺牲品。其中&#xff0c;高配Pro车型继续保留英伟达Xavier&…

Cpolar内网穿透工具在windows和Linux上具体使用

Cpolar内网穿透工具在windows和Linux上具体使用 一、Linux上部署的项目通过内网穿透实现外网访问项目二、Windows上部署的项目通过内网穿透实现外网访问项目 一、Linux上部署的项目通过内网穿透实现外网访问项目 一个免费的内网穿透方式&#xff0c;简单方便。 网址&#xff1a…

MyBatisPlus(十六)逻辑删除

说明 实际生产中的数据&#xff0c;一般不采用物理删除&#xff0c;而采用逻辑删除&#xff0c;也就是将一条记录的状态改为已删除。 逻辑删除&#xff0c;本质上是更新操作。 MyBatis Plus 框架&#xff0c;提供了逻辑删除功能。在配置了逻辑删除后&#xff0c;增删改查和统…

设计模式 - 状态模式

目录 一. 前言 二. 实现 一. 前言 状态模式&#xff08;State Pattern&#xff09;&#xff1a;它主要用来解决对象在多种状态转换时&#xff0c;需要对外输出不同的行为的问题。状态和行为是一一对应的&#xff0c;状态之间可以相互转换。当一个对象的内在状态改变时&#x…

超越React,JS代码体积减少90%!它为何是2023年最好的Web框架?

说到Web框架&#xff0c;大家最先想到的可能是 Vue、React&#xff0c;或者是Next.js。但不得不提&#xff0c;有个后起之秀“来势汹汹”&#xff0c;1.0版本发布至今仅一年&#xff0c;就出尽风头。它就是Astro。 Astro 是什么&#xff1f;一个现代化的静态站点生成器和前端框…

nginx反向代理实例

一、代理模式 如果域名没有备案&#xff0c;访问国内的云主机时&#xff0c;会被防火墙拦截&#xff0c;但是如果先解析到香港主机&#xff0c;然后反向代理到国内的云主机&#xff0c;就可以绕过备案访问了。 香港服务器可以在亿速云购买&#xff0c;域名可以在阿里云购买&a…

3.3.OpenCV技能树--二值图像处理--图像形态学操作

文章目录 1.图像形态学运算简介2.图像开运算处理2.1.图像开运算处理简介2.2.图像开运算处理代码2.3.图像开运算处理效果 3.图像闭运算处理3.1.图像闭运算处理简介3.2.图像闭运算处理代码3.3.图像闭运算处理效果 4.图像形态学梯度处理4.1.图像形态学梯度处理简介4.2.图像形态学梯…

网康 NS-ASG安全网关存在远程命令执行漏洞 复现

文章目录 网康 NS-ASG安全网关存在远程命令执行漏洞 复现0x01 前言0x02 漏洞描述0x03 影响版本0x04 漏洞环境0x05 漏洞复现1.访问漏洞环境2.构造POC3.复现 0x06 修复建议 网康 NS-ASG安全网关存在远程命令执行漏洞 复现 0x01 前言 免责声明&#xff1a;请勿利用文章内的相关技…

基于SpringBooy的安康旅游网站的设计与实现

目录 前言 一、技术栈 二、系统功能介绍 登录模块的实现 景点信息管理界面 酒店信息管理界面 特产管理界面 游客管理界面 景点购票订单管理界面 系统主界面 游客注册界面 景点信息详情界面 酒店详情界面 特产详情界面 三、核心代码 1、登录模块 2、文件上传模块…