javaEE - 2(11000字详解多线程)

一:多线程带来的的风险-线程安全

线程安全的概念:如果多线程环境下代码运行的结果是符合我们预期的,即在单线程环境应该的结果,则说这个程序是线程安全的。

当多个线程同时访问共享资源时,就会产生线程安全的风险,下面通过一段代码演示:

static class Counter {public int count = 0;void increase() {count++;}
}
public static void main(String[] args) throws InterruptedException {final Counter counter = new Counter();Thread t1 = new Thread(() -> {for (int i = 0; i < 50000; i++) {counter.increase();}});Thread t2 = new Thread(() -> {for (int i = 0; i < 50000; i++) {counter.increase();}});t1.start();t2.start();t1.join();//主线程等待t1线程结束t2.join();//主线程等待t2线程结束System.out.println(counter.count);
}

这个代码,是两个线程针对同一个变量各自自增 5w 次.

运行程序,预期结果应该是10w
实际是个随机值一样。每次的结果还不一样。

实际结果和预期结果不相符,就是 bug
由多线程引起的 bug => 线程不安全/线程安全问题

那么为什么会出现这个情况呢?

count++ 操作,本质上是三个cpu 指令构成~~

  1. load,把内存中的数据读取到cpu寄存器中.
  2. add,就是把寄存器中的值,进行+1运算
  3. save,把寄存器中的值写回到内存中.

在这里插入图片描述
由于多线程调度顺序是不确定的,实际执行过程中,这俩线程的++操作实际的指令排列顺序就有很多可能!!!

在这里插入图片描述
在这里插入图片描述在这里插入图片描述
在这里插入图片描述
等等等等许多的组合

1.1 组合1

在这里插入图片描述
对于这种组合来说:

第一步:
在这里插入图片描述
第二步(t1-load):
在这里插入图片描述
第三步(t1-add):
在这里插入图片描述
第四步(t1-save):
在这里插入图片描述
第五步(t2-load):
在这里插入图片描述
第六步(t2-add):
在这里插入图片描述
第七步(t2-save):
在这里插入图片描述
这种组合是没有问题的,能够无误的完成自增。

1.2 组合2

在这里插入图片描述
但是对于这种组合来说就有点问题了

第一步:
在这里插入图片描述
第二步(t1-load):
在这里插入图片描述
第三步(t2-load):
在这里插入图片描述
第四步(t2-add):
在这里插入图片描述
第五步(t1-add):
在这里插入图片描述
第六步(t1-save):
在这里插入图片描述
第七步(t2-save):
在这里插入图片描述
这就和我们的预期结果2不一致了,因为这有类似脏读的数据,一切罪恶的根源就是cpu调度的抢占式执行!这就是多线程带来的风险。

当我们修改共享数据的时候就很容易发生线程不安全的情况:

上面的线程不安全的代码中, 涉及到多个线程针对 counter.count 变量进行修改。此时这个 counter.count 是一个多个线程都能访问到的数据”,counter.count 这个变量就是在堆上。 因此可以被多个线程共享访问。

在这里插入图片描述
当我们用一个线程修改同一个变量 -> 安全
多个线程读取同一个变量 -> 安全
多个线程修改不同的变量 -> 安全
多个线程修改同一个变量 -> 不安全

1.3 原子性

多个线程修改同一个变量不安全的原因主要是因为修改变量的操作不是原子性的

原子性是指一个操作是不可中断的,要么全部执行完成,要么完全不执行。在多线程编程中,原子性保证了在并发环境下对共享资源进行操作时的正确性和一致性。

如果一个操作是原子的,那么在执行过程中不会被其他线程的干扰,同时也不会干扰其他线程的执行。当多个线程同时对共享资源进行读写操作时,如果没有原子性保证,就会导致数据不一致、并发错误、死锁等问题。因此,在并发编程中,需要特别注意保证操作的原子性。

1.4 可见性

可见性问题是指一个线程对共享变量的修改可能不会立即被其他线程所看到,导致线程间的数据不一致性。

在多线程环境中,每个线程都有自己的工作内存,其中包含了该线程使用到的共享变量的副本。当线程对共享变量进行读写操作时,实际上是在操作自己工作内存中的副本,而不是直接操作主内存中的真实变量。

