【Java 并发编程】深入理解 AQS - AbstractQueuedSynchronizer

深入理解 AQS - AbstractQueuedSynchronizer

  • 1. AQS
    • 1.1 什么是 AQS
    • 1.2 AQS 具备的特性
  • 2. AQS 原理解析
    • 2.1 AQS 原理概述
      • 2.1.1 什么是 CLH 锁
      • 2.1.2 AQS 中的队列
    • 2.2 AQS 共享资源的方式:独占式和共享式
      • 2.2.1 Exclusive(独占式)
      • 2.2.2 Share(共享式)
    • 2.3 AQS 底层使用了模板方法模式
    • 2.4 AQS 定义了两种队列
      • 2.4.1 AQS 定义了5个队列中节点状态
      • 2.4.2 同步等待队列
      • 2.4.3 条件等待队列
        • Condition 接口
  • 3. AQS 源码分析
    • 3.1 ReentrantLock 概述
    • 3.2 创建重入锁
    • 3.3 公平锁/⾮公平锁的 lock()
      • 3.3.1 公平锁 NonfairSync.lock() 方法
      • 3.3.2 非公平锁 NonfairSync.lock() 方法
      • 3.3.3 acquire()
      • 3.3.4 FairSync.tryAcquire(arg)
      • 3.3.5 NonFairSync.tryAcquire(arg)
      • 3.3.6 addWaiter(Node.EXCLUSIVE)
      • 3.3.7 acquireQueued(...)
    • 3.4 解锁
    • 3.4.1 tryRelease()
    • 3.4.2 unparkSuccessor

1. AQS

在介绍 AQS 之前,先一起回顾一下 CAS(Compare And Swap)。

1.1 什么是 AQS

AQS,全名 AbstractQueuedSynchronizer,是一个抽象同步队列,它的内部通过维护一个状态 volatile int state(共享资源的状态),一个 FIFO 线程等待队列来实现同步功能。

Java 并发包很多工具类底层都是基于 AQS 来实现的,比如:Lock 工具 ReentrantLock、栅栏 CountDownLatch、信号量 Semaphore 等。

1.2 AQS 具备的特性

  • 阻塞等待队列
  • 共享/独占
  • 公平/非公平
  • 可重入
  • 允许中断

2. AQS 原理解析

2.1 AQS 原理概述

AQS 内部维护 state 属性,state 用关键字 volatile 修饰,代表着该共享资源的状态一更改就能被所有线程可见,而 AQS 的加锁方式本质上就是多个线程在竞争 state,当 state 为 0 时代表线程可以竞争锁,不为 0 时代表当前对象锁已经被占有,其他线程来加锁时则会失败,加锁失败的线程会被放入一个 FIFO 的等待队列中,这些线程会被 UNSAFE.park() 操作挂起,等待其他获取锁的线程释放锁才能够被唤醒。

2.1.1 什么是 CLH 锁

  • 由 Craig、Landin 和 Hagersten 三位大佬发明,因此命名为 CLH 锁;

  • 单向链表实现的队列;

  • 是一个自旋公平(FIFO)锁,能确保无饥饿性;

  • 申请线程只在本地变量上自旋,它不断轮询前驱的状态,如果发现前驱节点释放了锁就结束自旋

一起看一下线程加锁的过程:

  1. 首先获得当前线程的当前节点 curNode,这里每次获取的 CLHNode 节点的 locked 状态都为false;

  2. 然后将当前 CLHNode 节点的 locked 状态赋值为 true,表示当前线程的一种有效状态,即获取到了锁或正在等待锁的状态;

注意,为了保证 locked 属性线程间可见,该属性被 volatile 修饰。

  1. 线程对 tail 域调用 getAndSet 方法,使自己成为队列的尾部,同时获取一个指向其前趋结点的引用 PreNode;

  2. 因为尾指针 tailNode 的总是指向了前一个线程的 CLHNode 节点,因此这里利用尾指针 tailNode 取出前一个线程的 CLHNode 节点,然后赋值给当前线程的前继节点 preNode,并且将尾指针重新指向最后一个节点即当前线程的当前 CLHNode 节点,以便下一个线程到来时使用;

  3. 根据前继节点(前一个线程)的 locked 状态判断,若 locked 为 false,则说明前一个线程释放了锁,当前线程即可获得锁,不用自旋等待;若前继节点的 locked 状态为 true,则表示前一线程获取到了锁或者正在等待,自旋等待。

