JUC并发编程2(高并发,AQS)

JUC

AQS核心

当有线程想获取锁时,其中一个线程使用CAS的将state变为1,将加锁线程设为自己。当其他线程来竞争锁时会,判断state是不是0,不是自己就把自己放入阻塞队列种(这个阻塞队列是用双向链表实现),当这个线程使用完,会把state变为0,该state使用volatile修饰。在AQS内部,每个Node节点都是等待锁的线程,队列中每个排队的个体就是一个Node节点,它的等待状态waitState成员变量,也是volatile修饰,Node节点里也记录该线程是否不再等待状态,还记录锁的模式独占锁还是共享锁。

三大核心:

  • state 状态,代表加锁状态,初始值是0 (state是被volatile修饰)
  • 获取到锁的线程
  • 还有一个阻塞队列(双向队列)

AQS锁有关:

  • ReentrantLock(可重入锁)
  • ReentrantReadWriteLock.ReadLock(可重入读写锁中的读锁)
  • ReentrantReadWriteLock.WriteLock(可重入读写锁中的写锁)
  • CountDownLatch
  • CyclickBarrier
  • Semaphore

Lock使用方式:

 Lock lock = new ReentrantLock();

 public  void doTicket(){ 
    lock.lock();  //加锁
    try {
       System.out.println(Thread.currentThread().getName());
    } catch (Exception e) {
       e.printStackTrace();
    } finally {
       lock.unlock();  // 解锁
    }
 }

Synchronized 和 Lock区别

  • Synchronized 内置的Java关键字,Lock是一个Java类

  • Synchronized 无法判断获取锁的状态,Lock可以判断

  • Synchronized 会自动释放锁,lock必须要手动加锁和手动释放锁!可能会遇到死锁

  • Synchronized 是可重入锁,不可以中断的,非公平的;Lock,可重入的,可以判断锁,可以自己设置

读写锁

ReadWriteLock

  • 读写锁:更加细粒度的锁
  • 读-读:可以共存
  • 读-写:不能共存
  • 写-写:不能共存

它允许读读共存,读写和写写是互斥的,适合读多写少的场景,但会有写锁饥饿问题。
锁降级:写锁可以降级到读锁,但读锁不能升级到写锁;

public class ReadWriteLockDemo {
    public static void main(String[] args) {
        MyCache mycache = new MyCache();
        //开启5个线程 写入数据
        for (int i = 1; i <=5 ; i++) {
            int finalI = i;
            new Thread(()->{ 
                mycache.put(String.valueOf(finalI),String.valueOf(finalI));
            }).start();
        }
        //开启10个线程去读取数据
        for (int i = 1; i <=10 ; i++) {
            int finalI = i;
            new Thread(()->{
                String o = mycache.get(String.valueOf(finalI));
            }).start();
        }
    }
}

class MyCache{

    private volatile Map<String,String> map = new HashMap<>();
    //普通锁
    //private Lock lock = new ReentrantLock();
	//使用读写锁
    private ReadWriteLock lock = new ReentrantReadWriteLock();

    public void put(String key,String value){
        //写锁
        lock.writeLock().lock();
        try {
            //写入
            System.out.println(Thread.currentThread().getName()+" 线程 开始写入");
            map.put(key, value);
            System.out.println(Thread.currentThread().getName()+" 线程 写入完成");
        } finally {
            lock.writeLock().unlock();
        }
    }

    public String get(String key){
        //读锁
        lock.readLock().lock();
        String o;
        try {
            System.out.println(Thread.currentThread().getName()+" 线程 开始读取");
            o = map.get(key);
            System.out.println(Thread.currentThread().getName()+" 线程 读取完成");
        } finally {
            lock.readLock().unlock();
        }
        return o;
    }
}

对于读取,我们运行多个线程同时读取,也能在一定程度上提高效率。

StampedLock

