第十三章:AQS

news2025/1/23 4:49:39

  • AQS 基础概念
  • 为什么 AQS 是 JUC 最重要的基石?
  • AQS 能干什么
  • AQS内部结构
  • AQS内部类Node
  • AQS 源码分析
    • 以 lock方法为入口讲解
      • nonfairTryAcquire 方法
      • addWaiter方法
        • 线程B
        • 线程C
      • acquireQueued 方法
        • B节点
        • C节点
    • unlock
    • cancelAcquire 方法
  • 总结

AQS 基础概念

AQS 全称:AbstractQueuedSynchronizer ,字面意思:抽象队列同步器

位于 java.util.concurrent.locks 包下:是一个抽象类

image-20221203214447679

AQS 是什么?

AQS 是用来实现锁或者其它同步器组件的公共基础部分的抽象实现,是重量级基础框架及整个JUC体系的基石

通过内置的FIFO队列来完成资源获取线程的排队工作,并通过一个int类变量表示持有锁的状态

队列的结构

image-20221203221307451

CLH:Craig、Landin and Hagersten 队列,是一个单向链表,AQS中的队列是CLH变体的虚拟双向队列FIFO

官方解释

image-20221203214918280

这么说有点抽象,举个例子

比如银行办理业务,一个窗口只能有一个人办理业务,此时其他人就必须在大厅中等待,这个等候大厅就相当于 队列 , 人 就相当于 队列中的线程。

image-20221203215632586

在通知下一个人办理业务时,我们都知道在银行中一般都有一个屏幕来显示 轮到xxx 号 办理业务

这个通知的屏幕就相当于 AQS 中的 state ,用来表示状态,比如 1表示有线程占用,0表示未占用。

image-20221203220146427

为什么 AQS 是 JUC 最重要的基石?

和 AQS 有关的锁

image-20221203221526700

在源码中的体现:

ReentrantLock

image-20221203221616749

CountDownLatch

image-20221203221631294

ReentrantReadWriteLock:

image-20221203221712202

Semaphore:

image-20221203221732907

从源码中也可以看出,几乎我们使用的锁都继承了这个AQS同步器,AQS 就像一个服务框架,定义通用的一些规则。

进一步理解锁和同步器的关系

  • 锁,面向锁的使用者
    • 定义了程序员和锁交互的使用层API,隐藏了实现细节,你调用即可。
  • 同步器,面向锁的实现者
    • 比如Java并发大神DougLee,提出统一规范并简化了锁的实现,屏蔽了同步状态管理、阻塞线程排队和通知、唤醒机制等。

AQS 能干什么

我们知道加锁就会导致阻塞,有阻塞就需要排队,排队必然就会用到队列。

抢到资源的线程直接使用处理业务,抢不到资源的必然涉及一种排队等候机制。抢占资源失败的线程继续去等待(类似银行业务办理窗口都满了,暂时没有受理窗口的顾客只能去候客区排队等候),但等候线程仍然保留获取锁的可能且获取锁流程仍在继续(候客区的顾客也在等着叫号,轮到了再去受理窗口办理业务)。

既然说到了排队等候机制,那么就一定会有某种队列形成,这样的队列是什么数据结构呢?

如果共享资源被占用,就需要一定的阻塞等待唤醒机制来保证锁分配。这个机制主要用的是CLH队列的变体实现的,将暂时获取不到锁的线程加入到队列中,这个队列就是AQS的抽象表现。它将请求共享资源的线程封装成队列的结点(Node),通过CAS、自旋以及LockSupport.park()的方式,维护state变量的状态,使并发达到同步的效果。

image-20221203222307808

AQS内部结构

image-20221203224204131

AQS使用一个volatile的int类型的成员变量来表示同步状态,通过内置的FIFO队列来完成资源获取的排队工作将每条要去抢占资源的线程封装成一个Node节点来实现锁的分配,通过CAS完成对State值的修改。