在这里插入图片描述

如上图:有3个并发线程同时启动执行 lock 操作,假如3个线程的实际执行顺序为:T1、T2、T3

  1. 线程1 过来,执行了 lock 操作,获得了锁,此时 locked 状态为 true

  2. 线程2 过来,执行了 lock 操作,由于线程1 还未释放锁,此时自旋等待,locked 状态也为 true

  3. 线程3 过来,执行了 lock 操作,由于线程2 处于自旋等待,此时线程3 也自旋等待(因此 CLH 锁是公平锁),locked 状态也为 true

以下为 CLH 锁的释放锁过程:

  1. 首先从当前线程的线程本地变量中获取出当前 CLHNode 节点,同时这个 CLHNode 节点被后面一个线程的 preNode 变量指向着;

  2. 然后将 locked 状态置为 false 即释放了锁;

注意:locked 因为被 volitile 关键字修饰,此时后面自旋等待的线程的局部变量 preNode.locked 也为 false,因此后面自旋等待的线程结束 while 循环即结束自旋等待,此时也获取到了锁。这一步骤也在异步进行着。

  1. 然后给当前线程的表示当前节点的线程本地变量重新赋值为一个新的 CLHNode。

2.1.2 AQS 中的队列

AQS 中的队列是 CLH 变体的虚拟双向队列,通过将每条请求共享资源的线程封装成一个节点来实现锁的分配:

在这里插入图片描述
AQS 中的 CLH 变体等待队列拥有以下特性:

  • AQS 中队列是个双向链表,也是 FIFO 先进先出的特性

  • 通过 head、tail 头尾两个节点来组成队列结构,通过 volatile 修饰保证可见性

  • head 指向节点为已获得锁的节点,是一个虚拟节点,节点本身不持有具体线程

  • 获取不到同步状态,会将节点进行自旋获取锁,自旋一定次数失败后会将线程阻塞,相对于 CLH 队列性能较好

AQS 使用一个 int 成员变量来表示同步状态,通过内置的 FIFO 队列来完成获取资源线程的排队工作。AQS 使用 CAS 对该同步状态进行原子操作实现对其值的修改:

// 共享变量,使用 volatile 关键字修饰保证线程可见性
private volatile int state;

资源的可用状态通过 protected 类型的 getState、setState、compareAndSetState 进行操作:

// 返回同步状态的当前值
protected final int getState() {return state;
}// 设置同步状态的值
protected final void setState(int newState) {state = newState;
}// 如果当前状态值等于期望值,则原子地将同步状态设置为给定的更新值。
protected final boolean compareAndSetState(int expect, int update) {return unsafe.compareAndSwapInt(this, stateOffset, expect, update);
}

2.2 AQS 共享资源的方式:独占式和共享式

AQS 定义了两种资源共享方式 :独占式(Exclusive)和共享式(Share)。

2.2.1 Exclusive(独占式)

只有一个线程能执行,如 ReentrantLock。又可分为公平锁和非公平锁,ReentrantLock 同时支持两种锁。

2.2.2 Share(共享式)

多个线程可以同时执行,如 Semaphore/CountDownLatch。

2.3 AQS 底层使用了模板方法模式

同步器的设计是基于模板方法模式的,如果需要自定义同步器一般的方式是这样(模板方法模式很经典的一个应用) :

  1. 使用者继承 AbstractQueuedSynchronizer重写指定的方法(这些重写方法很简单,无非是对于共享资源 state 的获取和释放)。

  2. AQS 组合在自定义同步组件的实现中,并调用其模板方法,而这些模板方法会调用使用者重写的方法。

