多线程与高并发——并发编程(6)

news2025/1/23 13:42:53

文章目录

  • 六、并发集合
    • 1 ConcurrentHashMap
      • 1.1 存储结构
      • 1.2 存储操作
        • 1.2.1 put方法
        • 1.2.2 putVal方法-散列算法
        • 1.2.3 putVal方法-添加数据到数组&初始化数组
        • 1.2.4 putVal方法-添加数据到链表
      • 1.3 扩容操作
        • 1.3.1 treeifyBin方法触发扩容
        • 1.3.2 tryPresize方法-针对putAll的初始化操作
        • 1.3.3 tryPreSize方法-计算扩容戳&查看BUG
        • 1.3.4 tryPreSize方法-对sizeCtl的修改&条件判断的BUG
        • 1.3.5 transfer方法-计算每个线程迁移的长度
        • 1.3.6 transfer方法-构建新数组&查看标识属性
        • 1.3.7 transfer方法-线程领取迁移任务
        • 1.3.8 transfer方法-迁移结束操作
        • 1.3.9 transfer方法-迁移数据(链表)
        • 1.3.10 helpThransfer方法-协助扩容
      • 1.4 红黑树操作
        • 1.4.1 什么是红黑树
        • 1.4.2 treeifyBin方法-封装TreeNode和双向链表
        • 1.4.3 TreeBin有参构造-双向链表转为红黑树
        • 1.4.4 balanceInsertion方法-保证红黑树平衡以及特性
        • 1.4.5 putTreeVal方法-添加节点
        • 1.4.6 TreeBin的锁操作
        • 1.4.7 transfer迁移数据
      • 1.5 查询数据
        • 1.5.1 get方法-查询数据的入口
        • 1.5.2 ForwardingNode的find方法
        • 1.5.3 ReservationNode的find方法
        • 1.5.4 TreeBin的find方法
        • 1.5.5 TreeNode的findTreeNode方法
      • 1.6 ConcurrentHashMap其它方法
        • 1.6.1 compute方法
        • 1.6.2 compute方法源码分析
        • 1.6.3 computeIfPresent、computeIfAbsent、compute的区别
        • 1.6.4 replace方法详解
        • 1.6.5 merge方法详解
      • 1.7 ConcurrentHashMap计数器
        • 1.7.1 addCount方法分析
        • 1.7.2 size方法分析
      • 1.8 JDK1.7的HashMap的环形链表问题
    • 2 CopyOnWriteArrayList
      • 2.1 CopyOnWriteArrayList介绍
      • 2.2 核心属性&方法
      • 2.3 读操作
      • 2.4 写操作
      • 2.5 移除数据
      • 2.6 覆盖数据&清空集合
      • 2.7 迭代器

六、并发集合

1 ConcurrentHashMap

1.1 存储结构

ConcurrentHashMap 是线程安全的 HashMap,在 JDK1.8 中是以 CAS + synchronized 实现的线程安全。

  • CAS:在没有 hash 冲突时(Node 要放在数组上时)
  • synchronized:在出现 hash 冲突时(Node 存放的位置已经有数据了)
  • 存储结构:数组+链表+红黑树

image.png

1.2 存储操作

1.2.1 put方法

public V put(K key, V value) {
   
    // 在调用put方法时,会调用putVal方法,第三个参数默认传递false
    // 在调用putIfAbsent时,会调用putVal方法,第三个参数传递true
    // false: 代表key一致时,直接覆盖数据
    // true: 代表key一致时,什么都不做,key不存在正常添加(类似Redis的setnx)
    return putVal(key, value, false);
}

1.2.2 putVal方法-散列算法