当一个线程对共享变量进行修改后,必须将修改后的值刷新到主内存中,以便其他线程可以看到这个变化。而其他线程在读取共享变量时,也需要将主内存中最新的值加载到自己的工作内存中。

可见性问题主要有以下两种情况:

  1. 修改后的值没有立即被其他线程看到:当一个线程修改了共享变量的值,但该值并没有被及时刷新到主内存中,其他线程在读取该共享变量时仍然看到的是过期的旧值,而不是最新的修改值。
  2. 对共享变量的修改对其他线程来说是不可见的:一个线程修改了共享变量的值并已经将其写入主内存,而其他线程在读取共享变量时却没有及时从主内存中加载最新的值,而是仍然从自己的工作内存中读取旧值。

可见性问题可能导致线程安全问题,因为不同线程之间对共享变量的读写操作是不同步的。如果一个线程在写入一个共享变量时,另一个线程在读取该共享变量时没有及时看到最新的值,就会导致数据的不一致性和错误的结果。

Java 内存模型 (JMM): Java虚拟机规范中定义了Java内存模型.目的是屏蔽掉各种硬件和操作系统的内存访问差异,以实现让Java程序在各种平台下都能达到一致的并发效果.

在这里插入图片描述

  • 线程之间的共享变量存在 主内存 (Main Memory).
  • 每一个线程都有自己的 “工作内存” (Working Memory) .
  • 当线程要读取一个共享变量的时候, 会先把变量从主内存拷贝到工作内存, 再从工作内存读取数据.
  • 当线程要修改一个共享变量的时候, 也会先修改工作内存中的副本, 再同步回主内存.

由于每个线程有自己的工作内存, 这些工作内存中的内容相当于同一个共享变量的 “副本”. 此时修改线程1 的工作内存中的值, 线程2 的工作内存不一定会及时变化.

  1. 初始情况下, 两个线程的工作内存内容一致.
    在这里插入图片描述
  2. 一旦线程1 修改了 a 的值, 此时主内存不一定能及时同步. 对应的线程2 的工作内存的 a 的值也不一定能及时同步.

在这里插入图片描述
这个时候代码中就容易出现问题.

此时引入了两个问题:
一:为啥要整这么多内存?
二:为啥要这么麻烦的拷来拷去?

  1. 为啥整这么多内存?
    实际并没有这么多 “内存”. 这只是 Java 规范中的一个术语, 是属于 “抽象” 的叫法.所谓的 “主内存” 才是真正硬件角度的 “内存”. 而所谓的 “工作内存”, 则是指 CPU 的寄存器和高速缓存.

  2. 为啥要这么麻烦的拷来拷去?
    因为 CPU 访问自身寄存器的速度以及高速缓存的速度, 远远超过访问内存的速度(快了 3 - 4 个数量级, 也就是几千倍, 上万倍).

比如某个代码中要连续 10 次读取某个变量的值, 如果 10 次都从内存读, 速度是很慢的. 但是如果只是第一次从内存读, 读到的结果缓存到 CPU 的某个寄存器中, 那么后 9 次读数据就不必直接访问内存了. 效率就大大提高了.

那么接下来问题又来了, 既然访问寄存器速度这么快, 还要内存干啥??

答案就是一个字: 贵

在这里插入图片描述

1.5指令重排序

指令重排序是编译器和处理器为了优化代码执行效率而进行的一种手段。它可以改变代码中指令的执行顺序.

编译器和处理器进行指令重排序的目的是通过优化指令的执行顺序,减少指令之间的依赖性,提高指令级并行度,从而提高程序的执行效率。

JVM 和 CPU 指令集对指令重排序进行优化的原因是提高代码执行效率,减少指令的等待时间,充分利用处理器的并行执行能力,提高程序的性能和响应能力。

尽管指令重排序可以改善程序的执行效率,但在多线程环境下,指令重排序可能会导致线程安全问题。因此,在编写多线程代码时,需要使用适当的同步机制来保证线程之间的正确交互。

二:解决线程安全问题

2.1 synchronized关键字

如何解决线程不安全问题?需要从原因入手!能否让 count++变成原子的呢?当然有,加锁!!