这和我们以往通过实现接口的方式有很大区别,这是模板方法模式很经典的一个运用,感兴趣的可以参考:https://refactoringguru.cn/design-patterns/template-method

AQS 使用了模板方法模式,自定义同步器时需要重写下面几个 AQS 提供的模板方法:

  • isHeldExclusively():该线程是否正在独占资源。只有用到 condition 才需要去实现它。

  • tryAcquire(int)独占方式。尝试获取资源,成功则返回 true,失败则返回 false。

  • tryRelease(int)独占方式。尝试释放资源,成功则返回 true,失败则返回 false。

  • tryAcquireShared(int)共享方式。尝试获取资源。负数表示失败;0 表示成功,但没有剩余可用资源;正数表示成功,且有剩余资源。

  • tryReleaseShared(int)共享方式。尝试释放资源,如果释放后允许唤醒后续等待结点返回 true,否则返回 false。

2.4 AQS 定义了两种队列

  • 同步等待队列: 主要用于维护获取锁失败时入队的线程。

  • 条件等待队列: 调用 await() 的时候会释放锁,然后线程会加入到条件队列,调用 signal()/signalAll() 唤醒的时候会把条件队列中的线程节点移动到同步队列中,等待再次获得锁。

2.4.1 AQS 定义了5个队列中节点状态

  1. 值为0,初始化状态,表示当前节点在 sync 队列中,等待着获取锁。

  2. CANCELLED,值为 1,表示当前的线程被取消;

  3. SIGNAL,值为 -1,表示当前节点的后继节点包含的线程需要唤醒,也就是 unpark;

  4. CONDITION,值为 -2,表示当前节点在等待 condition,也就是在 condition 队列中;

  5. PROPAGATE,值为 -3,表示当前场景下后续的 acquireShared 能够得以执行;


/** waitStatus value to indicate thread has cancelled */
static final int CANCELLED =  1;
/** waitStatus value to indicate successor's thread needs unparking */
static final int SIGNAL    = -1;
/** waitStatus value to indicate thread is waiting on condition */
static final int CONDITION = -2;
/*** waitStatus value to indicate the next acquireShared should* unconditionally propagate*/
static final int PROPAGATE = -3;

2.4.2 同步等待队列

AQS 当中的同步等待队列是 CLH 的变体,见:2.1.12.1.2

  1. 当前线程如果获取同步状态失败时,AQS 则会将当前线程已经等待状态等信息构造成一个节点(Node)并将其加入到 CLH 同步队列,同时再一次尝试获取锁,如果获取失败则会阻塞当前线程;

  2. 当同步状态释放时,会把首节点唤醒(公平锁),使其再次尝试获取同步状态;

  3. 通过 signal/signalAll 将条件队列中的节点转移到同步队列。(由条件队列转化为同步队列)。

在这里插入图片描述

2.4.3 条件等待队列

AQS 中条件队列是使用单向列表保存的,用 nextWaiter 来连接:

  • 调用 await 方法阻塞线程;

  • 当前线程存在于同步队列的头结点,调用 await 方法进行阻塞(从同步队列转化到条件队列)

Condition 接口

在这里插入图片描述

  1. 调用 Condition#await 方法会释放当前持有的锁,然后阻塞当前线程,同时向 Condition 队列尾部添加一个节点,所以调用 Condition#await 方法的时候必须持有锁。

  2. 调用 Condition#signal 方法会将 Condition 队列的首节点移动到阻塞队列尾部,然后唤醒因调用 Condition#await 方法而阻塞的线程(唤醒之后这个线程就可以去竞争锁了),所以调用 Condition#signal 方法的时候必须持有锁,持有锁的线程唤醒被因调用 Condition#await 方法而阻塞的线程。

3. AQS 源码分析

3.1 ReentrantLock 概述

ReentrantLock 是可重入的独占锁,只能有一个线程可以获取该锁,其它获取该锁的线程会被阻塞而被放入该锁的阻塞队列里面。当我们想要使⽤重⼊锁的时候,使⽤⽅式⼀般是如下3个步骤:

public class ReentrantLockTest {/** 1. 创建重入锁  */ReentrantLock lock = new ReentrantLock();public void doSomething() {// 2. 加锁 block until condition holdslock.lock();try {// do something} catch (Exception e) {// ...} finally {// 3. 解锁lock.unlock();}}
}

后面我们就是针对这3个步骤对其源码进⾏解析。在此之前,先一起看一下 Sync、FairSync、NonfairSync 是在哪⾥被使⽤的。

ReentrantLock 默认采用非公平锁,因为考虑获得更好的性能,通过 boolean 来决定是否用公平锁(传入 true 用公平锁)。

/** Synchronizer providing all implementation mechanics */
private final Sync sync;/*** 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();
}

观察源码可以发现 FairSync、NonfairSync 都是继承 Sync,而 Sync 又继承于 AbstractQueuedSynchronizer。

static final class FairSync extends Sync
static final class NonfairSync extends Sync
abstract static class Sync extends AbstractQueuedSynchronizer

3.2 创建重入锁

ReentrantLock 默认是非公平的:

public ReentrantLock() {sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {sync = fair ? new FairSync() : new NonfairSync();
}

当我们使⽤⽆参构造⽅法去创建重⼊锁的时候,底层使⽤的是⾮公平锁

  • 当⼊参 fair 等于 false 的时候,采⽤的就是⾮公平锁 - NonfairSync

  • 当⼊参 fair 等于 true 的时候,采⽤的就是公平锁 - FairSync

3.3 公平锁/⾮公平锁的 lock()

3.3.1 公平锁 NonfairSync.lock() 方法

公平锁直接就执行 acquire() 排队等待:

/*** 公平锁*/
static final class FairSync extends ReentrantLock.Sync {private static final long serialVersionUID = -3000897897090466540L;final void lock() {// 执行等待acquire(1);}protected final boolean tryAcquire(int acquires) {final Thread current = Thread.currentThread();int c = getState();if (c == 0) {// 和非公平锁相比,这里多了一个是否有线程在等待的判断if (!hasQueuedPredecessors() &&compareAndSetState(0, acquires)) {setExclusiveOwnerThread(current);return true;}}else if (current == getExclusiveOwnerThread()) {int nextc = c + acquires;if (nextc < 0)throw new Error("Maximum lock count exceeded");setState(nextc);return true;}return false;}
}

3.3.2 非公平锁 NonfairSync.lock() 方法

/*** 非公平锁*/
static final class NonfairSync extends ReentrantLock.Sync {private static final long serialVersionUID = 7316153563782823691L;/*** 立即执行抢占锁操作。如果抢占成功,则加锁;如果抢占失败,则等待*/final void lock() {// AQS的state如果成功被设置为1,则表示本线程已抢占锁成功;否则,抢占锁失败if (compareAndSetState(0, 1))// 将 AOS.excLusive0wnerThread 设置为当前线程setExclusiveOwnerThread(Thread.currentThread());else// 执行等待acquire(1);}// 是否获得非公平锁protected final boolean tryAcquire(int acquires) {return nonfairTryAcquire(acquires);}
}

先判断是否有锁,0 是无锁、1 是有锁,等于 0 的话则修改成 1 并且获得锁:

// 如果当前状态值等于期望值 0,则自动将同步状态设置为给定的更新值 1
protected final boolean compareAndSetState(int expect, int update) {// See below for intrinsics setup to support thisreturn unsafe.compareAndSwapInt(this, stateOffset, expect, update);
}

设置当前独占锁的拥有者线程:

protected final void setExclusiveOwnerThread(Thread thread) {// 将当前线程设置到这把独占锁上exclusiveOwnerThread = thread;
}

抢锁失败就进行则等待:

public final void acquire(int arg) {if (!tryAcquire(arg) &&acquireQueued(addWaiter(Node.EXCLUSIVE), arg))selfInterrupt();
}

公平锁和非公平锁只有两处不同:

  1. 非公平锁在调用 lock 后,首先就会调用 CAS 进行一次抢锁,如果这个时候恰巧锁没有被占用,那么直接就获取到锁返回了。