final V putVal(K key, V value, boolean onlyIfAbsent) {
   
    // ConcurrentHashMap不允许key或者value出现为null的值,跟HashMap的区别
    if (key == null || value == null) throw new NullPointerException();
    // 根据key的hashCode计算出一个hash值,后期得出当前key-value要存储在哪个数组索引位置
    int hash = spread(key.hashCode());
    int binCount = 0; // 一个标识,在后面有用
    // ...省略大量代码
}
// 计算当前Node的hash值的方法
static final int spread(int h) {
   
    // 将key的hashCode值的高低16位进行^运算,最终又与HASH_BITS进行了&运算
    // 将高位的hash也参与到计算索引位置的运算当中,尽可能将数据打散
    // 为什么HashMap、ConcurrentHashMap,都要求数组长度为2^n
    // HASH_BITS让hash值的最高位符号位肯定为0,代表当前hash值默认情况下一定是正数,因为hash值为负数时,有特殊的含义
    // static final int MOVED     = -1; // 代表当前hash位置的数据正在扩容
    // static final int TREEBIN   = -2; // 代表当前hash位置下挂载的是一个红黑树
    // static final int RESERVED  = -3; // 预留当前索引位置
    return (h ^ (h >>> 16)) & HASH_BITS;
    // 计算数组放到哪个索引位置的方法   (f = tabAt(tab, i = (n - 1) & hash)
    // n:是数组的长度
}
运算方式
00000000 00000000 00000000 00001111  - 15 (n - 1)
&
(
    (
        00001101 00001101 00101111 10001111  - h
        ^
        00000000 00000000 00001101 00001101  - h >>> 16
    )
    &
    01111111 11111111 11111111 11111111  - HASH_BITS
)

1.2.3 putVal方法-添加数据到数组&初始化数组

  • 添加数据到数组:CAS
  • 初始化数组:DCL + CAS
final V putVal(K key, V value, boolean onlyIfAbsent) {
   
    // 省略部分代码...
    // 将Map的数组赋值给tab,死循环
    for (Node<K,V>[] tab = table;;) {
   
        // n: 数组长度;i: 当前Node需要存放的索引位置
        // f: 当前数组i索引位置的Node对象;fn: 当前数组i索引位置上数据的hash值
        Node<K,V> f; int n, i, fh;
        // 判断当前数组是否还没有初始化
        if (tab == null || (n = tab.length) == 0)
            tab = initTable();	// 将数组进行初始化
        // 基于 (n - 1) & hash 计算出当前Node需要存放在哪个索引位置
        // 基于tabAt获取到i位置的数据
        else if ((f = tabAt(tab, i = (n - 1) & hash)) == null) {
   
            // 现在数组的i位置上没有数据,基于CAS的方式将数据存在i位置上
            if (casTabAt(tab, i, null, new Node<K,V>(hash, key, value, null)))
                break; // 如果成功,执行break跳出循环,插入数据成功
        }
        // 判断当前位置数据是否正在扩容
        else if ((fh = f.hash) == MOVED)
            tab = helpTransfer(tab, f);	// 让当前插入数据的线程协助扩容
        // 省略部分代码...
    }
    addCount(1L, binCount);
    return null;
}
// 初始化数组方法
private final Node<K,V>[] initTable() {
   
    Node<K,V>[] tab; int sc;
    // 再次判断数组没有初始化,并且完成tab的赋值
    while ((tab = table) == null || tab.length == 0) {
   
        // sizeCtl:是数组在初始化和扩容操作时的一个控制变量。
        // -1: 代表当前数组正在初始化;
        // 小于-1: 低16位代表当前数组正在扩容的线程个数(如果1个线程扩容,值为-2,如果2个线程扩容,值为-3);
        // 0: 代表数组还没初始化;
        // 大于0: 代表当前数组的扩容阈值,或者是当前数组的初始化大小
        // 将sizeCtl赋值给sc变量,并判断是否小于0
        if ((sc = sizeCtl) < 0)
            Thread.yield(); // lost initialization race; just spin
        // 可以尝试初始化数组,线程会以CAS的方式,将sizeCtl修改为-1,代表当前线程可以初始化数组
        else if (U.compareAndSwapInt(this, SIZECTL, sc, -1)) {
   
            try {
   	// 尝试初始化
                // 再次判断当前数组是否已经初始化完毕
                if ((tab = table) == null || tab.length == 0) {
   
                    // 开始初始化: 如果sizeCtl > 0,就初始化sizeCtl长度的数组;如果sizeCtl == 0,就初始化默认的长度16
                    int n = (sc > 0) ? sc : DEFAULT_CAPACITY;
                    // 初始化数组
                    Node<K,V>[] nt = (Node<K,V>[])new Node<?,?>[n];
                    // 将初始化的数组nt,赋值给tab和table
                    table = tab = nt;
                    // sc赋值为了数组长度 - 数组长度 右移 2位    16 - 4 = 12,将sc赋值为下次扩容的阈值
                    sc = n - (n >>> 2);
                }
            } finally {
   
                // 将赋值好的sc,设置给sizeCtl
                sizeCtl = sc;
            }
            break;
        }
    }
    return tab;
}