image-20221203223203581

  • AQS的同步状态State成员变量:类似于银行的业务窗口的状态,0 表示空闲状态,>= 1 表示有人占用。
  • AQS的CLH队列 :是一个虚拟的双向队列,想象成 银行的等待大厅。

image-20221203224539330

小总结

有阻塞就需要排队,实现排队必然需要队列

AQS 就是 state变量+CLH双端队列

AQS内部类Node

内部结构

image-20221203225252085

对应的属性说明

image-20221203225315704

Node的int变量 waitStatus:队列中其他线程的等待状态。一共分为四种:CANCELLED、SIGNAL、CONDITION、PROPAGATE

想象成银行等待大厅中等待的顾客的状态。

AQS 源码分析

AQS作为 JUC 的基石,几乎所有的类都继承了AQS,本次分析以 ReentrantLock 为例。

以 lock方法为入口讲解

ReentrantLock 的架构图

Sync 为 ReentrantLock 中的内部类

image-20221205145553019

首先从构造器方法入手,ReentrantLock 可以实现公平锁和非公平锁。

对于非公平锁和公平锁提供了俩个类: NonfairSync、FairSync,这俩个类都继承了 Sync,同时Sync又继承了AQS类。


    /**
     * Creates an instance of {@code ReentrantLock}.
     * This is equivalent to using {@code ReentrantLock(false)}.
     */
    public ReentrantLock() {
        sync = new NonfairSync();
    }

    /**
     * Creates an instance of {@code ReentrantLock} with the
     * given fairness policy.
     *
     * @param fair {@code true} if this lock should use a fair ordering policy
     */
    public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }

而ReentrantLock 中的 lock 方法,实际上调用了 lock 接口中定义的 lock 方法。

image-20221205145943345

而Lock 接口中的方法在 NonfairSync、FairSync 类中有了不同的实现方式:

image-20221205150139047

在 FairSync 中 lock 方法的定义

image-20221205150225321

实际上调用了 AQS 中的 acquire 方法:

image-20221205150306563

而 acquire 方法中,调用了 AQS 类中的 tryAcquire 方法,而 AQS 中对于 tryAcquire 方法并没有定义具体的实现,而是下放到子类 FairSync、NonfairSync 中。这里就是 模板方法设计模式。AQS 中的方法相当于一个钩子,供子类进行重写:

image-20221205151202282

FairSync 实现了tryAcquire的具体方法逻辑:

经过这么多次的调用,实际上使用公平锁时,具体的实现方式在 FairSync 中的 tryAcquire方法中、

image-20221205151424340

在 NonfairSync 中 lock 方法的定义

NonfairSync 方法比 FairSync 多了一个 if 判断,compareAndSetState方法对 AQS 中的 同步状态 state 做判断。如果没有线程占用锁,也就是期望值为0,那么好,当前线程就占用,并且修改状态值。如果有线程占用,仍然执行 acquire 方法。

image-20221205150359653

acquire 方法同样也调用了 tryAcquire 方法。

image-20221205151737016

在 tryAcquire 中继续调用了 nonfairTryAcquire方法

image-20221205152357685

使用 非公平锁时,实际上的实现逻辑,在 Sync 中的 nonfairTryAcquire 方法中:

image-20221205152529931

总结

image-20221205152840513

可以明显看出公平锁与非公平锁的lock()方法唯一的区别就在于公平锁在获取同步状态时多了一个限制条件:
hasQueuedPredecessors()

image-20221205153310993

hasQueuedPredecessors是公平锁加锁时判断是否需要排序以及等待队列中是否存在有效节点的方法

公平锁:公平锁讲究先来先到,线程在获取锁时,如果这个锁的等待队列中已经有线程在等待,那么当前线程就会进入等待队列中;

非公平锁:不管是否有等待队列,如果可以获取锁,则立刻占有锁对象。也就是说队列的第一个排队线程在unpark(),之后还是需要竞争锁(存在线程竞争的情况下)

image-20221205153222401

源码重点分析