  2. 非公平锁在 CAS 失败后,和公平锁一样都会进入到 tryAcquire 方法,在 tyAcquire 方法中,如果发现锁这个时候被释放了(state == 0),非公平锁会直接 CAS 抢锁,但是公平锁会判断等待队列是否有线程处于等待状态,如果有则不去抢锁,乖乖排到后面。

非公平锁如果这两次 CAS 都不成功,那么后面和公平锁是一样的, 都要进入到阻塞队列等待唤醒。

相对来说,非公平锁会有更好的性能,因为它的吞吐量比较大。当然,非公平锁让获取锁的时间变得更加不确定,可能会导致在阻塞队列中的线程长期处于饥饿状态

3.3.3 acquire()

⽆论是公平锁还是⾮公平锁,他们都有机会调⽤相同的⽅法,即:acquire(1)

/*** 以独占模式获取,忽略中断 (interrupts)* 通过调用至少一次 tryAcquire(int) 来实现,成功返回。否则线程排队,可能重复阻塞和解除阻塞,调用 tryAcquire(int) 直到成功* 此方法可用于实现方法 Lock.lock()*/
public final void acquire(int arg) {/*** tryAcquire(1): 进行抢锁操作,返回是否抢锁成功* addWaiter(Node.EXCLUSIVE): 构建一个独占式节点 Node,维护好该节点的前后指针 Node* acquireQueued(addWaiter(Node.EXCLUSIVE), 1) 判断获取锁失败的时候,是否应该挂起该线程,如果是,则挂起当前线*/if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg))// 设置当前线程的中断标识selfInterrupt();
}

也就是分为三步 tryAcquire(尝试抢锁)、addWaiter(构建节点加入到队列中)、acquireQueued(获取锁失败的时候将线程挂起)

3.3.4 FairSync.tryAcquire(arg)

/*** 进行抢锁操作,返回是否成功** case1> 没人抢占锁(state==0),线程A尝试执行抢占锁操作,如果抢占成功,则返回true; 如果抢占失败,则返回false。* case2> 有人已经抢占了这个锁(state != 0),但是抢占这个锁的线程就是自己,那么对自己执行重入加锁操作,返回true;如果不是自己抢占的锁,那么返回false。*/
protected final boolean tryAcquire(int acquires) {final Thread current = Thread.currentThread();int c = getState();/** case1: 如果c等于0,说明可以抢占锁 */if (c == 0) {// 如果线程不需要排队 && 抢占锁成功(即:如果state=0,则将该值修改为1,CAS操作成功)if (!hasQueuedPredecessors() && compareAndSetState(0, acquires)) {setExclusiveOwnerThread(current);return true;}}/** case2: 如果c不等于0,判断是否是重入操作 (即: 锁本来就是被自己抢占的,支持多次抢占) */else if (current == getExclusiveOwnerThread()) {int nextc = c + acquires;if (nextc < 0)throw new Error("Maximum lock count exceeded");setState(nextc);return true;}return false;
}

通过获取 state 来判断锁是否被获取,等于 0 的话不要排队直接获取锁,可以看到比 NonFairSync 多了 hasQueuedPredecessors 是否需要进入队列排队。

如果不等于 0,判断是否是重入;否则返回 false。

/*** 主要是用来判断线程需不需要排队。true:线程需要排队。false:线程不需要排队。* 因为队列是 FIF0 的,所以需要判断队列中有没有相关线程的节点已经在排队了。有则返回true表示线程需要排队;没有则返回 false 表示线程无需排队;*/
public final boolean hasQueuedPredecessors() {// 读取头节点Node t = tail;// 读取尾节点Node h = head;// s是首节点h的后继节点Node s;/*** h != t -> 队列中有 >= 2个节点* (s = h.next) == null -> 头节点没有后继节点,即:只有自己一个node或者创建了h的后置节点,但是还没有执行h.next=node* s.thread != Thread.currentThread() -> 第二个节点承载的线程不是当前线程*/return h != t && ((s = h.next) == null || s.thread != Thread.currentThread());
}

3.3.5 NonFairSync.tryAcquire(arg)

/*** 是否获得非公平锁*/
protected final boolean tryAcquire(int acquires) {return nonfairTryAcquire(acquires);
}/*** 进行抢锁操作,是否抢到非公平锁** 处理内容:* 1>如果抢到锁,返回true*   1.1>如果当前线程第一次抢到锁:*        AQS.state由0变为1*        AQS.exclusiveOwnerThread=Thread.currentThread()*        返回true*   1.2>如果当前线程再次抢到锁(重入加锁):*        AQS.status++*        返回true* 2>如果没抢到锁,返回false*/
final boolean nonfairTryAcquire(int acquires) {final Thread current = Thread.currentThread();int c = getState();if (c == 0) {if (compareAndSetState(0, acquires)) {setExclusiveOwnerThread(current);return true;}}else if (current == getExclusiveOwnerThread()) {/*** 获得当前独享线程,如果就是当前线程,那么执行重入操作* 执行tryLock()时:*      如果第二次进入,则 nextc = 0 + 1 = 1*      如果第三次进入,则 nextc = 1 + 1 = 2*      如果第四次进入,则 nextc = 2 + 1 = 3*/int nextc = c + acquires;if (nextc < 0) // overflowthrow new Error("Maximum lock count exceeded");setState(nextc);return true;}return false;
}

3.3.6 addWaiter(Node.EXCLUSIVE)

private Node addWaiter(Node mode) {Node node = new Node(Thread.currentThread(), mode);// Try the fast path of enq; backup to full enq on failureNode pred = tail;if (pred != null) {node.prev = pred;if (compareAndSetTail(pred, node)) {pred.next = node;return node;}}enq(node);return node;
}
private Node enq(final Node node) {for (;;) {Node t = tail;if (t == null) { // Must initializeif (compareAndSetHead(new Node()))tail = head;} else {node.prev = t;if (compareAndSetTail(t, node)) {t.next = node;return t;}}}
}

3.3.7 acquireQueued(…)

final boolean acquireQueued(final Node node, int arg) {boolean failed = true;try {boolean interrupted = false;for (;;) {final Node p = node.predecessor();if (p == head && tryAcquire(arg)) {setHead(node);p.next = null; // help GCfailed = false;return interrupted;}if (shouldParkAfterFailedAcquire(p, node) &&parkAndCheckInterrupt())interrupted = true;}} finally {if (failed)cancelAcquire(node);}
}

3.4 解锁

public void unlock() {sync.release(1);
}public final boolean release(int arg) {if (tryRelease(arg)) {Node h = head;if (h != null && h.waitStatus != 0)unparkSuccessor(h);return true;}return false;
}

3.4.1 tryRelease()

首先执行tryRelease,再执行 unpark 操作

protected final boolean tryRelease(int releases) {int c = getState() - releases;if (Thread.currentThread() != getExclusiveOwnerThread())throw new IllegalMonitorStateException();boolean free = false;if (c == 0) {free = true;setExclusiveOwnerThread(null);}setState(c);return free;
}

3.4.2 unparkSuccessor

private void unparkSuccessor(Node node) {/** If status is negative (i.e., possibly needing signal) try* to clear in anticipation of signalling.  It is OK if this* fails or if status is changed by waiting thread.*/int ws = node.waitStatus;if (ws < 0)compareAndSetWaitStatus(node, ws, 0);/** Thread to unpark is held in successor, which is normally* just the next node.  But if cancelled or apparently null,* traverse backwards from tail to find the actual* non-cancelled successor.*/Node s = node.next;if (s == null || s.waitStatus > 0) {s = null;for (Node t = tail; t != null && t != node; t = t.prev)if (t.waitStatus <= 0)s = t;}if (s != null)LockSupport.unpark(s.thread);
}

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

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

相关文章

JVM学习笔记(中)

1、垃圾回收算法 标记清除法 特点&#xff1a; 速度较快会产生内存碎片 注意&#xff1a;这里的清除并不是真正意义上的清除&#xff0c;即每个字节都清0&#xff0c;而是记录一下被清除的对象的起始和结束的地址&#xff0c;当下一次分配给一个新对象时&#xff0c;新对象…

《Java并发编程实战》课程笔记(四)

互斥锁 原子性问题到底该如何解决呢&#xff1f; “同一时刻只有一个线程执行”这个条件非常重要&#xff0c;我们称之为互斥。如果我们能够保证对共享变量的修改是互斥的&#xff0c;那么&#xff0c;无论是单核 CPU 还是多核 CPU&#xff0c;就都能保证原子性了。 锁模型 …

RK3588平台开发系列讲解(驱动基础篇)设备树常用 of 函数

平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、查找节点的 of 函数二、获取属性值的 of 函数三、实验示例3.1、查找的节点代码3.2、获取属性内容代码沉淀、分享、成长,让自己和他人都能有所收获!😄 📢 设备树描述了设备的详细信息,这些信息包括数字类型的…

chatgpt赋能python:Python中-1的用法介绍

Python中-1的用法介绍 什么是-1&#xff1f; 在Python中&#xff0c;-1是一个特殊的索引值&#xff0c;它表示从序列的末尾开始向前数1个元素。这在对于列表、字符串、元组等序列类型进行操作时非常有用。 如何使用-1&#xff1f; 假设我们有一个列表&#xff1a; l [1, …

SpringBoot框架理解

1 SpringBoot入门 1.2 什么是SpringBoot 1 官网的解释 ​ Spring在官方首页是这么说的&#xff1a;说使用SpringBoot可以构造任何东西&#xff0c;SpringBoot是构造所有基于Spring的应用程序的起点,SpringBoot在于通过最少的配置为你启动程序。 2 我的理解 SpringBoot是Sp…

Flask-RESTful的使用

Flask-RESTful的使用 Flask-RESTful基本使用安装定义资源Resources创建API实例添加资源到API运行Flask应用 请求处理请求解析参数校验 响应处理数据序列化定制返回格式 其他功能蓝图装饰器集合路由命名规范路由名称 Flask-RESTful Flask-RESTful是一个用于构建RESTful API的扩展…

计算机对社会的应用是什么,电子计算对人类社会有什么贡献?应用的领域又有哪些?...

在人类历史上&#xff0c;蒸汽机的发明和电力的使用&#xff0c;曾经在生产技术上引起过划时代的工业革命&#xff0c;然而&#xff0c;这种革命&#xff0c;从本质上来讲&#xff0c;它仅仅是涉及到代替人的体力劳动&#xff0c;但电子计算机的发明&#xff0c;已经涉及到代替…

一位美女博士的人脸识别历程

2019-01-28 16:44:37 1月21日&#xff0c;科技评论期刊《麻省理工科技评论》发布了2018年“35岁以下科技创新35人”&#xff08;35 Innovators Under 35&#xff09;中国榜单。商汤科技研究总监、年仅29岁的石建萍博士荣登此榜&#xff0c;凭借在计算机视觉原创技术的卓越创新…

Android仿微信发图片的样式,做IM的同学的病有救了

一&#xff1a;前言 最近在搞IM&#xff0c;真的特别痛苦。脑袋大&#xff0c;对于我这种菜鸟来说太难了&#xff0c;比现在社会娶个媳妇还难&#xff0c;硬着头皮搞&#xff0c;终于文字&#xff0c;语音&#xff0c;表情搞完了&#xff0c;开始搞图片&#xff0c;看着微信发…

软件压力测试图片60张,看图测压力,你抗压么?

你压力大么&#xff1f;快跟K线君一起来测测&#xff01; 平行线 下图里的横线都是平行的 涉世越深的人&#xff0c;受社会侵蚀越严重 看到的直线越变形 你还是单纯的你吗&#xff1f; 你能看出几条笔直的横线&#xff1f; 我想静静 这是一张静止的图片 你的心理压力越大&#…

网络安全入门学习:社会工程学

在电影《我是谁&#xff1a;没有绝对安全的系统》中&#xff0c;主角本杰明充分利用自己高超的黑客技术&#xff0c;非法入侵国际安全系统&#xff0c;并在最后逃之夭夭。在电影中&#xff0c;有一句经典的台词&#xff1a; 所有黑客手段中最有效的、最伟大的幻想艺术——社会…

《计算机组成原理》唐朔飞 第5章 输入输出系统 - 学习笔记

写在前面的话&#xff1a;此系列文章为笔者学习计算机组成原理时的个人笔记&#xff0c;分享出来与大家学习交流。使用教材为唐朔飞第3版&#xff0c;笔记目录大体与教材相同。 网课 计算机组成原理&#xff08;哈工大刘宏伟&#xff09;135讲&#xff08;全&#xff09;高清_…

easyui 表单提交与图片上传,图片添加、删除

提交表单和图片是web中经常要用到的。我这里用easyui做了一个表单&#xff0c;里面可以上传多张图片&#xff0c;并且可以进行新增和删除。 前端代码如下&#xff1a; <!DOCTYPE html> <html> <head> <meta charset"UTF-8"> …

社会化媒体营销方案简介

最近公司经营走到了死胡同里面&#xff0c;就开始研究运营的营销方案&#xff0c;发现有很多的IT公司都在走社会化营销&#xff0c;感觉这会是以后每个公司都会用到的一种营销策略&#xff01;社会化媒体营销-亦称社会化营销&#xff0c;是利用社会化网络&#xff0c;在线社区&…

神经网络提取图片特征,神经网络算法识别图像

如何用Python和深度神经网络寻找相似图像 代码首先&#xff0c;读入TuriCreate软件包import turicreate as tc我们指定图像所在的文件夹image&#xff0c;让TuriCreate读取所有的图像文件&#xff0c;并且存储到data数据框data tc.image_analysis.load_images(./image/)我们来…

发一张你认为很漂亮的美女照片?

顾锦盒 &#xff0c;INFP, 4w5 565 人赞同 谢谢大家的赞。更新几张。 春风再美也比不过你的笑。 编辑于 2016-04-11 75 条评论 感谢 分享 收藏 • 没有帮助 • 举报 • 作者保留权利 201赞同 反对&#xff0c;不会显示你的姓名 吴名士 &#xff0c;公众号&#xf…

机器学习在社会科学中的应用

本文把目前机器学习技术在社会科学研究中的应用分成三类&#xff1a;第一&#xff0c;数据生成&#xff08;Data Generating Process&#xff09;&#xff1a;机器学习可以帮助学者获得以前很难或无法获得的数据&#xff1b;第二&#xff0c;预测&#xff08;Prediction&#x…

[RUST/腐蚀]Windows-开服服务端下载以及配置

一、前置要求 1.SteamCMD&#xff1a;SteamCMD - Valve Developer Communityhttps://developer.valvesoftware.com/wiki/SteamCMD 2.通过SteamCMD下载RUST/腐蚀服务端。 二、SteamCMD 注意&#xff1a;所有目录均应避免出现中文。 1.建立SteamCMD文件夹&#xff0c;如 D:\st…

chatgpt赋能python:Python中的_--了解这个神秘的下划线

Python中的_ – 了解这个神秘的下划线 Python是一种流行的编程语言&#xff0c;它具有简单易学的语法和强大的功能。一些Python的特殊语法经常会让初学者感到困惑。其中&#xff0c;一个神秘的下划线符号在Python中出现的频率非常高&#xff0c;而且它的含义和使用也非常多样化…

查看MySQL服务器是否启用了SSL连接,并且查看ssl证书是否存在

文章目录 一、查看MySQL服务器是否启用了SSL连接 1.登录MySQL服务器 2.查看SSL配置 二、查看证书是否存在 前言 查看MySQL服务器是否启用了SSL连接&#xff0c;并且查看ssl证书是否存在 一、查看MySQL服务器是否启用了SSL连接 1.登录MySQL服务器 在Linux终端中&#xf…