1.2.4 putVal方法-添加数据到链表

  • 添加数据到链表:利用 synchronized 基于当前索引位置的Node,作为锁对象
final V putVal(K key, V value, boolean onlyIfAbsent) {
   
    // 省略部分代码...
    int binCount = 0;
    for (Node<K,V>[] tab = table;;) {
   
        // n: 数组长度;i: 当前Node需要存放的索引位置
        // f: 当前数组i索引位置的Node对象;fn: 当前数组i索引位置上数据的hash值
        Node<K,V> f; int n, i, fh;
        // 省略部分代码...
        else {
   
            V oldVal = null;	// 声明变量为oldVal
            synchronized (f) {
   	// 基于当前索引位置的Node,作为锁对象
                // 判断当前位置的数据还是之前的f么……(避免并发操作的安全问题)
                if (tabAt(tab, i) == f) {
   
                    if (fh >= 0) {
   	// 再次判断hash值是否大于0(不是树)
                        // binCount设置为1(在链表情况下,记录链表长度的一个标识)
                        binCount = 1;
                        // 死循环,每循环一次,对binCount
                        for (Node<K,V> e = f;; ++binCount) {
    
                            K ek;
                            // 当前i索引位置的数据,是否和当前put的key的hash值一致
                            if (e.hash == hash &&
                                // 如果当前i索引位置数据的key和put的key == 返回为true
                                // 或者equals相等
                                ((ek = e.key) == key || (ek != null && key.equals(ek)))) {
   
                                // key一致,可能需要覆盖数据,当前i索引位置数据的value赋值给oldVal
                                oldVal = e.val;
                                // 如果传入的是false,代表key一致,覆盖value;如果传入的是true,代表key一致,什么都不做
                                if (!onlyIfAbsent)
                                    e.val = value; // 覆盖value
                                break;
                            }
                            Node<K,V> pred = e;	// 拿到当前指定的Node对象
                            // 将e指向下一个Node对象,如果next指向的是一个null,可以挂在当前Node下面
                            if ((e = e.next) == null) {
   
                                // 将hash,key,value封装为Node对象,挂在pred的next上
                                pred.next = new Node<K,V>(hash, key, value, null);
                                break;
                            }
                        }
                    }
                    // 省略部分代码...
                }
            }
            if (binCount != 0) {
   
                if (binCount >= TREEIFY_THRESHOLD)	// binCount是否大于8(链表长度是否 >= 8)
                    // 尝试转为红黑树或者扩容
                    // 基于treeifyBin方法和上面的if判断,可以得知链表想要转为红黑树,必须保证数组长度大于等于64,并且链表长度大于等于8
                    // 如果数组长度没有达到64的话,会首先将数组扩容
                    treeifyBin(tab, i);
                if (oldVal != null)	// 如果出现了数据覆盖的情况,返回之前的值
                    return oldVal;
                break;
            }
        }
    }
    // 省略部分代码...
}

为什么链表长度为8转换为红黑树,不是能其他数值嘛?

因为泊松分布