既然非公平锁和公平锁都会调用 acquire 方法, 那么重点就放在这个 acquire 方法中,acquire 方法中分成了三个流程走向:
image-20221205154423432

image-20221205155704144

nonfairTryAcquire 方法

以非公平锁的 nonfairTryAcquire 为例,公平锁的 tryAcquire 方法仅仅是多了一个hasQueuedPredecessors 方法判断。

当第一个线程A尝试占用锁时,其实在 lock 方法中的就已经占用成功了,修改了同步状态state的值,并设置占用锁的线程。也就是说第一个线程A并不会执行 acquire 方法,也就不会调用 nonfairTryAcquire

image-20221205162018312

而在第二个 线程B想要占用锁时,由于state已经被第一个线程A所修改,因此第二个线程B会执行 acquire 方法,最终调用 nonfairTryAcquire

image-20221205163114231

第二个线程B尝试获取锁失败,返回false ,取反为 true,下一步执行 addWaiter 则进行入队操作。

image-20221205162720588

addWaiter方法

线程B

前面我们说过,每一个等待的线程都会被封装成一个Node节点,就是在 addWaiter中封装的。

image-20221205191155119

enq 方法中进行入队操作:

第一次循环

    private Node enq(final Node node) {
        for (;;) {
            // tail = null
            Node t = tail;
            // t==null 条件成立
            if (t == null) { // Must initialize
                // 设置头结点,此时的头结点并不是节点B,而是一个虚拟节点,不保存任何信息
                if (compareAndSetHead(new Node()))
                    // 将尾结点指向头结点 参考图一
                    tail = head;
            } else {
                node.prev = t;
                if (compareAndSetTail(t, node)) {
                    t.next = node;
                    return t;
                }
            }
        }
    }

image-20221205194132721

双向链表中,第一个节点为虚节点(也叫哨兵节点),其实并不存储任何信息,只是占位。
真正的第一个有数据的节点,是从第二个节点开始的。

第二次循环:

    private Node enq(final Node node) {
        for (;;) {
            // 此时tail指向了虚拟节点,因此不为null
            Node t = tail;
            // t==null 条件不成立
            if (t == null) { // Must initialize
                if (compareAndSetHead(new Node()))
                    tail = head;
            } else {
                // 将节点B的前驱节点,指向虚拟节点
                node.prev = t;
                // 设置尾节点指向 节点 b
                if (compareAndSetTail(t, node)) {
                    // 尾节点的后驱节点指向节点B
                    t.next = node;
                    // 图示参考: 图二
                    return t;
                }
            }
        }
    }

此时线程B才算真正的入队成功

image-20221205194111984

线程C

加入此时又来一个线程 C,对象锁仍然被线程A占用这。

image-20221205192553039

acquire方法

image-20221205191723633

addWaiter方法:

由于节点B入队的时候,将队列已经初始化一次,不会再执行 enq 方法。图示参考图三

image-20221205192105697

image-20221205194044001

此时B节点 和 C 节点都已经入队成功,但是不能干等这啊,毕竟还要抢占对象锁,因此 在入队完之后就要执行 acquireQueued 方法

acquireQueued 方法

B节点

首先B节点在入队之后,执行acquireQueued 方法

    final boolean acquireQueued(final Node node, int arg) {
        // 失败的标志,比如线程B看线程A占用时间太长,不等了,直接走了。
        boolean failed = true;
        try {
            // 阻塞的标志
            boolean interrupted = false;
            // 执行第一次循环:
            for (;;) {
                // 获取节点B的前驱节点- 头结点。predecessor方法看图四
                final Node p = node.predecessor();
                // p 就是 头结点。此时尝试获取对象锁。但是不好意思,线程A还在占用这,因此返回false, if 条件不成立
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return interrupted;
                }
                // 返回 false ,if 条件不成立。请看图五。紧接着第二次循环....
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                // 取消当前正在尝试的节点
                cancelAcquire(node);
        }
    }

predecessor 方法就是获取当前节点的前驱节点

image-20221205194507969

shouldParkAfterFailedAcquire方法:检查当前线程是否应该中断。返回