JDK8新增的读写锁(邮戳锁),它采用乐观锁,其他线程尝试获取锁不会被阻塞,对读锁优化。获取锁的方法,返回一个邮戳Stamp,邮戳Stamp为0表示获取失败,其他是成功;释放锁的方法,需要一个邮戳Stamp与成功获取锁的邮戳Stamp一致,但它不支持可重入锁;

StampedLock stampedLock = new StampedLock();
// 乐观锁
public void tryOptimisticRead() {
       
	long stamp = stampedLock.tryOptimisticRead();  // 乐观读
    int result = number; // 获取一下最新的值
    //判断是否发生改变 stampedLock.validate()
    System.out.println("alidate方法值(true无修改,false有修改)" + "\t" + stampedLock.validate(stamp));
    if (!stampedLock.validate(stamp)) {
        System.out.println("有人修改过------有写操作");
        stamp = stampedLock.readLock();
        try {
            System.out.println("从乐观读 升级为 悲观读");
            result = number;
            System.out.println("重新悲观读后result:" + result);
        } finally {
            stampedLock.unlockRead(stamp);
        }
    }
}      

常用的辅助类

CountDownLatch

这个类使一个线程等待其他线程各自执行完毕后再执行。

主要方法:

  • countDown 减一操作;
  • await 等待计数器归零。
public static void main(String[] args) throws InterruptedException {
    //总数6个
    CountDownLatch countDownLatch = new CountDownLatch(6);

    for (int i = 1; i <= 6; i++) {
        new Thread(()->{
            System.out.println(Thread.currentThread().getName() +" 执行do");
            //每个线程都数量-1
            countDownLatch.countDown();
        },String.valueOf(i)).start();
    }
    //等待计数器归零
    countDownLatch.await();
    System.out.println("必须其他线程都执行完,在执行这里");
    //最后执行的...
}

CyclickBarrier

用于对多个线程任务进行同步执行。

主要方法:

  • await 在所有线程任务都到达之前,线程任务都是阻塞状态
public static void main(String[] args) {
    CyclicBarrier cyclicBarrier = new CyclicBarrier(7,()->{
        System.out.println("召唤神龙的线程~");
    });

    for (int i=1;i<=7;i++){
        int atI = i;
        new Thread(()->{
            try {
                System.out.println(Thread.currentThread().getName()+" 收集了第" + atI +"颗龙珠");
                cyclicBarrier.await(); //加法计数 等待
            } catch (InterruptedException e) {
                e.printStackTrace();
            } catch (BrokenBarrierException e) {
                e.printStackTrace();
            }
        },"线程"+i).start();
    }
}

应用:

  • CyclickBarrier可以根据基于子线程进行处理其他线程的结果,处理比较复杂的业务。并且可以通过reset方法重新执行方法。
  • CountDownLoatch则必须在主线程才能处理,一般用于任务执行初始化数据

Semaphore

信号量,在信号量定义两种操作:

  • acquire(获取)当一个线程调用acquire操作,它通过成功获取信号量(信号量-1),有阻塞,直到有线程释放信号量,或者超时。
  • release(释放)实际上将信号量的值+1,然后唤醒等待的线程。
public static void main(String[] args) {
    //停车位为3个
    Semaphore semaphore = new Semaphore(3);

    for (int i=1 ; i<=10; i++){
        int atI = i;
        new Thread(()->{
            try {
                semaphore.acquire(); //得到
                System.out.println(Thread.currentThread().getName() + "  抢到停车位" + atI);
                TimeUnit.SECONDS.sleep(2);
                System.out.println(Thread.currentThread().getName() + "  离开停车场");
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                semaphore.release();  //释放
            }
        },"线程"+i).start();
    }
}

作用: 多个共享资源互斥的使用! 并发限流,控制最大的线程数!

异步回调

CompletableFuture

就是另启一个线程来完成调用中的部分计算,使调用继续运行或返回,而不需要等待计算结果。一个completetableFuture就代表了一个任务。相当于前端的ajax异步请求。