在Java中,可以使用synchronized关键字来保证原子性。synchronized关键字用于修饰方法或代码块,确保同一时刻只有一个线程可以执行被修饰的代码。

  1. 同步方法:将关键字synchronized直接应用于方法。当一个线程进入这个方法时,它将锁住整个方法,其他线程必须等待该线程执行完毕才能进入该方法。
public synchronized void synchronizedMethod() {// 线程安全代码
}
  1. 同步代码块:将关键字synchronized应用于代码块,指定一个对象作为锁。当一个线程进入代码块时,它将锁住指定的对象,其他线程必须等待锁的释放才能进入该代码块,当这个线程执行完代码块的代码时,会自动解锁。
public void synchronizedBlock() {synchronized (lock) {// 线程安全代码}
}

加锁的意思就是获取到锁,解锁的意思就说释放锁,加锁意味着获得对某个资源的独占访问权限,而解锁则是释放这个资源,使其他线程可以访问它。

在以上示例中,lock可以是任意Java对象。如果多个线程共享同一个锁对象,那么同一时刻只能有一个线程执行代码块内的代码。

synchronized关键字通过获取和释放锁来实现原子性。当一个线程获取到锁时,其他线程将无法进入相同的同步方法或代码块。这确保了被synchronized修饰的代码在同一时刻只能由一个线程执行,从而保证了原子性。

当两个线程对同一个对象加锁时,它们会争夺这个对象的锁资源,即发生了锁竞争。只有一个线程能够获得该对象的锁资源,而另一个线程将被阻塞,直到获得锁的线程释放锁资源。如果两个线程针对不同的对象加锁是不会发生锁竞争的,各自获取各自的锁即可。

下面是一个简单的代码示例,展示了两个线程对同一个对象进行加锁的情况:

public class LockExample {private final Object lock = new Object();public void thread1() {synchronized (lock) {// 临界区1// 线程1获得了锁资源,执行一些操作}}public void thread2() {synchronized (lock) {// 临界区2// 线程2获得了锁资源,执行一些操作}}
}

在上面的代码示例中,lock对象是一个共享资源,两个线程thread1()thread2()分别对lock对象进行加锁。

当线程1执行到synchronized (lock)时,它会尝试获取lock对象的锁资源。如果锁资源可用,线程1将获得锁,并执行临界区1的代码。此时,如果线程2尝试执行synchronized (lock),它将无法获得锁资源,因为锁资源已经被线程1占用。线程2将被阻塞,直到线程1释放锁资源。

同样地,当线程2执行到synchronized (lock)时,它会尝试获取lock对象的锁资源。如果锁资源可用,线程2将获得锁,并执行临界区2的代码。此时,如果线程1尝试执行synchronized (lock),它将无法获得锁资源,因为锁资源已经被线程2占用。线程1将被阻塞,直到线程2释放锁资源。

理解 “阻塞等待”:

  • 针对每一把锁, 操作系统内部都维护了一个等待队列. 当这个锁被某个线程占有的时候, 其他线程尝试进行加锁, 就加不上了, 就会阻塞等待, 一直等到之前的线程解锁之后, 由操作系统唤醒一个新的线程, 再来获取到这个锁.

注意:

  • 上一个线程解锁之后, 下一个线程并不是立即就能获取到锁. 而是要靠操作系统来 “唤醒”. 这也就是操作系统线程调度的一部分工作.

假设有 A B C 三个线程, 线程 A 先获取到锁, 然后 B 尝试获取锁, 然后 C 再尝试获取锁, 此时 B和 C 都在阻塞队列中排队等待. 但是当 A 释放锁之后, 虽然 B 比 C 先来的, 但是 B 不一定就能获取到锁, 而是和 C 重新竞争, 并不遵守先来后到的规则.

2.1.1synchronized 的特性

当谈到synchronized关键字时,它具有三个重要的特性:互斥性、内存可见性和可重入性。下面将逐个解释这些特性:

  1. 互斥性:
    synchronized关键字用于保护临界区,确保同时只有一个线程可以执行临界区代码。这意味着当一个线程进入synchronized代码块时,其他线程将被阻塞,直到该线程执行完临界区代码并释放锁。

互斥性保证了线程之间不会相互干扰,避免了竞态条件(race condition)和数据不一致的问题。

  1. 内存可见性:

synchronized 的工作过程:

  1. 获得互斥锁
  2. 从主内存拷贝变量的最新副本到工作的内存
  3. 执行代码
  4. 将更改后的共享变量的值刷新到主内存
  5. 释放互斥锁

synchronized关键字还确保了线程之间的内存可见性。当线程进入synchronized代码块并获取锁时,它会把修改后的值刷新回主内存,并且当其他线程获取锁时,它们会重新从主内存中读取最新的值。这确保了共享变量的值在不同线程之间保持一致。

内存可见性避免了线程之间读取过时的数据或者脏数据,确保了程序的正确性和可靠性。

  1. 可重入性:
    synchronized 同步块对同一条线程来说是可重入的,不会出现自己把自己锁死的问题;

理解 “把自己锁死”:
一个线程没有释放锁, 然后又尝试再次加锁.

// 第一次加锁, 加锁成功
lock();
// 第二次加锁, 锁已经被占用, 阻塞等待.
lock();

按照之前对于锁的设定, 第二次加锁的时候, 就会阻塞等待. 直到第一次的锁被释放, 才能获取到第二个锁. 但是释放第一个锁也是由该线程来完成, 结果这个线程已经躺平了, 啥都不想干了, 也就无法进行解锁操作. 这时候就会死锁。

在这里插入图片描述

Java 中的 synchronized 是 可重入锁, 因此没有上面的问题.

代码示例:

static class Counter {public int count = 0;synchronized void increase() {count++;}synchronized void increase2() {increase();}
}

在上述的代码中,
increase 和 increase2 两个方法都加了 synchronized, 此处的 synchronized 都是针对 this 当前对象加锁的.

在调用 increase2 的时候, 先加了一次锁, 执行到 increase 的时候, 又加了一次锁. (上个锁还没释放, 相当于连续加两次锁)

这个代码是完全没问题的. 因为 synchronized 是可重入锁.

在可重入锁的内部, 包含了 “线程持有者” 和 “计数器” 两个信息.

  • 如果某个线程加锁的时候, 发现锁已经被人占用, 但是恰好占用的正是自己, 那么仍然可以继续获取到锁, 并让计数器自增

  • 解锁的时候计数器递减为 0 的时候, 才真正释放锁. (才能被别的线程获取到)

2.2.2synchronized的锁对象

  1. 直接修饰普通方法: 相当于对this加锁
public class SynchronizedDemo {public synchronized void methond() {}
}
  1. 修饰静态方法: 相当于对类对象加锁
public class SynchronizedDemo {public synchronized static void method() {}
}
  1. 修饰代码块: 明确指定锁哪个对象.
  1. 锁当前对象
public class SynchronizedDemo {public void method() {synchronized (this) {}}
}
  1. 锁类对象
public class SynchronizedDemo {public void method() {synchronized (SynchronizedDemo.class) {}}
}

我们重点要理解,synchronized 锁的是什么. 两个线程竞争同一把锁, 才会产生阻塞等待.
在这里插入图片描述

2.2.3 Java 标准库中的线程安全类

Java 标准库中很多都是线程不安全的. 这些类可能会涉及到多线程修改共享数据, 又没有任何加锁措施.

  • ArrayList
  • LinkedList
  • HashMap
  • TreeMap
  • HashSet
  • TreeSet
  • StringBuilder

但是还有一些是线程安全的. 使用了一些锁机制来控制.

  • Vector (不推荐使用)
  • HashTable (不推荐使用)
  • ConcurrentHashMap
  • StringBuffer

在这里插入图片描述
StringBuffer 的核心方法都带有 synchronized .

还有的虽然没有加锁, 但是不涉及 “修改”, 仍然是线程安全的

  • String

2.2volatile 关键字

volatile关键字在Java中用于确保多线程环境下的变量在不同线程之间的可见性。当一个变量被声明为volatile时,它的值会被强制从主内存中读取,而不是从线程的本地缓存中读取。

在多线程环境中,每个线程都有自己的本地缓存,这样可以提高运行效率。然而,由于本地缓存与主内存之间的同步延迟,当一个线程修改了变量的值时,其他线程可能无法立即感知到这个修改,而继续读取本地缓存中的旧值。这就导致了内存可见性问题。

volatile关键字的作用就是解决这个问题。当一个变量被声明为volatile时,所有对该变量的读写操作都会直接同步到主内存中,这样其他线程就能够及时地看到最新的值。换句话说,volatile关键字可以防止指令重排序优化,并保证读写操作的顺序性和原子性。

代码在写入 volatile 修饰的变量的时候,

  • 改变线程工作内存中volatile变量副本的值
  • 将改变后的副本的值从工作内存刷新到主内存

代码在读取 volatile 修饰的变量的时候,

  • 从主内存中读取volatile变量的最新值到线程的工作内存中
  • 从工作内存中读取volatile变量的副本

加上 volatile , 强制读写内存. 速度是慢了, 但是数据变的更准确了.

注意:volatile关键字只能保证变量的可见性,并不能解决线程安全问题。如果多个线程同时修改一个变量的值,那么仍然会存在竞态条件和数据一致性问题。在这种情况下,需要使用synchronized关键字或其他线程安全的机制来保证数据的一致性。

下面是一个简单的示例,演示了volatile关键字的使用:

public class Worker implements Runnable {private volatile boolean isRunning = true;public void run() {while (isRunning) {// 业务逻辑}}public void stop() {isRunning = false;}
}

在上述代码中,isRunning变量被声明为volatile,所以当调用stop()方法时,其他线程能够立即看到isRunning变量的修改,从而终止循环并停止线程的执行。

volatile还有一个效果:禁止指令重排序

注意:

  • synchronized 既能保证原子性, 也能保证内存可见性.
  • volatile只能保证内存可见性,不能保证原子性

三:wait 和 notify

由于线程之间是抢占式执行的,因此线程之间执行的先后顺序难以预知,但是实际开发中有时候我们希望合理的协调多个线程之间的执行先后顺序。

完成这个协调工作, 主要涉及到这几个个方法

  • wait() / wait(long timeout): 让当前线程进入等待状态.
  • notify() / notifyAll(): 唤醒在当前对象上等待的线程.

注意: wait, notify, notifyAll 都是 Object 类的方法.

3.1 wait

对于Java中的wait,它是一个用于多线程同步的方法。下面是关于wait方法的解释:

  1. wait使当前执行代码的线程进行等待,并将线程放入等待队列中。同时,wait方法会释放当前的锁,让其他线程有机会获得这个锁。

  2. wait方法必须与synchronized关键字一起使用。如果在没有synchronized的代码块中使用wait方法,会直接抛出异常。

  3. wait方法在满足以下条件之一时结束等待:

    • 其他线程调用了该对象的notify方法,唤醒了等待线程。
    • 等待时间超时。wait方法提供了一个带有timeout参数的版本,可以指定等待的时间。
    • 其他线程调用了等待线程的interrupted方法,导致wait方法抛出InterruptedException异常。

wait方法必须在synchronized代码块或synchronized方法中使用的主要原因是它依赖于对象的监视器(也称为锁)。当一个线程调用wait方法时,它会释放持有的锁,然后进入等待状态,直到其他线程调用相同对象上的notifynotifyAll方法来唤醒它。

在没有synchronized的代码块中调用wait方法会导致IllegalMonitorStateException异常。这是因为wait方法需要获取对象的监视器才能释放锁并等待,而没有synchronized关键字的代码块无法给对象上锁。因此,调用wait方法会让JVM抛出异常,以确保遵守正确的同步约定。

代码示例:

public static void main(String[] args) throws InterruptedException {Object object = new Object();synchronized (object) {System.out.println("等待中");object.wait();System.out.println("等待结束");}
}

这样在执行到object.wait()之后就一直等待下去,那么程序肯定不能一直这么等待下去了。这个时候就需要使用到了另外一个方法唤醒的方法notify()。

3.2 notify()

notify 方法是唤醒等待的线程.

方法notify()也要在同步方法或同步块中调用,该方法是用来通知那些可能等待该对象的对象锁的其它线程,对其发出通知notify,并使它们重新获取该对象的对象锁。

如果有多个线程等待,则有线程调度器随机挑选出一个呈 wait 状态的线程。(并没有 “先来后到”)

以下是notify方法的详细解释:

  • notify方法是在Object类中定义的,它用于唤醒等待在同一对象上的某个线程。当线程调用wait方法在对象上等待时,它会释放对象的锁,并进入等待状态。
  • 如果有其他线程调用了相同对象上的notify方法,那么其中的一个线程(无法确定具体是哪个线程)会被唤醒,并且开始竞争对象的锁。
  • notify方法只会唤醒一个等待线程。如果有多个线程在等待状态,不能确定哪个线程会被唤醒。
  • 唤醒的线程然后可以继续执行,但是它需要重新获得对象的锁才能进入同步代码块或方法。
  • 如果没有线程在等待该对象的锁,调用notify方法不会有任何影响。

下面是一个使用notify方法唤醒线程的简单Java代码示例:

class MyThread extends Thread {private final Object lock;public MyThread(Object lock) {this.lock = lock;}public void run() {synchronized (lock) {System.out.println("线程开始等待");try {lock.wait();} catch (InterruptedException e) {e.printStackTrace();}System.out.println("线程被唤醒");}}
}public class Main {public static void main(String[] args) throws InterruptedException {Object lock = new Object();MyThread thread = new MyThread(lock);synchronized (lock) {thread.start();Thread.sleep(2000); // 主线程休眠2秒钟,模拟执行一些其他操作System.out.println("主线程执行notify操作");lock.notify();}}
}

在上面的示例中,我们创建了一个MyThread类继承自Thread类,该线程在synchronized块中调用wait方法进行等待。在主线程中,我们先让主线程休眠2秒钟,然后调用notify方法来唤醒等待的线程。

当运行该程序时,它会输出以下内容:

线程开始等待
主线程执行notify操作
线程被唤醒

可以看到,等待的线程被成功唤醒并继续执行。请注意,在使用notify方法前后,需要通过synchronized关键字锁住共享对象。

notify方法只是唤醒某一个等待线程. 使用notifyAll方法可以一次唤醒所有的等待线程.

注意:
notifyAll()方法是用于唤醒所有等待中的线程。它会将所有等待中的线程从等待状态转变为可执行状态。尽管被唤醒的线程数量可能是多个,但在实际执行时,这些线程仍然需要竞争锁对象的所有权。只有获取到锁对象的线程才能够执行,而其他线程仍然需要等待。因此,尽管被同时唤醒,实际执行的先后顺序还是要根据线程竞争锁的情况来确定。

理解 notify 和 notifyAll:

  • notify 只唤醒等待队列中的一个线程. 其他线程还是乖乖等着

在这里插入图片描述

  • notifyAll 一下全都唤醒, 需要这些线程重新竞争锁

在这里插入图片描述

3.3 wait 和 sleep 的对比

当涉及到多线程编程时,我们经常会遇到waitsleep这两个方法。尽管它们都可以暂停当前线程的执行,但在功能和应用上存在一些异同点。下面是它们的详细解释:

  1. wait()方法:

    • wait方法属于Object类,在Java中,所有对象都可调用该方法。
    • 调用wait方法会使线程进入等待状态,直到其他线程调用同一对象上的notifynotifyAll方法来唤醒等待线程。
    • 在调用wait方法之前,线程必须获取对象的监视器锁,即在同步块或同步方法中调用。
    • 线程在等待状态时会释放对象的监视器锁,允许其他线程访问这个对象。
    • wait方法可以通过以下方式被唤醒:
      • 其他线程调用了同一对象上的notify方法,唤醒一个等待的线程;
      • 其他线程调用了同一对象上的notifyAll方法,唤醒所有等待的线程;
      • 当前线程被中断(调用了Thread.interrupt()方法)。
  2. sleep()方法:

    • sleep方法属于Thread类,可以直接在任何线程中调用。
    • 调用sleep方法不会释放锁或资源,只是让线程休眠一段指定的时间,然后线程继续执行。
    • sleep方法传入一个时间参数,以指定线程挂起的时间,单位是毫秒或纳秒。
    • sleep期间,线程不会释放锁或资源,其他线程无法访问该线程持有的资源。
    • sleep方法的唯一方法被打断休眠是通过线程中断(调用了Thread.interrupt()方法)。

综上所述,主要的区别在于:

  • wait方法是Object类的方法,用于线程之间的同步和通信,需要在同步块或同步方法中使用,调用时会释放锁,只能被唤醒或者中断才能继续执行;
  • sleep方法是Thread类的方法,用于线程的暂停一段指定时间,不会释放锁,只能通过时间结束或被中断才能继续执行。

两者的使用场景和目的也不同,需要根据具体需求来选择使用哪个方法。

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

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

相关文章

【AIGC核心技术剖析】扩大富有表现力的人体姿势和形状估计SMPLer-X模型

富有表现力的人体姿势和形状估计 (EHPS) 将身体、手和面部运动捕捉与众多应用结合起来。尽管取得了令人鼓舞的进展,但当前最先进的方法仍然在很大程度上依赖于有限的训练数据集。在这项工作中,我们研究了将 EHPS 扩展到第一个通用基础模型(称为 SMPLer-X),以 ViT-Huge 作为…

【译】快速开始 Compose 跨平台项目

原文&#xff1a; Compose Multiplatform application 作者&#xff1a;JetBrains 注意 Compose Multiplatform 中的 iOS 部分目前处于 Alpha 状态。以后可能会有不兼容的更改&#xff0c;届时也许需要手动进行迁移。 你可以使用这个模板来开发同时支持桌面、安卓和 iOS 的跨平…

GitHub验证的2FA

一、 起因&#xff1a; GitHub需要双重身份验证 (2FA) 是登录网站或应用时使用的额外保护层。启用 2FA 时&#xff0c;必须使用您的用户名和密码登录&#xff0c;并提供另一种只有您知道或可以访问的身份验证形式。 二、解决&#xff1a; 2.1 这里使用chrome的身份验证插件进…

在emacs中,设置latex的主文档

文档&#xff1a; chapter1.tex chapter2.tex main.tex 在chapter1.tex中&#xff0c;先按下 ctrlc ctrln&#xff0c;再按下ctrlc ctrla&#xff0c;在下方的提示框中输入主文档。

支付宝小程序介入人脸识别(金融级--前端部分)

在这里只做前端部分说明: 详情参考文档:如何通过集成支付宝小程序唤起实人认证服务_实人认证-阿里云帮助中心 操作步骤 调用 API 发起认证。 发起认证服务。 调用 startBizService 接口请求认证。 function startAPVerify(options, callback) {my.call(startBizService, {n…

AWS S3加密

Hello大家好&#xff61; 在本课时我们将讨论S3加密相关的内容。 S3加密相关是认证考试的一个重要的主题考点&#xff0c;您需要了解亚马逊S3的几种不同类型的加密方式。| 首先是静态数据的加密&#xff0c;静态数据加密是指数据存储在亚马逊S3 数据中心的磁盘上时&#xff0…

stable diffusion和midjourney哪个好

midjourney和stable diffusion哪个好&#xff1f;midjourney和stable diffusion的区别&#xff1f;那么今天就从这2款软件入手&#xff0c;来探索一下他们的功能的各项区别吧&#xff0c;让你选择更适合你的一款ai软件。 截至目前&#xff0c;我们目睹了生成式人工智能工具的在…

Linux —— 网络基础(一)

目录 一&#xff0c;计算机网络背景 二&#xff0c;网络协议初识 三&#xff0c;网络传输基本流程 四&#xff0c;网络中的地址管理 一&#xff0c;计算机网络背景 网络发展 独立模式&#xff0c;计算机之间相互独立&#xff1b;网络互联&#xff0c;多台计算机连接在一起…

【OpenCV实现图片以及视频的读取、显示、保存以及绘图函数】

文章目录 图片视频从文件读取视频保存一个视频绘图函数 图片 OpenCV&#xff08;Open Source Computer Vision Library&#xff09;是一个广泛应用于计算机视觉和图像处理领域的开源库。它提供了丰富的图像处理工具和算法&#xff0c;使得开发者能够轻松实现各种图像处理任务。…

科普丨语音芯片烧录流程概述

语音芯片的烧录是将特定的固件或软件加载到芯片中&#xff0c;以使其能够执行特定的语音处理功能。以下是一般的语音芯片烧录过程&#xff1a; 1. 准备固件或软件&#xff1a;开发人员需要编写或获取特定的固件或软件&#xff0c;这些固件或软件包含了语音处理算法和功能的代码…

【周末闲谈】VR新视界,“眼”见未来

个人主页&#xff1a;【&#x1f60a;个人主页】 系列专栏&#xff1a;【❤️周末闲谈】 系列目录 ✨第一周 二进制VS三进制 ✨第二周 文心一言&#xff0c;模仿还是超越&#xff1f; ✨第二周 畅想AR 文章目录 系列目录前言虚拟现实(VR)技术虚拟现实技术的原理虚拟现实技术发…

高精度时间测量(TDC)电路MS1022

MS1022 是一款高精度时间测量电路&#xff0c;内部集成了模拟比 较器、模拟开关、施密特触发器等器件&#xff0c;从而大大简化了外 围电路。同时内部增加了第一波检测功能&#xff0c;使抗干扰能力大 大提高。通过读取第一个回波脉冲的相对宽度&#xff0c;用户可以获 得接…

laravel的默认首页怎么改-laravel框架默认欢迎页面如何修改

laravel的默认首页怎么改 搭建好的laravel的默认首页怎么改 我们有两种改动方式&#xff1a; 第一种修改默认路由&#xff1a; 下一步是要移除Laravel应用程序默认的欢迎页路由。这个路由可以在routes/web.php文件的顶部找到&#xff0c;看起来类似于以下代码&#xff1a; …

OpenGL —— 2.7、绘制多个自旋转的贴图正方体(附源码,glfw+glad)

源码效果 C源码 纹理图片 需下载stb_image.h这个解码图片的库&#xff0c;该库只有一个头文件。 具体代码&#xff1a; vertexShader.glsl #version 330 corelayout(location 0) in vec3 aPos; layout(location 1) in vec2 aUV;out vec2 outUV;uniform mat4 _modelMatrix; …

微信小程序进阶——Flex弹性布局轮播图会议OA项目(首页)

目录 一、Flex弹性布局 1.1 什么是Flex弹性布局 1.1.1 详解 1.1.2 图解 1.1.3 代码演示效果 1.2 Flex弹性布局的核心概念 1.3 Flex 弹性布局的常见属性 1.4 Flex弹性布局部分属性详解 1.4.1 flex-direction属性 1.4.2 flex-wrap属性 1.4.3 flex-flow属性 1.4.4 ju…

DFS(分布式文件系统)与 DFSR(分布式文件系统复制)的区别

DFS&#xff08;分布式文件系统&#xff09;和 DFSR&#xff08;分布式文件系统复制&#xff09;是两种不同的技术&#xff0c;尽管它们在名称上有一些相似之处&#xff0c;但它们的用途和功能有所不同。 DFS&#xff08;分布式文件系统&#xff09; DFS 是一种用于创建和管理…

Win10系统开机启动文件夹在哪里找?

Win10系统开机启动文件夹在哪里找&#xff1f;Win10系统开机启动文件夹是一个非常重要的目录&#xff0c;它决定了电脑在开机的时候&#xff0c;会有哪些应用程序是自动启动。但是&#xff0c;很多新手用户不知道Win10电脑内开机启动文件夹的具体位置&#xff0c;下面小编介绍开…

eNSP笔记②

动态路由 RIP [适用于小型网络] 静态路由是加上非直连的网段&#xff0c;动态路由是加上直连的网段 动态路由就是要宣告它要去的网段&#xff0c;在图中可以看到&#xff0c;一台路由需要宣告两个网段&#xff0c;路由A分别宣告10.0.0.0与192.168.1.0&#xff0c;路由B宣告10.…

ArcGIS在VUE框架中的构建思想

项目快要上线了&#xff0c;出乎意料的有些空闲时间。想着就把其他公司开发的一期代码里面&#xff0c;把关于地图方面的代码给优化一下。试运行的时候&#xff0c;客户说控制台有很多飘红的报错&#xff0c;他们很在意&#xff0c;虽然很不情愿&#xff0c;但能改的就给改了吧…

JSX的本质

一、本质 React.createElement即h函数&#xff0c;返回vnode第一个参数&#xff0c;可能是组件&#xff0c;也可能是html tag组件名&#xff0c;首字母必须大写&#xff08;React规定&#xff09; 二、babel试一试 &#xff08;babel集成了jsx的编译环境&#xff09; // JSX…