image-20221205201113173

waitStatus 表示当前节点在队列中的状态,在 AQS内部类Node 中讲过。

image-20221205195334220

经过第一次循环,头节点中的 waitStatus 被修改为 -1 :

image-20221205201227927

开始第二次循环

    final boolean acquireQueued(final Node node, int arg) {
        // 失败的标志,比如线程B看线程A占用时间太长,不等了,直接走了。
        boolean failed = true;
        try {
            // 阻塞的标志
            boolean interrupted = false;
            // 死循环,执行第二次循环:
            for (;;) {
                // 获取节点B的前驱节点- 头结点。predecessor方法看图四
                final Node p = node.predecessor();
                // p 就是 头结点。此时线程B尝试获取对象锁。但是不好意思,线程A还在占用这,因此返回false, if 条件不成立
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return interrupted;
                }
                // 当执行第二次循环时,shouldParkAfterFailedAcquire返回true,请看图六
                if (shouldParkAfterFailedAcquire(p, node) &&
                    // parkAndCheckInterrupt会阻塞线程B,请看图七
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

shouldParkAfterFailedAcquire 方法:

image-20221205195800403

parkAndCheckInterrupt 方法: 此时B线程被阻塞在这个方法中

park/unpark 方法讲解在第五章

image-20221205203800887

C节点

此时B节点阻塞在队列中,当线程C执行入完队列,执行 acquireQueued

    final boolean acquireQueued(final Node node, int arg) {
        boolean failed = true;
        try {
            boolean interrupted = false;
            // 第一次循环
            for (;;) {
                // 获取 C 节点的前驱节点 - B节点,p= NodeB
                final Node p = node.predecessor();
                // B 节点不是头结点,因此直接为false
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return interrupted;
                }
                // 返回 false,请看图八,仅接着执行第二次循环
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

image-20221205202306652

第二次循环:

    final boolean acquireQueued(final Node node, int arg) {
        boolean failed = true;
        try {
            boolean interrupted = false;
            // 第一次循环
            for (;;) {
                // 获取 C 节点的前驱节点 - B节点,p= NodeB
                final Node p = node.predecessor();
                // B 节点不是头结点,因此直接为false
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return interrupted;
                }
                // shouldParkAfterFailedAcquire 返回 true,请看图九
                if (shouldParkAfterFailedAcquire(p, node) &&
                    // // 此时线程C又被阻塞在 parkAndCheckInterrupt 方法中
                    parkAndCheckInterrupt())
                    
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

image-20221205202522131

节点B的 waitStatus被修改为 -1 :

image-20221205203223492

image-20221206175047957

总结

AQS 底层阻塞线程使用的是LockSupport的 park 方法。

每次入队列的节点,都会使前一个节点的 waitStatus值设置为 -1.表示此线程准备好,等待释放对象锁。

unlock

当线程A执行完,是如何释放锁,线程B、C 都被阻塞在 parkAndCheckInterrupt 方法中,是如何被唤醒并且抢到锁的呢? 一步步看

线程A调用 unlock 方法,仍然是 Lock 接口中的方法,在 ReentrantLock 中实现。

image-20221206172033790

unlock 中调用了 AQS 中的release 方法。

image-20221206172203109

release 方法:

    public final boolean release(int arg) {
        // 尝试释放正在占用的对象锁,并返回true。进入 if 语句请看图十
        if (tryRelease(arg)) {
            Node h = head;
            // 头结点不为空,此时正在指向虚拟节点。并且头结点的waitStatus=-1
            // 进入 if 语句,执行 unparkSuccessor。请看图 11 
            if (h != null && h.waitStatus != 0)
                // 
                unparkSuccessor(h);
            return true;
        }
        return false;
    }

这里和上面的 tryAcquire 一样,AQS 中的 tryRelease 作为钩子方法,在 ReentrantLock 重写

image-20221206172757673

tryRelease 方法: 释放锁

image-20221206173243979

此时修改完同步状态:

image-20221206173630336


在执行 unpark方法之前,线程B 被阻塞在 parkAndCheckInterrupt 方法中,请看 图12 。

image-20221206174355137

此时在 unparkSuccessor 方法中唤醒线程B,线程会顺着在被阻塞的地方接着执行,也就是在哪跌倒在哪爬起来。执行 parkAndCheckInterrupt 方法中的 return 返回 fasle。

image-20221206174834886

线程 B 在被唤醒后,接着执行第三次循环,抢占对象锁。

image-20221206180312598

第三次循环:

    final boolean acquireQueued(final Node node, int arg) {
        boolean failed = true;
        try {
            boolean interrupted = false;
            for (;;) {
                // 获取 B节点的前驱节点,也就是头结点
                final Node p = node.predecessor();
                // p==head 条件成立
                // 此时会再次执行 tryAcquire 抢占对象锁。具体抢占过程请看图13
                // 为什么会再次抢占锁呢?不是已经该线程B执行了吗?
                // 队列中的线程会按顺序抢占对象锁没有错,但是我们使用的是非公平锁,非公平锁在进入队列之前就会尝试获取对象锁
                if (p == head && tryAcquire(arg)) {
                    // 线程B抢占成功后,设置头结点为 B节点,具体的设置过程请看图 14
                    setHead(node);
                    // 将头结点的后驱节点设置为null
                    p.next = null; // help GC
                    // 设置失败标志位 false
                    failed = false;
                    // 此时循环结束,线程B成功上位,抢到锁。
                    return interrupted;
                }
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

经过多次调用 ,具体在 nonfairTryAcquire 方法中尝试抢占。

image-20221206181213164

image-20221206181227687

线程B抢占对象锁之后,返回 true

image-20221206181543848

设置头结点的过程: 将前驱节点、线程 设置为null。

image-20221206182204407

此时等待队列的状态

image-20221206182451393

线程C如何被唤醒、抢占锁和B一样,我就不重复了。


cancelAcquire 方法

到此为止,如何抢占锁、阻塞 以及唤醒线程才算一个完整的流程,但是我们分析的都是在理想的状态下,也就是在抢占锁时没有出现意外情况,但事实真是如此吗?

答案肯定是不,在高并发下,任何情况都有可能发生。因此下面看看在出现异常后,执行取消尝试的流程。

比如:在等待过程中,节点B由于等待时间太长,不想等了,那么节点B就需要取消等待-获取锁的资格,并重写设置pre,next

取消尝试的流程 也就是 cancelAcquire 方法

image-20221206183153317

下面分析不同的情况 cancelAcquire 的执行流程:

队列的初始情况,至于队列如何形成的,就不在演示了,和上面步骤一样。

image-20221206184839542第一种情况: 节点5 不想等了,那么它就不会执行 for 循环,而是执行 cancelAcquire

image-20221206211508446

  private void cancelAcquire(Node node) {
       // 节点5 不为 null,条件失败
        if (node == null)
            return;

      // 将节点5的线程置null
        node.thread = null;

        // 节点5 的前驱节点为节点4
        Node pred = node.prev;
      // 节点4 的 waitStatus = -1 ,条件不成立
        while (pred.waitStatus > 0)
            node.prev = pred = pred.prev;

      // 节点4 的后驱节点为节点5
        Node predNext = pred.next;

		// 通过 waitStatus状态图看出 CANCELLED 的值为 1,表示取消的节点
      	// 此时节点5的 waitStatus = 1
        node.waitStatus = Node.CANCELLED;

        // 节点5是尾结点,因此 node == tail 条件成立
       // CAS 操作,将尾结点指向节点4
        if (node == tail && compareAndSetTail(node, pred)) {
            // 将节点4的后驱节点置为 null
            compareAndSetNext(pred, predNext, null);
        } else {
            int ws;
            if (pred != head &&
                ((ws = pred.waitStatus) == Node.SIGNAL ||
                 (ws <= 0 && compareAndSetWaitStatus(pred, ws, Node.SIGNAL))) &&
                pred.thread != null) {
                Node next = node.next;
                if (next != null && next.waitStatus <= 0)
                    compareAndSetNext(pred, predNext, next);
            } else {
                unparkSuccessor(node);
            }
            node.next = node; // help GC
        }
    }

执行完cancelAcquire ,此时队列的情况:
image-20221206212307804

第二种情况:如果中间节点不想等了【以节点4为例】

 private void cancelAcquire(Node node) {
        // 节点4不为null,条件不成立
        if (node == null)
            return;
     // 将节点4 的线程置位null
        node.thread = null;

        // pred = 节点3
        Node pred = node.prev;
     	// 节点3的waitStatus =-1,条件不成立
        while (pred.waitStatus > 0)
            node.prev = pred = pred.prev;

        // predNext = 节点4
        Node predNext = pred.next;

        // 将节点4的waitStatus设置为 1
        node.waitStatus = Node.CANCELLED;

        // 节点4不是尾结点,条件不成立,执行 else 里面的语句
        if (node == tail && compareAndSetTail(node, pred)) {
            compareAndSetNext(pred, predNext, null);
        } else {
            int ws;
            // 节点3不是头结点,条件成立
            if (pred != head &&
                // 节点3的 waitStatus = -1 条件成立,ws = -1,因此也就不会执行 || 后面的
                ((ws = pred.waitStatus) == Node.SIGNAL ||
                 (ws <= 0 && compareAndSetWaitStatus(pred, ws, Node.SIGNAL))) &&
                // 节点3的线程不为null,条件成立
                pred.thread != null) {
                // next = 节点5
                Node next = node.next;
                // 节点5不为null,条件成立
                // 节点5的waitStatus = 0,条件成立
                if (next != null && next.waitStatus <= 0)
                    // 将节点3的后驱节点指向节点5
                    compareAndSetNext(pred, predNext, next);
            } else {
                unparkSuccessor(node);
            }
            // 节点4 的后驱节点指向自己
            node.next = node; // help GC
        }
    }

对应的队列图
image-20221206221826469

第三种情况:如果节点3、节点4,都不想等了

 private void cancelAcquire(Node node) {
        // 节点4不为null,条件不成立
        if (node == null)
            return;
     // 将节点4 的线程置位null
        node.thread = null;

        // pred = 节点3
        Node pred = node.prev;
     	// 此时节点3也不想等了,它的waitStatus的值=1 ,条件成立
        while (pred.waitStatus > 0)
            // 节点3 的前驱节点节点2 赋给 节点4的 前驱节点
            // 也就是说,节点4的prev跳过节点3直接指向节点2
            // 此时 pred = 节点2
            node.prev = pred = pred.prev;

        // predNext = 节点3
        Node predNext = pred.next;

        // 将节点4的waitStatus设置为 1
        node.waitStatus = Node.CANCELLED;

        // 节点4不是尾结点,条件不成立,执行 else 里面的语句
        if (node == tail && compareAndSetTail(node, pred)) {
            compareAndSetNext(pred, predNext, null);
        } else {
            int ws;
            // 节点2不是头结点,条件成立
            if (pred != head &&
                // 节点2的 waitStatus = -1 条件成立,不会执行 || 后边的判断
                ((ws = pred.waitStatus) == Node.SIGNAL ||
                 (ws <= 0 && compareAndSetWaitStatus(pred, ws, Node.SIGNAL))) &&
                // 节点3的线程不为null,条件成立
                pred.thread != null) {
                // next = 节点5
                Node next = node.next;
                // 节点5不为null,条件成立
                // 节点5的waitStatus = 0,条件成立
                if (next != null && next.waitStatus <= 0)
                    // 将节点2的后驱节点指向节点5
                    compareAndSetNext(pred, predNext, next);
            } else {
                unparkSuccessor(node);
            }
            // 节点4 的后驱节点指向自己
            node.next = node; // help GC
        }
    }

对应的队列图
image-20221206221932027

总结

此时AQS核心源码已经完毕,虽然一步一步都整理了下来,但是还是有点懵的,如果有错误的地方还请各位靓仔指出,共同进步…感谢各位观看…

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

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

相关文章

【树莓派】了解wiringPi库、控制继电器

目录一、wiringPi库二、继电器1、继电器介绍及接线说明2、树莓派控制继电器一、wiringPi库 wiringPi是一个很棒的树莓派IO控制库&#xff0c;使用C语言开发&#xff0c;提供了丰富的接口&#xff1a;GPIO控制&#xff0c;中断&#xff0c;多线程等。 在树莓派命令行输入gpio -…

供应商管理软件有哪些特点和优势?

在这个快节奏的商业环境中&#xff0c;企业常常需要同时处理多个供应商。手动处理所有这些流程会有不少困难&#xff0c;为了克服这个问题&#xff0c;供应商管理软件是市场上可用的最佳解决方案。 好用的供应商管理软件&#xff0c;比如广受客户好评的8Manage SRM&#xff0c…

Spring 长事务导致connection closed,又熬了一个大夜!

大家好&#xff0c;我是不才陈某~ 是的&#xff0c;今早一到公司就收到了机器人的告警&#xff0c;从异常日志来看是数据库连接已关闭&#xff0c;然后我在解决这个问题的过程中发现了几个问题&#xff0c;不急&#xff0c;听我一一道来 异常被try后没有继续抛出&#xff0c;导…

CN_广域网WAN@PPP协议

文章目录WAN和LANPPP协议PPP协议有三个组成部分&#xff1a;LCPNCP成帧方法PPP帧的格式信息部分范围工作过程PPP协议特点透明传输WAN&InternetWAN和LAN WAN:广域网&#xff08;全写为 wide area network) 广 域 网局 域 网覆盖范围很广,通常跨区域较小,通常在一个区域内连…

Ubuntu内核OverlayFS权限逃逸漏洞(CVE-2021-3493)

文章目录前言关于linux kernel一、漏洞介绍二、漏洞原理三、漏洞影响版本四、漏洞复现五、修复方法前言 关于linux kernel Linux Kernel 一般指Linux内核。Linux是一种开源电脑操作系统内核。它是一个用C语言写成&#xff0c;符合POSIX标准的类Unix操作系统。 一、漏洞介绍 …

如何掌握HEC-RAS建模方法与涉河建设项目防洪评价报告编制

随着社会经济的快速发展&#xff0c;我国河道周边土地开发利用率不断增大&#xff0c;临河建筑物与日俱增&#xff0c;部分河道侵占严重&#xff0c;导致防洪压力增大。迫切需要对全国从事防洪评价咨询类的技术人员开展防洪评价技术方面的学习&#xff0c;为了让相关工程技术人…

深度学习-支持向量机(SVM)

1. 简介 在机器学习领域&#xff0c;支持向量机SVM(Support Vector Machine)是一个有监督的学习模型&#xff0c;通常用来进行模式识别、分类(异常值检测)以及回归分析。SVM算法中&#xff0c;我们将数据绘制在n维空间中&#xff08;n代表数据的特征数&#xff09;&#xff0c;…

C++ 函数指针探幽

首先看下面两个声明代表什么意思&#xff1f; double* (*(*pf)[2])(double*,int); double* (*pa[2])(double*,int);要搞清楚这两个式子&#xff0c;则先要清楚 指向指针的指针指针数组与指向数组的指针函数指针 指向指针的指针 指针的指针特殊点在于指向的是一个指针而已&am…

栈与队列2:用队列实现栈

主要是我自己刷题的一些记录过程。如果有错可以指出哦&#xff0c;大家一起进步。 转载代码随想录 原文链接&#xff1a; 代码随想录 leetcode链接&#xff1a;344. 反转字符串 题目&#xff1a; 请你仅使用两个队列实现一个后入先出&#xff08;LIFO&#xff09;的栈&#x…

计量经济学复习

计量经济学 习题&#xff08;史浩江版&#xff09; 习题一 一. 单项选择题 1、横截面数据是指&#xff08;A&#xff09;。 A 同一时点上不同统计单位相同统计指标组成的数据 B 同一时点上相同统计单位相同统计指标组成的数据 C 同一时点上相同统计单位不同统计指标组成的…

GPT-Chinese 复现

github 环境准备 conda -create gpt_cn python3.7 conda activate gpt_cnconda install pytorch1.10.0 torchvision0.11.0 torchaudio0.10.0 -c pytorch pip install -r requirements.txt错误 module distutils has no attribute version解决方案&#xff1a; pip uninstal…

[附源码]计算机毕业设计基于Springboot游戏交易平台

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

MinIO实战

1.简介 MinIO 是一款基于Go语言发开的高性能、分布式的对象存储系统。客户端支持Java,Net,Python,Javacript, Golang语言。 2.部署 2.1单机器单节点&#xff08;docker&#xff09; 官网教程&#xff1a;https://min.io/docs/minio/container/index.html mkdir -p ~/minio/dat…

Node.js编程

Node.js编程 一、实验目的与要求 实验任务 用户信息增删改查 掌握数据库软件的安装了解集合、文档的概念掌握使用mongoose创建集合的方法创建集合掌握对数据库中的数据进行增删改查操作 二、实验任务和步骤 实验1. 用户信息增删改查 需求说明 (1)搭建网站服务器&#xf…

第十章 降维与度量学习

10.1 k近邻学习 k近邻学习&#xff08;kNN&#xff09;是一种常用的监督学习方法&#xff0c;其工作机制非常简单&#xff1a;给定测试样本&#xff0c;基于某种距离度量找出训练集中与其最靠近的k个训练样本&#xff0c;然后基于这k个邻居的信息来进行预测。 k近邻学习似乎与…

2022 计网复习计算题【太原理工大学】

期末复习汇总&#xff0c;点这里&#xff01;https://blog.csdn.net/m0_52861684/category_12095266.html?spm1001.2014.3001.5482 三、计算题 1. 假定 1km 长的 CSMA/CD 网络的数据率为 1Gb/s&#xff0c;设信号在网络上的传播速率为 200000km/s。求能够使用此协议的最短帧长…

java swing(GUI) MySQL实现的学生选课成绩管理系统源码+运行教程

今天给大家演示一下由Java swing mysql实现的一款学生选课成绩信息管理系统&#xff0c;主要实现的功能有&#xff1a;学生教师信息管理、年级班级信息管理、课程信息管理、选课、成绩录入功能、成绩统计功能&#xff0c;实现学生、教师、管理员三个角色的登录&#xff0c;三个…

【计算机图形学入门】笔记1:图形学概述

前言&#xff1a;今天开始开启一个新篇章的学习&#xff0c;那就是games101闫令琪老师讲的《现代计算机图形学入门》课程&#xff0c;我会根据闫老师每节课讲的内容记录重点笔记&#xff0c;每节课都会整理一篇发布出来&#xff0c;希望自己可以坚持下去&#xff0c;从图形学小…

可视化音视频分析工具:好用工具大集锦,快转发给你兄弟看看丨音视频工具

&#xff08;本文基本逻辑&#xff1a;音画原始数据分析工具介绍 → 编码数据分析工具介绍 → 封装格式分析工具介绍&#xff09; 工欲善其事&#xff0c;必先利其器。在音视频开发中&#xff0c;为了方便、快捷、直观的分析音视频数据&#xff0c;最好能有一些可视化的分析工…

《爱在 ZStack Cube 超融合》三部曲

一、始于初识&#xff1a;很高兴见到你 这一天东川路最靓的仔打开了 ZStack Cube 宝盒 &#xff0c;这可能是我们的第一次相遇&#xff0c;我们相谈甚欢&#xff0c;相遇恨晚。 我的名字是 ZStack Cube&#xff0c;一个基于超融合架构的云平台。我拥有3300、5300、7300、7300…