The main disadvantage of per-bin locks is that other update
 * operations on other nodes in a bin list protected by the same
 * lock can stall, for example when user equals() or mapping
 * functions take a long time.  However, statistically, under
 * random hash codes, this is not a common problem.  Ideally, the
 * frequency of nodes in bins follows a Poisson distribution
 * (http://en.wikipedia.org/wiki/Poisson_distribution) with a
 * parameter of about 0.5 on average, given the resizing threshold
 * of 0.75, although with a large variance because of resizing
 * granularity. Ignoring variance, the expected occurrences of
 * list size k are (exp(-0.5) * pow(0.5, k) / factorial(k)). The
 * first values are:
 *
 * 0:    0.60653066
 * 1:    0.30326533
 * 2:    0.07581633
 * 3:    0.01263606
 * 4:    0.00157952
 * 5:    0.00015795
 * 6:    0.00001316
 * 7:    0.00000094
 * 8:    0.00000006
 * more: less than 1 in ten million

1.3 扩容操作

1.3.1 treeifyBin方法触发扩容

// 在链表长度大于等于8时,尝试将链表转为红黑树
private final void treeifyBin(Node<K,V>[] tab, int index) {
   
    Node<K,V> b; int n, sc;
    // 数组不能为空
    if (tab != null) {
   
        // 数组的长度n,是否小于64
        if ((n = tab.length) < MIN_TREEIFY_CAPACITY)
            // 如果数组长度小于64,不能将链表转为红黑树,先尝试扩容操作
            tryPresize(n << 1);
        // 省略部分代码……
    }
}

1.3.2 tryPresize方法-针对putAll的初始化操作

// size是将之前的数组长度 左移 1位得到的结果
private final void tryPresize(int size) {
   
    // 如果扩容的长度达到了最大值,就使用最大值,否则需要保证数组的长度为2的n次幂
    // 这块的操作,是为了初始化操作准备的,因为调用putAll方法时,也会触发tryPresize方法
    // 如果刚刚new的ConcurrentHashMap直接调用了putAll方法的话,会通过tryPresize方法进行初始化
    int c = (size >= (MAXIMUM_CAPACITY >>> 1)) ? MAXIMUM_CAPACITY :
        tableSizeFor(size + (size >>> 1) + 1);
    // 这些代码和initTable一模一样
    int sc;
    // 将sizeCtl的值赋值给sc,并判断是否大于0,这里代表没有初始化操作,也没有扩容操作
    while ((sc = sizeCtl) >= 0) {
   
        // 将ConcurrentHashMap的table赋值给tab,并声明数组长度n
        Node<K,V>[] tab = table; int n;
        // 数组是否需要初始化
        if (tab == null || (n = tab.length) == 0) {
   
            // 进来执行初始化
            // sc是初始化长度,初始化长度如果比计算出来的c要大的话,直接使用sc,如果没有sc大,说明sc无法容纳下putAll中传入的map,使用更大的数组长度
            n = (sc > c) ? sc : c;
            // 设置sizeCtl为-1,代表初始化操作
            if (U.compareAndSwapInt(this, SIZECTL, sc, -1)) {
   
                try {
   
                    // 再次判断数组的引用有没有变化
                    if (table == tab) {
   
                        // 初始化数组
                        Node<K,V>[] nt = (Node<K,V>[])new Node<?,?>[n];
                        // 数组赋值
                        table = nt;
                        // 计算扩容阈值
                        sc = n - (n >>> 2);
                    }
                } finally {
   
                    // 最终赋值给sizeCtl
                    sizeCtl = sc;
                }
            }
        }
        // 如果计算出来的长度c小于等于c,或者数组长度大于等于最大长度,直接退出循环结束方法
        else if (c <= sc || n >= MAXIMUM_CAPACITY)
            break;
        // 省略部分代码...
    }
}

// 将c这个长度设置到最近的2的n次幂的值,   15 -> 16     17 -> 32
// c == size + (size >>> 1) + 1
// size = 17
00000000 00000000 00000000 00010001
+ 
00000000 00000000 00000000 00001000
+
00000000 00000000 00000000 00000001
// c = 26
00000000 00000000 00000000 00011010
private static final int tableSizeFor(int c) {
    // c = 26
    // 00000000 00000000 00000000 00011001
    int n = c - 1;
    // 00000000 00000000 00000000 00011001
    // 00000000 00000000 00000000 00001100
    // 00000000 00000000 00000000 00011101
    n |= n >>> 1;
    // 00000000 00000000 00000000 00011101
    // 00000000 00000000 00000000 00000111
    // 00000000 00000000 00000000 00011111
    n |= n >>> 2;
    // 00000000 00000000 00000000 00011111
    // 00000000 00000000 00000000 00000001
    // 00000000 00000000 00000000 00011111
    n |= n >>> 4;
    // 00000000 00000000 00000000 00011111
    // 00000000 00000000 00000000 00000000
    // 00000000 00000000 00000000 00011111
    n |= n >>> 8;
    // 00000000 00000000 00000000 00011111
    n |= n >>> 16;
    // 00000000 00000000 00000000 00100000
    return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
}

1.3.3 tryPreSize方法-计算扩容戳&查看BUG

private final void tryPresize(int size) {
   
    // n:数组长度
    while ((sc = sizeCtl) >= 0) {
   
        // 省略部分代码…
        // 判断当前的tab是否和table一致
        else if (tab == table) {
   
            // 计算扩容标识戳,根据当前数组的长度计算一个16位的扩容戳
            // 第一个作用是为了保证后面的sizeCtl赋值时,保证sizeCtl为小于-1的负数
            // 第二个作用用来记录当前是从什么长度开始扩容的
            int rs = resizeStamp(n);
            // BUG --- sc < 0,永远进不去
            if (sc < 0) {
    // 如果sc小于0,代表有线程正在扩容
                // 省略部分代码……协助扩容的代码(进不来~~~~)
            }
            // 代表没有线程正在扩容,我是第一个扩容的。
            else if (U.compareAndSwapInt(this, SIZECTL, sc,
                                         (rs << RESIZE_STAMP_SHIFT) + 2))
                // 省略部分代码……第一个扩容的线程……
        }
    }
}
// 计算扩容标识戳
// 32 =  00000000 00000000 00000000 00100000
// Integer.numberOfLeadingZeros(32) = 26
// 1 << (RESIZE_STAMP_BITS - 1) 
// 00000000 00000000 10000000 00000000
// 00000000 00000000 00000000 00011010
// 00000000 00000000 10000000 00011010
static final int resizeStamp(int n) {
   
    return Integer.numberOfLeadingZeros(n) | (1 << (RESIZE_STAMP_BITS - 1));
}

1.3.4 tryPreSize方法-对sizeCtl的修改&条件判断的BUG

private final void tryPresize(int size) {
   
    // sc默认为sizeCtl
    while ((sc = sizeCtl) >= 0) {
   
        else if (tab == table) {
   
            // rs: 扩容戳  00000000 00000000 10000000 00011010
            int rs = resizeStamp(n);
            if (sc < 0) {
   
                // 说明有线程正在扩容,过来帮助扩容
                Node<K,V>[] nt;
                // 依然有BUG
                // 当前线程扩容时,老数组长度是否和我当前线程扩容时的老数组长度一致
                // 00000000 00000000 10000000 00011010
                if ((sc >>> RESIZE_STAMP_SHIFT) != rs  
                    // 10000000 00011010 00000000 00000010 
                    // 00000000 00000000 10000000 00011010
                    // 这两个判断都是有问题的,核心问题就应该先将rs左移16位,再追加当前值
                    // 判断当前扩容是否已经即将结束
                    || sc == rs + 1   // sc == rs << 16 + 1 BUG
                    // 判断当前扩容的线程是否达到了最大限度
                    || sc == rs + MAX_RESIZERS   // sc == rs << 16 + MAX_RESIZERS BUG
                    // 扩容已经结束了
                    || (nt = nextTable) == null 
                    // 记录迁移的索引位置,从高位往低位迁移,也代表扩容即将结束
                    || transferIndex <= 0)
                    break;
                // 如果线程需要协助扩容,首先就是对sizeCtl进行+1操作,代表当前要进来一个线程协助扩容
                if (U.compareAndSwapInt(this, SIZECTL, sc, sc + 1))
                    // 上面的判断没进去的话,nt就代表新数组
                    transfer(tab, nt);
            }
            // 是第一个来扩容的线程
            // 基于CAS将sizeCtl修改为  10000000 00011010 00000000 00000010 
            // 将扩容戳左移16位之后,符号位是1,就代码这个值为负数,低16位在表示当前正在扩容的线程有多少个
            // 为什么低位值为2时,代表有一个线程正在扩容
            // 每一个线程扩容完毕后,会对低16位进行-1操作,当最后一个线程扩容完毕后,减1的结果还是-1,当值为-1时,要对老数组进行一波扫描,查看是否有遗漏的数据没有迁移到新数组
            else if (U.compareAndSwapInt(this, SIZECTL, sc,(rs << RESIZE_STAMP_SHIFT) + 2))
                // 调用transfer方法,并且将第二个参数设置为null,就代表是第一次来扩容!
                transfer(tab, null);
        }
    }
}

1.3.5 transfer方法-计算每个线程迁移的长度

// 开始扩容   tab: oldTable
private final void transfer(Node<K,V>[] tab, Node<K,V>[] nextTab) {
   
    // n: 数组长度
    // stride: 每个线程一次性迁移多少数据到新数组
    int n = tab.length, stride;
    // 基于CPU的内核数量来计算,每个线程一次性迁移多少长度的数据最合理
    // NCPU = 4
    // 举个栗子:数组长度为1024 - 512 - 256 - 128 / 4 = 32
    // MIN_TRANSFER_STRIDE = 16,为每个线程迁移数据的最小长度
    // 根据CPU计算每个线程一次迁移多长的数据到新数组,如果结果大于16,使用计算结果。 如果结果小于16,就使用最小长度16
    if ((stride = (NCPU > 1) ? (n >>> 3) / NCPU : n) < MIN_TRANSFER_STRIDE)
        stride = MIN_TRANSFER_STRIDE; 
    // 省略部分代码...
}

1.3.6 transfer方法-构建新数组&查看标识属性

// 以32长度数组扩容到64位例子
private final void transfer(Node<K,V>[] tab, Node<K,V>[] nextTab) {
   
    // 省略部分代码...
    // n: 老数组长度   32
    // stride: 步长   16
    // 第一个进来扩容的线程需要把新数组构建出来
    if (nextTab == null) {
   
        try {
   
            // 将原数组长度左移一位,构建新数组长度
            Node<K,V>[] nt = (Node<K,V>[])new Node<?,?>[n << 1];
            // 赋值操作
            nextTab = nt;
        } catch (Throwable ex) {
      
            // 到这说明已经达到数组长度的最大取值范围
            sizeCtl = Integer.MAX_VALUE;
            // 设置sizeCtl后直接结束
            return;
        }
        // 对成员变量的新数组赋值
        nextTable = nextTab;
        // 迁移数据时,用到的标识,默认值为老数组长度
        transferIndex = n;   // 32
    }
    // 新数组长度
    int nextn = nextTab.length;  // 64
    // 在老数组迁移完数据后,做的标识
    ForwardingNode<K,V> fwd = new ForwardingNode<K,V>(nextTab);
    // 迁移数据时,需要用到的标识
    boolean advance = true;
    boolean finishing = false; 
    // 省略部分代码...
}

1.3.7 transfer方法-线程领取迁移任务

// 以32长度扩容到64位为例子
private final void transfer(Node<K,V>[] tab, Node<K,V>[] nextTab) {
   
    // 省略部分代码…
    // n: 32
    // stride: 16
    int n = tab.length, stride;
    if (nextTab == null) {
    
        // 省略部分代码…
        nextTable = nextTab;	// 新数组
        // transferIndex:0
        transferIndex = n;
    }
    // nextn:64
    int nextn = nextTab.length;
    ForwardingNode<K,V> fwd = new ForwardingNode<K,V>(nextTab);
    // advance:true,代表当前线程需要接收任务,然后再执行迁移;如果为false,代表已经接收完任务
    boolean advance = true;
    boolean finishing = false; // 是否迁移结束
    // i = 15     代表当前线程迁移数据的索引值
    for (int i = 0, bound = 0;;) {
   
        Node<K,V> f; int fh;	// f = null,fh = 0
        while (advance) {
   		// 当前线程要接收任务
            // nextIndex = 16,nextBound = 16
            int nextIndex, nextBound;
            // 对i进行--,并且判断当前任务是否处理完毕!
            if (--i >= bound || finishing) // 第一次进来,这两个判断肯定进不去
                advance = false;
            // 判断transferIndex是否小于等于0,代表没有任务可领取,结束了
            // 在线程领取任务会,会对transferIndex进行修改,修改为transferIndex - stride
            // 在任务都领取完之后,transferIndex肯定是小于等于0的,代表没有迁移数据的任务可以领取
            else if ((nextIndex = transferIndex) <= 0) {
   
                i = -1

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

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

相关文章

VMware17 下载以及 配置虚拟机 一条龙全教程

前言&#xff1a;在网络安全相关方面的学习时&#xff0c;我们难免要使用虚拟机来模拟实现网络攻击或者防御&#xff0c;在这里&#xff0c;笔者就给大家分享 VMware 17 虚拟机下载安装使用的保姆级教程 目录 虚拟性软件版本推荐 下载 VMware 下载镜像 配置虚拟机 虚拟性软…

系统软件启动过程

实验一&#xff1a;系统软件启动过程 参考 重要文件 调用顺序 1. boot/bootasm.S | bootasm.asm&#xff08;修改了名字&#xff0c;以便于彩色显示&#xff09;a. 开启A20 16位地址线 实现 20位地址访问 芯片版本兼容通过写 键盘控制器8042 的 64h端口 与 60h端口。b.…

数学的魅力

数学的魅力 数学的历史古代数学古希腊数学中世纪数学文艺复兴数学 数学的分支1. 代数学2. 几何学3. 微积分学4. 概率论与统计学5. 数论 数学的重要性1. 科学和技术2. 经济学和金融3. 医学和生物学4. 社会科学5. 环境科学 数学的未来1. 人工智能2. 网络安全3. 空间探索 结论 数学…

Spring与Docker:如何容器化你的Spring应用

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

BI技巧丨Window应用之移动平均

Window函数白茶之前介绍过&#xff0c;可以用来处理同环比问题&#xff0c;因为其函数内部特性&#xff0c;我们还可以用其来处理移动平均问题。 先来看看本期的案例数据&#xff1a; 案例数据非常的简单&#xff0c;随机模拟一下销售挂蛋的情况即可。 将其导入到PowerBI中&…

YOLO目标检测——棉花病虫害数据集+已标注txt格式标签下载分享

实际项目应用&#xff1a;目标检测棉花病虫害数据集的应用场景涵盖了棉花病虫害的识别与监测、研究与防治策略制定、农业智能决策支持以及农业教育和培训等领域。这些应用场景可以帮助农业从业者更好地管理棉花病虫害&#xff0c;提高棉花产量和质量&#xff0c;推动农业的可持…

医院信息化、数字医学影像、DICOM、PACS源码

PACS系统适合卫生院、民营医院、二甲或以下公立医院的放射科、超声科使用。功能强大且简洁&#xff0c;性能优异&#xff0c;具备MPR&#xff08;三维重建&#xff09;、VR&#xff08;容积重建&#xff09;、胶片打印功能&#xff0c;能够快速部署。 PACS系统支持DR、CT、磁共…

SpringCloud-Config配置中心

接上文 SpringCloud-GetWay 路由网关 针对每个模块都要配置一个配置文件&#xff0c;例如当进行数据库修改时候&#xff0c;又有很多服务&#xff0c;则需要一个一个去修改。因此需要一种集中化的配置文件管理工具&#xff0c;集中得对配置文件进行配置。 1.部署配置中心服务…

PS基础-新建窗口工作区(保存合适自己的窗口布局)

保存自己的工作区 一&#xff0c;打开自己需要的窗口 2,打开的窗口拖拽到边&#xff0c;就可放置 3&#xff0c;保存自己习惯的工作区分布 下次打开时&#xff0c;点击保存时的图标就会看到自己命名的工作区

Shape Completion Enabled Robotic Grasping

摘要-这项工作提供了一个架构&#xff0c;使机器人能够通过形状完成抓取规划。形状完成是通过使用3D卷积神经网络(CNN)来完成的。该网络是在我们自己的新的开源数据集上训练的&#xff0c;该数据集包含了从不同视角捕获的超过44万个3D样本。运行时&#xff0c;从单个视角捕获的…

8.Xaml Border控件

1.运行图片 2.运行源码 <Grid Name="Grid1"><!--Border 里面只能有一个子元素--><!--BorderBrush="Red" 边框颜色-->

Nginx配置踩坑:一定注意location和proxy_pass的是否以“/”结尾

Nginx是开源、高性能、高可靠的Web和反向代理服务器&#xff0c;而且支持热部署&#xff0c;几乎可以做到 7 * 24 小时不间断运行&#xff0c;即使运行几个月也不需要重新启动&#xff0c;还能在不间断服务的情况下对软件版本进行热更新。性能是Nginx最重要的考量&#xff0c;其…

数据库的内连接和外连接的区别

内连接&#xff1a; 指连接结果仅包含符合连接条件的行&#xff0c;参与连接的两个表都应该符合连接条件。 1.等值连接 依据两个表中相同内容的字段进行连接 表TESTA,TESTB,TESTC&#xff0c;各有A, B两列。 内连接 内连接&#xff0c;即最常见的等值连接&#xff0c;例&…

机器学习实战-系列教程4:手撕线性回归3之多特征线性回归(项目实战、原理解读、源码解读)

&#x1f308;&#x1f308;&#x1f308;机器学习 实战系列 总目录 本篇文章的代码运行界面均在Pycharm中进行 本篇文章配套的代码资源已经上传 手撕线性回归1之线性回归类的实现 手撕线性回归2之单特征线性回归 手撕线性回归3之多特征线性回归 手撕线性回归4之非线性回归 8…

Vue中数据可视化关系图展示与关系图分析

Vue中数据可视化关系图展示与关系图分析 数据可视化是现代Web应用程序的重要组成部分之一&#xff0c;它可以帮助我们以图形的方式呈现和分析复杂的数据关系。Vue.js是一个流行的JavaScript框架&#xff0c;它提供了强大的工具来构建数据可视化应用。本文将介绍如何使用Vue.js…

基于python实现贪心算法、蛮力法、动态规划法解决分数背包问题和0-1背包问题(附完整源码下载)

背包问题算法设计 问题要求在一个物品集合中选择合适的物品放入背包&#xff0c;在放入背包中的物品总重量不超过背包容量的前提下&#xff0c;希望放入背包的物品总价值最大。根据是否允许部分物品放入背包的要求&#xff0c;背包问题可以分为【分数背包问题】和【0-1背包问题…

Spring Web Flow远程代码执行漏洞复现(CVE-2017-4971)

一、搭建环境 cd vulhub/spring/CVE-2017-4971 docker-compose up -d 影响版本:Spring Web Flow 2.4.0 ~ 2.4.4 触发条件: 1.MvcViewFactoryCreator对象的useSpringBeanBinding参数需要设置为false&#xff08;默认值&#xff09; 2. flow view对象中设置BinderConfiguration…

网页版QQ签到加速源码 QQ音乐等级加速源码 CF活动一键领取源码 QQ手游等级加速

QQ网页签到加速小工具PHP源码二次优化版 包含QQ空间功能 QQ空白昵称 QQ大会员签到 CF活动一键领取 清空QQ空间说说 QQ每日打卡加速 QQ空间删除说说 QQ手游等级加速 QQ微视等级加速 QQ音乐等级加速签到

【操作系统】24王道考研笔记——第四章 文件管理

第四章 文件管理 一、文件系统基础 1.基本概念 2.文件的逻辑结构 顺序文件&#xff1a; 索引文件&#xff1a; 索引顺序文件&#xff1a; 效率分析&#xff1a; 多级索引顺序文件&#xff1a; 总结&#xff1a; 3.文件目录 文件控制块&#xff08;FCB&#xff09; 目录的基本…

第三方软件测评报告怎么做?

第三方软件测试 总体来说&#xff0c;软件产品验收测试一般主要包括以下几个步骤&#xff1a; 1.制定测试计划&#xff0c;测试项&#xff0c;测试策略及验收通过准则&#xff0c;并经过客户参与的计划评审。 2.建立测试环境&#xff0c;设计测试用例&#xff0c;并经过评审…