没有返回值的异步回调:

@Test
void test1() throws ExecutionException, InterruptedException {

    System.out.println(System.currentTimeMillis());
    System.out.println("---------------------");
    CompletableFuture<Void> future = CompletableFuture.runAsync(()->{
        //发起一个异步任务
        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName()+".....");
    });
    System.out.println(System.currentTimeMillis());
    System.out.println("------------------------------");
    //输出执行结果
    System.out.println(future.get());  //获取执行结果
}

有返回值的异步回调supplyAsync:

void test2() throws ExecutionException, InterruptedException {
    System.out.println(System.currentTimeMillis());
    System.out.println("---------------------");
    //有返回值的异步回调
    CompletableFuture<Integer> completableFuture=CompletableFuture.supplyAsync(()->{
        System.out.println(Thread.currentThread().getName());
        try {
            TimeUnit.SECONDS.sleep(2);
            //自定义手动的异常
            int i=1/0;
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return 1024;
    });
    System.out.println(completableFuture.whenComplete((t, u) -> {
        //success 回调
        System.out.println("t=>" + t); //正常的返回结果
        System.out.println("u=>" + u); //抛出异常的 错误信息
    }).exceptionally((e) -> {
        //error回调  如果异常,返回404
        System.out.println(e.getMessage());
        return 404;
    }).get());
}

ThreadLocal

它是线程本地变量,解决多线程并发时访问共享变量的问题。它并不解决线程之间共享数据的问题,它适用于变量在线程之间隔离且在方法间共享的场景,每个线程持有一个只属于自己的专属的Map并维护ThreadLocal对象与具体实例的映射,该Map只有被持有它的线程访问,就没有线程安全问题以及锁的问题。

# 使用
static ThreadLocal<Object> threadLocal = new ThreadLocal<>();

# 初始化
private static final ThreadLocal<Object> threadLocal = ThreadLocal.withInitial(Object::new);
private static final ThreadLocal<Object> threadLocal = ThreadLocal.withInitial(()->{初始化的值});

# 设置线程本地变量的内容
threadLocal.set(user);

# 获取线程本地变量的内容
threadLocal.get();

# 移除线程本地变量
threadLocal.remove();

线程中断机制

一个线程中断,应该由线程自己停止,不能由其他线程进行停止。

# 中断此线程
interrupt()
# 判断当前线程是否被中断
isInterrupted()
# 案例
Thread t1 = new Thread(() -> {
   while (true) {
         // Thread.currentThread().isInterrupted() 判断线程是否被中断,true:中断状态
         if (Thread.currentThread().isInterrupted()) {
             System.out.println(Thread.currentThread().getName() + "\t 【实现方式3】,程序停止");
             break;
    }
    System.out.println("t1 -----正在运行!!!");
} }, "t1");
t1.start();
System.out.println("-----t1的默认中断标志位:" + t1.isInterrupted());

LockSupport

可以阻塞当前线程以及唤醒指定被阻塞的线程,有park()和unpark()进行通知与唤醒。

 Thread t1 = new Thread(() -> {
    TimeUnit.SECONDS.sleep(3);
    System.out.println(Thread.currentThread().getName() + "\t ----【t1】开始执行" + System.currentTimeMillis());
    LockSupport.park();
    System.out.println(Thread.currentThread().getName() + "\t ----【t1】 被唤醒" + System.currentTimeMillis());
}, "t1");
t1.start();

new Thread(() -> {
     LockSupport.unpark(t1);
     System.out.println(Thread.currentThread().getName() + "\t 【t2】唤醒t1线程----发出通知");
 }, "t2").start();

volatile关键字

Volatile 是 Java 虚拟机提供 轻量级的同步机制。

volatile三大特性

  1. 保证可见性 (要么都完成,要么都不完成)
  2. 不保证原子性
  3. 禁止指令重排

JMM:Java内存模型,是java虚拟机规范中所定义的一种内存模型,Java内存模型是标准化的,屏蔽掉了底层不同计算机的区别(注意这个跟JVM完全不是一个东西,现在还有小伙伴搞错的)。

关于JMM的一些同步的约定

  1. 线程解锁前,必须把共享变量立刻刷回主存。
  2. 线程加锁前,必须读取主存中的 新值到工作内存中!
  3. 加锁和解锁是同一把锁

首先我们要了解JMM (Java内存模型),线程分为 工作内存主内存。 图中,有两个线程,每个线程都有属于自己的工作内存;我们首先在主存中有Flag变量,有一个线程A,从主存中读取值,把值加载工作内容中,这样工作内存中也有一个Flag变量。线程A修改值先修改工作内容,并把值再写入到主存中。这个流程叫内存交互。

内存交互有8种:

  • lock (锁定):作用于主内存的变量,把一个变量标识为线程独占状态
  • unlock (解锁):作用于主内存的变量,它把一个处于锁定状态的变量释放,释放后的变量才可以被其他线程锁定
  • read (读取):作用于主内存变量,它把一个变量的值从主内存传输到线程的工作内存中,以便 随后的load动作使用
  • load (载入):作用于工作内存的变量,它把read操作从主存中变量放入工作内存中
  • use (使用):作用于工作内存中的变量,它把工作内存中的变量传输给执行引擎,每当虚拟机 遇到一个需要使用到变量的值,就会使用到这个指令
  • assign (赋值):作用于工作内存中的变量,它把一个从执行引擎中的值放入工作内存的变量副本中
  • store (存储):作用于主内存中的变量,它把一个从工作内存中一个变量的值传送到主内存中, 以便后续的write使用
  • write (写入):作用于主内存中的变量,它把store操作从工作内存中得到的变量的值放入主内存的变量中

问题: 程序不知道主内存的值已经被修改过了,但是volatile关键字,可以解决这个问题。

例子:private volatile boolean flag = false;

volatile 是保证原子性的,

原子性 : 不可分割

线程A在执行任务的时候,不能被打扰的,也不能被分割。要么同时成功,要么同时失败。

解决原子性问题,加锁/使用原子类

原子变量

原子变量:来源于 java.util.concurrent.atomic 类的小工具包,支持在单个变量上解除锁的线程安全编程,包下提供了常用的原子变量:
- AtomicBoolean 、AtomicInteger 、AtomicLong 、 AtomicReference
- AtomicIntegerArray 、AtomicLongArray
- AtomicMarkableReference
- AtomicReferenceArray

  • AtomicStampedReference

类中的变量都是volatile类型:保证内存可见性
使用CAS算法:保证数据的原子性

例子:

public class VDemo02 {
    // volatile 不保证原子性
    // private volatile static int num = 0;
    // 原子类的 Integer
    private volatile static AtomicInteger num = new AtomicInteger();
    
    public static void add(){
        // num++;   // 不是一个原子性操作
        // 原子类的 加减  // AtomicInteger + 1 方法, CAS
        num.getAndIncrement();
    }
    
    public static void main(String[] args) {
        //理论上num结果应该为 2 万
        for (int i = 1; i <= 20; i++) {
            new Thread(() -> {
                for (int j = 0; j < 1000; j++) { add(); }
            }).start();
        }
        while (Thread.activeCount() > 2) { Thread.yield();}
        System.out.println(Thread.currentThread().getName() + " " + num);
    }

}

指令重排

你写的程序,计算机并不是按照你写的那样去执行的。

源代码–>编译器优化的重排–> 指令并行也可能会重排–> 内存系统也会重排—> 执行

Volatile 是可以保持 可见性。不能保证原子性,由于内存屏障(一种屏障指令,使用CPU对屏障指令前后发出内存操作,执行一个排序约束),可以保证避免指令重排的现象产生!

CAS算法

(Compare-And-Swap) 是一种硬件对并发的支持,针对多处理器操作而设计的处理器中的一种特殊指令,用于管理对共享数据的并发访问。

CAS 比较并交换,当且仅当V==A时,B的值才更新给A,否则将不做任何操作。

public static void main(String[] args) {
    // 原子类
    AtomicInteger atomicInteger = new AtomicInteger(2020);
    // public final boolean compareAndSet(int expect, int update) ,参数1 期望的值,参数2,更新的值
    // 如果我期望的值达到了,那么就更新,否则,就不更新, CAS 是CPU的并发原语!
    // compareAndSet 也是CAS
    atomicInteger.compareAndSet(2020, 2021);
    // 输出值
    System.out.println(atomicInteger.get());
}

Unsafe 类:

底层调用这个类,JAVA无法操作内存,C++可以操作内存,但是JAVA可以调用C++,Unsafe 类,就是java可以通过这个类操作内存。

CAS : 比较当前工作内存中的值和主内存中的值,如果这个值是期望的,那么则执行操作!如果不是就 一直循环!

缺点:

  • 循环会耗时
  • 一次性只能保证一个共享变量的原子性
  • ABA问题

解决ABA 问题

比如一个线程1从内存位置V中取出A ,线程2也执行,将A–>B–>A,这时线程1进行CAS操作发现内存仍是A,然后线程1操作成功。

就是有一个线程速度快,把一个值改变,然后在改变原来的值,其他线程进行CSA,比较并交换,发现值一样,替换新值,并不知道他之前有人已经改过值了。

解决这问题:

  • 将类变成原子类
  • 操作过程添加版本号
// 原子引入
static AtomicStampedReference<Integer> atomic = new AtomicStampedReference<>(1,1);
public static void main3(String[] args) {

    new Thread(()->{
        // 获得版本号
        int stamp = atomic.getStamp();
        System.out.println("线程a1=>"+stamp);
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        // 比较并交换 ,增加一个参数,版本号 ,然后再修改版本号的值
        // 获取最新的版本号  atomic.getStamp()
        atomic.compareAndSet(1, 2, atomic.getStamp(), atomic.getStamp() + 1);
        System.out.println("线程a2=>"+ atomic.getStamp());
        System.out.println(atomic.compareAndSet(2, 1, atomic.getStamp(), atomic.getStamp() + 1));
        System.out.println("线程a3=>"+atomic.getStamp());
    },"线程A").start();

    // 乐观锁的原理相同!
    new Thread(()->{
        int stamp = atomic.getStamp(); // 获得版本号
        System.out.println("线程b1=>"+stamp);
        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(atomic.compareAndSet(1, 6, stamp, stamp + 1));
        System.out.println("线程B2=>"+atomic.getStamp());
    },"线程B").start();

}

原子类

基本类型类:AtomicBoolean、AtomicIntegern、AtomicLong
数组类型类:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray
引用类型类:AtomicReference、AtomicReferenceFieldUpdater、AtomicMarkableReference、AtomicMarkableReference
LongAdder的效率比AtomicLong高(减少自旋次数)

AtomicLong

  • 原理:CAS + 自旋(它是多线程对单个热点值进行原子操作)
  • 缺点:当线程大量自旋会导致CPU生高,

LongAdder

  • 原理:CAS + Base + Cell数组(用空间换时间,采用分散热点数据)
  • 缺点:它的sum求和,对于最后结果不够准确

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

聚丙烯PP材料粘接方法?泰达克TADHE专用于PP材料塑料粘接的UV胶水提供了解决方案

PP&#xff08;聚丙烯&#xff09;&#xff0c;简称PP。 PP是一种疏水性的塑料&#xff0c;需要特殊的处理后再和胶水粘接&#xff0c;以确保良好的粘接效果。常用的PP材料粘接方法&#xff1a; ​1.表面处理 因PP表面的疏水性&#xff0c;可以先进行表面处理。使用酒精或丙酮…

完整的项目源码!在线考试完整系统源码(可接私活)

最近有一些读者问我有没有完整的基于SpringbootVue的项目源码&#xff0c;今天给大家整理了一下&#xff0c;并且录制了搭建的教程&#xff0c;无偿分享给大家。 一、系统运行图 1、登陆页面 2、后台管理 3、全套环境资源 ​源码文件部分截图&#xff0c;带视频教程 ​ 在实际…

【图论】Dijkstra单源最短路径-朴素方法-简单模板(迪杰斯特拉算法)

Dijkstra单源最短路径 问题描述 输入n 表示n个结点&#xff0c;m表示m条边&#xff0c;求编号1的结点到每个点的最短路径 输出从第一个点到第n个点的最短路径 思路 将图g[][]中所有的权值初始化为0x3f表示正无穷 将dist[]中所有的值初始化为0x3f表示从第一个点到所有点的距离…

linux的io的知识大全

C语言的io操作 写文件 #include<stdio.h> #include<string.h>#define FILE_NAME "log.txt" int main() {FILE * fp fopen(FILE_NAME, "w");if(fpNULL){printf("fopen error!\n");}const char* msg "hello zk\n";int c…

如何准确测量电源噪声

目录 电源噪声的特点 影响噪声测试的因素 总结 电源噪声的特点 以往电源噪声的幅度规范一般在几十mV&#xff0c;但是随着芯片电源电压的降低&#xff0c;很多芯片的电源噪声的规范已经低至mV的量级&#xff0c;某些对电源噪声敏感的芯片要求甚至到了百uV的量级。 电源上的…

基于Android studio 实现外卖(点)订餐系统-编程乐学最新原创

&#x1f345;文章末尾有获取完整项目源码方式&#x1f345; 目录 一、实现介绍 视频演示 1.1 启动页 1.2登录页 1.3注册页 1.4商家主页 1.5商家发布商品页面 1.6商家我的页面 1.7商家个人信息修改页 1.8商家商品信息修改页 1.9用户首页 1.10用户我的订单页面 1.1…

C++学习进阶:二进制与位运算

目录 1.进制与原反补码 2.位运算 2.1.按位与 2.2.按位或 2.3.异或 2.4.取反 2.5.移位 3.部分面试题 3.1.不创建新的变量&#xff0c;实现两个变量的交换 3.2.求一个整数存储在内存中二进制中1的个数 这一部分本来是C语言的内容&#xff0c;当学习位图时&#xff0c…

期货分账户软件|程序化软件|风控软件|资产管理软件开发用到哪些技术?

期货/股票资管分仓软件分账户系统APP的开发涉及多个技术领域&#xff0c;以确保软件的功能性、安全性和易用性。以下是一些在开发过程中可能需要用到的关键技术&#xff1a; 前端开发技术&#xff1a;前端部分主要负责用户界面的设计和实现。通常使用HTML、CSS和JavaScript等技…

YARN-Client 与 YARN-Cluster 区别

YARN-Client 与 YARN-Cluster 区别 理解YARN-Client和YARN-Cluster深层次的区别之前先清楚一个概念&#xff1a;Application Master。在YARN中&#xff0c;每个Application实例都有一个ApplicationMaster进程&#xff0c;它是Application启动的第一个容器。它负责和ResourceMa…

【HTML】制作一个简单的实时字体时钟

目录 前言 HTML部分 CSS部分 JS部分 效果图 总结 前言 无需多言&#xff0c;本文将详细介绍一段HTML代码&#xff0c;具体内容如下&#xff1a; 开始 首先新建文件夹&#xff0c;创建一个文本文档&#xff0c;两个文件夹&#xff0c;其中HTML的文件名改为[index.html]&am…

Jmeter杂记:测试计划参数详解

测试计划各参数详解 1&#xff0c;用户自定义变量&#xff0c;是全局变量&#xff0c;供所有线程组使用&#xff0c;可用配置元件&#xff1a;用户自定义变量替代 2&#xff0c;连续的运行线程组&#xff0c;默认不勾选&#xff0c;则随机的运行多个线程组中的取样器&#xff…

在隐私计算应用中和数链具备哪些技术特点?

在加速“可信数字化”进程的背景下&#xff0c;我国区块链产业将在打造新型平台经济&#xff0c;开启共享经济新时代的同时&#xff0c;带动数字经济“脱虚向实”服务实体经济。 和数软件在加速数字化进程的同时&#xff0c;进一步服务实体经济&#xff0c;提高实体经济的活力…

VS2019 VS2022 LNK2019 无法解析的外部符号sprintf

解决方案&#xff1a; 项目属性》配置属性》链接接-》输入》附加依赖项&#xff0c;增加 legacy_stdio_definitions.lib legacy_stdio_definitions.lib 是一个库文件&#xff0c;通常与使用 Visual Studio 编译的 C/C 项目相关。它的作用是解决在使用新版本的 Visual Studio 编…

IIS中部署netcore程序出现500错误如何处理?

500错误在IIS部署中经常出现&#xff0c;但是解决非常耗时 官方也有给出一些指引&#xff0c;但是无法解决根本问题 建议检查netcore相关组件是否正确安装&#xff0c;如下&#xff1a; aspnetcore-runtime-3.1.32-win-x64 dotnet-hosting-3.1.32-win dotnet-runtime-3.1.…

两数相加(leetcode)

给你两个 非空 的链表&#xff0c;表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的&#xff0c;并且每个节点只能存储 一位 数字。 请你将两个数相加&#xff0c;并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外&#xff0c;这两个数都不会以 0 …

Intel VAAPI/QSV/oneVPL区别简介

一、常用视频加速接口汇总 libmfx就是Intel QSV. 二、VAAPI VAAPI (视频加速API&#xff0c;Video Acceleration API)包含一套开源的库(LibVA) 以及API规范, 用于硬件加速下的视频编解码以及处理&#xff0c;只有Linux上的驱动提供支持。由Intel主导&#xff0c;但是AMD&…

51单片机上面的IIC协议

1、什么是IIC协议 2、模拟IIC协议 51单片机上面是没有与IIC协议相关的寄存器的&#xff08;没有相关的硬件&#xff09;&#xff0c;不像串口可以配置对应的寄存器达到目的&#xff08;比如修改波特率9600 or 115200&#xff09;&#xff0c;要配置IIC只能够根据用户手册里面的…

未来课堂革命:OpenAI 发布 ChatGPT 使用指南,探索生成式 AI 如何重塑教育景观

随着新学期的来临&#xff0c;众多初登教师舞台的 00 后们&#xff0c;也完成了他们的第一个教师身份下的暑期生活。 对于开学的抵触情绪&#xff0c;不仅学生们普遍存在&#xff0c;许多 00 后的新晋教师们也同样感同身受。某种程度上&#xff0c;这些抗拒上班的年轻教师群体…

Java日期正则表达式(附Demo)

目录 前言1. 基本知识2. Demo 前言 对于正则匹配&#xff0c;在项目实战中运用比较广泛 原先写过一版Python相关的&#xff1a;ip和端口号的正则表达式 1. 基本知识 对于日期的正则相对比较简单 以下是一些常见的日期格式及其对应的正则表达式示例&#xff1a; 年-月-日&a…

亚马逊的核心壁垒:物流

物流为美国电商市场渗透及格局的核心影响因素&#xff0c;也是亚马逊的核心壁垒所在。 从行业规模来看&#xff0c;美国电商渗透率低于中国&#xff0c;主要由于 两国地理及人口密度差异导致美国物流履约难度更大&#xff0c;此外美国更发达的实 体零售业和更为严苛的电商政策…