深入理解CAS和Atomic工具类

news2024/11/22 18:31:59

CAS

CAS(Compare And Swap,比较交换)指的是对于一个变量,比较它的内存的值与期望值是否相同,如果相同则将内存值修改为新的指定的值。即CAS包括两个步骤:1.比较内存值与期望值是否相同;2.相同则赋予新值,使用伪代码实现如下:

if(value == expectedValue) {
    value = newValue;
}

CAS可以看作是这两个步骤的原子操作,并且原子性是在硬件层面得到保障的。CAS可以看作是一种乐观锁的实现,使用CAS时不会加锁,而是假设没有冲突去完成,发生冲突就不断重试直到成功。

Java中的CAS

Java中的CAS是由Unsafe类提供的,它按照类型提供了三种CAS操作。

//Object
public final native boolean compareAndSwapObject(java.lang.Object o, long l, java.lang.Object o1, java.lang.Object o2);
//Int
public final native boolean compareAndSwapInt(java.lang.Object o, long l, int i, int i1);
//Long
public final native boolean compareAndSwapLong(java.lang.Object o, long l, long l1, long l2);

这三个方法都是native的,即具体是在JVM中实现的,不同虚拟机的实现可能会不同,这很容易理解,因为CAS在操作系统中是一个原子指令,不同操作系统的指令也不同。

需要注意的是,Unsafe类的构造函数是私有的,即不能通过new关键字的方式创建对象;它提供了一个getUnsafe()方法,返回值是Unsafe()对象,但该方法是提供给Java API使用的,我们使用此方法获取Unsafe对象,会报SecurityException错误。

但我们可以通过反射来获取Unsafe类的对象,如下:

public static Unsafe getUnsafe() {
    try {
		Field field = Unsafe.class.getDeclaredField("theUnsafe");
        field.setAccessible(true);
        return (Unsafe) field.get(null);
    } catch (Exception e) {
        e.printStackTrace();
    }
    return null;
}

下面以compareAndSwapInt()方法举例介绍,该方法的参数分别表示要替换的对象实例、要替换的字段在对象中的内存偏移量、字段的期望值、字段的新值。首先使用CAS设置User类的age字段的值为3,随后判断该值如果为3则再将age设置为5,最后判断该值如果为3则设置为8。User类定义如下:

public class User {
    private int num;
	private int age;
	private String name;
}

测试代码如下:

public static void main(String[] args) {
    User user = new User();
    Unsafe unsafe = UnsafeFactory.getUnsafe();
    boolean successful;

    // 4个参数分别是:对象实例、字段的内存偏移量、字段期望值、字段更新值
    successful = unsafe.compareAndSwapInt(user, 16, 0, 3);
    System.out.println(successful + "\t" + user.getAge());

    successful = unsafe.compareAndSwapInt(user, 16, 3, 5);
    System.out.println(successful + "\t" + user.getAge());

    successful = unsafe.compareAndSwapInt(user, 16, 3, 8);
    System.out.println(successful + "\t" + user.getAge());
}

其中字段内存偏移量是16,原因是User对象头中的Markword占8个字节,Klass Pointer占4个字节(指针压缩),User对象的num是int类型的占4个字节,我们要修改的age就是从User对象的第16个字节开始的。

程序执行结果如下:

CAS虽然高效地解决了原子操作,但也存在以下缺陷:

  • 自旋CAS长时间不成功,会给CPU带来很大开销;
  • 只能保证一个共享变量的原子操作;
  • ABA问题。

ABA问题

ABA问题指的是当有多个线程对一个原子类进行操作的时候,某个线程在短时间内将原子类的值A修改为B,随后又将其修改为A,这个过程对于其他线程是感知不到的,其他线程在用A值与修改后的A值比较还是相等的,最终可以修改成功。

一个比较容易想到的解决方案是为这个原子类加一个类似版本号的东西,线程每次对该原子类修改后都要相应的修改其版本号,这样某个线程在使用CAS修改该原子类时判断版本号是否与线程存储的版本号相同即可判断是否出现了ABA问题。

Java为以上方式提供了一个原子类AtomicStampedReference类,其中除了我们实际要计算的变量外,还包括一个类似版本号的变量stamp,AtomicStampedReference类的部分实现如下。

public class AtomicStampedReference<V> {

    private static class Pair<T> {
        final T reference;
        final int stamp;
        private Pair(T reference, int stamp) {
            this.reference = reference;
            this.stamp = stamp;
        }
        static <T> Pair<T> of(T reference, int stamp) {
            return new Pair<T>(reference, stamp);
        }
    }
}

我们使用AtomicStampedReference来实验这样的一个ABA问题:线程1获取AtomicStampedReference变量value的值为1且版本为1随后阻塞1S,线程2将该变量值修改为2且版本加1,然后线程2再次将该变量值修改为1且版本加1,这时线程1尝试修改变量值为3。程序代码如下:

public class AtomicStampedReferenceTest {

    public static void main(String[] args) {
        // 定义AtomicStampedReference    Pair.reference值为1, Pair.stamp为1
        AtomicStampedReference atomicStampedReference = new AtomicStampedReference(1,1);

        new Thread(()->{
            int[] stampHolder = new int[1];
            int value = (int) atomicStampedReference.get(stampHolder);
            int stamp = stampHolder[0];
            log.debug("Thread1 read value: " + value + ", stamp: " + stamp);

            // 阻塞1s
            LockSupport.parkNanos(1000000000L);
            // Thread1通过CAS修改value值为3   stamp是版本,每次修改可以通过+1保证版本唯一性
            if (atomicStampedReference.compareAndSet(value, 3,stamp,stamp+1)) {
                log.debug("Thread1 update from " + value + " to 3");
            } else {
				atomicStampedReference.get(stampHolder);
                log.debug("Thread1 update fail,oldStamp:" + stamp + ",newStamp:" + stampHolder[0]);
            }
        },"Thread1").start();

        new Thread(()->{
            int[] stampHolder = new int[1];
            int value = (int)atomicStampedReference.get(stampHolder);
            int stamp = stampHolder[0];
            log.debug("Thread2 read value: " + value+ ", stamp: " + stamp);
            // Thread2通过CAS修改value值为2
            if (atomicStampedReference.compareAndSet(value, 2,stamp,stamp+1)) {
                log.debug("Thread2 update from " + value + " to 2");

                // do something

                value = (int) atomicStampedReference.get(stampHolder);
                stamp = stampHolder[0];
                log.debug("Thread2 read value: " + value+ ", stamp: " + stamp);
                // Thread2通过CAS修改value值为1
                if (atomicStampedReference.compareAndSet(value, 1,stamp,stamp+1)) {
                    log.debug("Thread2 update from " + value + " to 1");
                }
            }
        },"Thread2").start();
    }
}

控制台打印结果如下:

可以看到线程1最后是修改失败的,因为线程1虽然感知不到变量值的变化(一直是1),但版本号不一样了,线程1一开始获取到的版本号是1,经过线程2修改两次之后变成3了,因此线程1修改失败。

Java还提供了另一个类AtomicMarkableReference也可以解决ABA问题,与AtomicStampedReference类不同的是,这个类没有记录版本号(修改次数),只是记录了存储的变量是否被改变。

Atomic原子工具类

在并发编程中为了防止出现并发安全问题,最常用的方法是通过synchronized或Lock加锁来进行控制,但对于一些很简单的i++操作如果也使用这么重量级的锁,就会降低系统的性能。因此JUC(java.util.current)包为我们提供了一个atomic包,其中包含了各种原子工具类,这些原子工具类都是通过Unsafe类提供的CAS方法来完成相应的功能。

atomic包下的原子工具类根据操作数据的类型可以分为以下几类:

  • 基本类型:AtomicInteger、AtomicLong、AtomicBoolean;
  • 引用类型:AtomicReference、AtomicStampedReference、AtomicMarkableReference;
  • 数组类型:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray;
  • 对象属性原则修改器:AtomicIntegerFieldUpdater、AtomicLongFieldUpdater、AtomicReferenceFieldUpdater;
  • JDK1.8新增的原子类型累加器:DoubleAccumulator、DoubleAdder、LongAccumulator、LongAdder、Striped64。

下面分别以每个类型中的一个类举例说明这些原子工具类的简单使用。

基本类型

以AtomicInteger为例,其常用的几个方法如下:

//返回旧值并将当前变量值加1
public final int getAndIncrement() {
    return unsafe.getAndAddInt(this, valueOffset, 1);
}
//返回旧值并将当前变量的值修改为指定的值
public final int getAndSet(int newValue) {
    return unsafe.getAndSetInt(this, valueOffset, newValue);
}
//将当前变量的值加1并返回计算后的新值
public final int incrementAndGet() {
    return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
}
//CAS操作
public final boolean compareAndSet(int expect, int update) {
    return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
}
//将变量值与指定的值相加并返回计算后的新值
public final int addAndGet(int delta) {
    return unsafe.getAndAddInt(this, valueOffset, delta) + delta;
}

我们使用多个线程并发对一个普通int变量自增100000次,最终得到的结果总是小于100000,因为自增操作不是原子性的,这些线程之间计算的结果会有覆盖的情况,例如下面的程序使用10个线程分别对变量count执行自增操作10000次。

public class AtomicIntegerTest {
    
    private static int count;

    public static void main(String[] args) {

        for (int i = 0; i < 10; i++) {
            Thread thread = new Thread(() -> {
                for (int j = 0; j < 10000; j++) {
                    count++;

                }
            });
            thread.start();
        }

        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
		System.out.println(count);
    }

}

控制台打印结果:

使用AtomicInterger类就可以避免线程对变量值的相互覆盖的问题。

public class AtomicIntegerTest {

    static AtomicInteger sum = new AtomicInteger(0);

    public static void main(String[] args) {

        for (int i = 0; i < 10; i++) {
            Thread thread = new Thread(() -> {
                for (int j = 0; j < 10000; j++) {
                    // 原子自增  CAS
                    sum.incrementAndGet();

                }
            });
            thread.start();
        }

        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(sum.get());

    }

}

控制台打印结果如下:

数组类型

以AtomicIntegerArray为例,其常用的几个方法如下:

//为数组指定下标的变量加上指定的值,并返回计算后的新值
public final int addAndGet(int i, int delta) {
    return getAndAdd(i, delta) + delta;
}
//为指定下标的变量加1,并返回旧值
public final int getAndIncrement(int i) {
    return getAndAdd(i, 1);
}
//CAS设置指定下标的值
public final boolean compareAndSet(int i, int expect, int update) {
    return compareAndSetRaw(checkedByteOffset(i), expect, update);
}

测试程序:

public class AtomicIntegerArrayTest {

    static int[] value = new int[]{ 1, 2, 3, 4, 5 };
    static AtomicIntegerArray atomicIntegerArray = new AtomicIntegerArray(value);


    public static void main(String[] args) throws InterruptedException {

        //设置索引0的元素为100
        atomicIntegerArray.set(0, 100);
        System.out.println(atomicIntegerArray.get(0));
        //以原子更新的方式将数组中索引为1的元素与输入值相加
        atomicIntegerArray.getAndAdd(1,5);

        System.out.println(atomicIntegerArray);
    }
}

控制台打印结果:

引用类型

AtomicStampedReference和AtomicMarkableReference已经大概介绍过了,此处以AtomicReference为例,AtomicReference是对普通对象的封装,可以保证在修改对象引用时的线程安全。需要注意的是,它只会保证修改对象引用的线程安全。

测试程序:

public class AtomicReferenceTest {

    public static void main( String[] args ) {
        User user1 = new User("张三", 23);
        User user2 = new User("李四", 25);
        User user3 = new User("王五", 20);

        //初始化为 user1
        AtomicReference<User> atomicReference = new AtomicReference<>();
        atomicReference.set(user1);

        //把 user2 赋给 atomicReference
        atomicReference.compareAndSet(user1, user2);
        System.out.println(atomicReference.get());

        //把 user3 赋给 atomicReference
        atomicReference.compareAndSet(user1, user3);
        System.out.println(atomicReference.get());

        //修改user2对象的值,再将user3赋给atomicReference
		user2.setAge(100);
		user2.setName("赵六");
		atomicReference.compareAndSet(user2,user3);
		System.out.println(atomicReference.get());

    }

}

控制台打印结果:

对象属性原子修改器

以AtomicIntegerFieldUpdater为例,它可以线程安全地更新对象中的整型变量,但它的使用有以下限制:

  • 对象的属性必须是volatile修饰的,因为CAS只保证原子性,不保证可见性,事实上AtomicInteger中存储值的属性也是volatile修饰的;
  • 字段的描述类型与调用者与操作对象字段的关系一致,即调用者能够直接操作对象字段,就可以通过反射进行原子操作。但对于父类的字段,子类是不能直接操作的;
  • 只能是实例变量,不能是类变量;
  • 只能是可修改变量,不能是final变量;
  • AtomicIntegerFieldUpdater和AtomicLongFieldUpdater只能修改int或long类型的字段,不能修改其包装类型,如果需要修改包装类型需要使用AtomicReferenceFieldUpdater。

测试程序:

public class AtomicIntegerFieldUpdaterTest {


    public static class Candidate {
        //字段必须是volatile类型
        volatile int score = 0;

        AtomicInteger score2 = new AtomicInteger();
    }

    public static final AtomicIntegerFieldUpdater<Candidate> scoreUpdater =
            AtomicIntegerFieldUpdater.newUpdater(Candidate.class, "score");

    public static AtomicInteger realScore = new AtomicInteger(0);

    public static void main(String[] args) throws InterruptedException {

        final Candidate candidate = new Candidate();

        Thread[] t = new Thread[10000];
        for (int i = 0; i < 10000; i++) {
            t[i] = new Thread(new Runnable() {
                @Override
                public void run() {
                    if (Math.random() > 0.4) {
                        candidate.score2.incrementAndGet();
                        scoreUpdater.incrementAndGet(candidate);
                        realScore.incrementAndGet();
                    }
                }
            });
            t[i].start();
        }
        for (int i = 0; i < 10000; i++) {
            t[i].join();
        }
        System.out.println("AtomicIntegerFieldUpdater Score=" + candidate.score);
        System.out.println("AtomicInteger Score=" + candidate.score2.get());
        System.out.println("realScore=" + realScore.get());

    }
}

控制台打印结果:

原子类型累加器

AtomicLong是利用了底层的CAS操作来提供并发性的,逻辑是采用自旋的方式不断尝试更新目标值,直到更新成功。在并发量较低的环境下,线程冲突的概率比较小,自旋的次数和时间可能不多。但在高并发场景下,就可能会有很多个线程同时在自旋,也就会出现大量失败并不断自旋的情况,这就让AtomicLong的自旋成为性能瓶颈。

Java为了解决以上问题,引入了LongAdder等累加器类。

设计思路

AtomicLong有个内部变量保存实际的long值,所有的操作都是针对这个变量进行的。在高并发场景下,这个变量可以看作是一个热点,N个线程竞争一个热点,性能自然就下降了。LongAdder的基本思路就是分散热点,将这个变量的值分散到一个数组中,不同线程会命中到数组的不同槽中,各个线程只对自己槽中的那个值进行CAS,这样就达到了分散热点的目的,线程之间的冲突概率变小。如果要获取真正的long值,只需要将所有槽的变量值累加返回即可。

内部结构

LongAdder内部包含一个base变量和一个Cell[]数组。当没有发生并发竞争时,直接使用base变量累加值,否则将各个线程的值累加到它所在的Cell[]槽中。

LongAdder的核心方法是add()方法,作用是为当前变量值原子加上一个指定的值,该方法的逻辑可以表示如下图:

LongAdder和AtomicLong

LongAdder和AtomicLong在低并发时差距并不明显,随着并发的增多,它们的效率差距就会越来越大。例如下面的程序,随着线程数和操作数的增加,LongAdder与AtomicLong的耗时差距越来越明显。

public class LongAdderTest {

    public static void main(String[] args) {
        testAtomicLongVSLongAdder(10, 10000);
        System.out.println("==================");
        testAtomicLongVSLongAdder(10, 200000);
        System.out.println("==================");
        testAtomicLongVSLongAdder(100, 200000);
    }

    static void testAtomicLongVSLongAdder(final int threadCount, final int times) {
        try {
            long start = System.currentTimeMillis();
            testLongAdder(threadCount, times);
            long end = System.currentTimeMillis() - start;
            System.out.println("条件>>>>>>线程数:" + threadCount + ", 单线程操作计数" + times);
            System.out.println("结果>>>>>>LongAdder方式增加计数" + (threadCount * times) + "次,共计耗时:" + end);

            long start2 = System.currentTimeMillis();
            testAtomicLong(threadCount, times);
            long end2 = System.currentTimeMillis() - start2;
            System.out.println("条件>>>>>>线程数:" + threadCount + ", 单线程操作计数" + times);
            System.out.println("结果>>>>>>AtomicLong方式增加计数" + (threadCount * times) + "次,共计耗时:" + end2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    static void testAtomicLong(final int threadCount, final int times) throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(threadCount);
        AtomicLong atomicLong = new AtomicLong();
        for (int i = 0; i < threadCount; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for (int j = 0; j < times; j++) {
                        atomicLong.incrementAndGet();
                    }
                    countDownLatch.countDown();
                }
            }, "my-thread" + i).start();
        }
        countDownLatch.await();
    }

    static void testLongAdder(final int threadCount, final int times) throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(threadCount);
        LongAdder longAdder = new LongAdder();
        for (int i = 0; i < threadCount; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for (int j = 0; j < times; j++) {
                        longAdder.add(1);
                    }
                    countDownLatch.countDown();
                }
            }, "my-thread" + i).start();
        }

        countDownLatch.await();
    }
}

控制台打印结果:

Java休眠线程

在Java中,让一个线程休眠有三种方法,分别为:Thread.sleep()、Object.wait()和LockSupport.park()方法。

Thread.sleep()

sleep()方法必须指定线程休眠时间,调用此方法后线程在Java层面的状态是TIMED_WAITING,且需要捕获InterruptedException异常。sleep()方法不会释放线程持有的锁。

Object.wait()

wait()方法是和notify()或notifyAll()方法配合使用的,它们都是Object提供的方法,可用来实现等待唤醒机制,但拥有以下缺点:

  • 在使用这几个方法前必须要获取锁对象,即这几个方法只能在synchronized方法或synchronized块中运行;
  • 当对象的等待队列有多个线程时,notify()方法只能随机选择其中一个线程唤醒,不能指定唤醒某个线程。

LockSupport.park()

LockSupport是JDK中用来实现线程阻塞和唤醒的工具。使用它可以在任何场合使线程阻塞,且可以指定任何线程进行唤醒,不用担心阻塞和唤醒操作的顺序,但连续多次唤醒和一次唤醒效果是一样的。JUC包下的锁和其他同步工具的底层实现大量地使用了LockSupport进行线程的阻塞和唤醒。需要注意的是,park()方法不会释放锁。

LockSupport的park()和unpark()方法作用分别是使线程阻塞和唤醒线程,简单使用方法如下面的程序:

public class LockSupportTest {

    public static void main(String[] args) throws InterruptedException {
        Thread parkThread = new Thread(new ParkThread());
        parkThread.start();

		Thread.sleep(1000);
        System.out.println("唤醒parkThread");
        //为指定线程parkThread提供“许可”
        LockSupport.unpark(parkThread);
    }

    static class ParkThread implements Runnable{

        @Override
        public void run() {
            System.out.println("ParkThread开始执行");
            // 等待“许可”
            LockSupport.park();
            System.out.println("ParkThread执行完成");
        }
    }
}

控制台打印结果:

实现原理

LockSupport的park()和unpark()方法也是通过Unsafe类来实现的,这两个方法在Unsafe中都是native方法,即都是在JVM中实现的。这两个方法原理是:使某个线程阻塞需要消耗线程的一个凭证,这个凭证至多有一个。当调用park()方法时,线程如果有凭证则直接消耗掉这个凭证并正常退出;如果线程没有凭证,则阻塞该线程直到凭证可用。当调用unpark()方法时,它会为线程增加一个凭证,但至多有一个,即调用多次unpark()方法与调用一次效果是一样的。

这个凭证在JVM的C++实现中就对应着Parker实例中的_counter变量,每个线程都有Parker实例。

class Parker : public os::PlatformParker {
 private:
  volatile int _counter ;
  ...
 public:
  void park(bool isAbsolute, jlong time);
  void unpark();
  ...
 }
 class PlatformParker : public CHeapObj<mtInternal> {
  protected:
  pthread_mutex_t _mutex [1] ;
  pthread_cond_t _cond [1] ;
  ...
}

LockSupport就是通过控制_counter变量来对控制线程的阻塞和唤醒,具体如下:

  • 当调用park()方法时,将_counter置为0,同时判断修改前的值如果大于0则说明前面调用过unpark()方法,直接退出;否则使线程阻塞。
  • 当调用unpark()方法时,将_counter置为1,同时判断修改前的值如果小于1则说明前面调用过unpark()方法,进行线程唤醒;否则直接退出。

需要注意的是,如果先调用两次unpark()方法再调用两次park()方法,线程仍会阻塞,因此两次unpark()方法与一次unpark()方法效果一致,只是将_counter的值设为1,但park()方法调用一次都需要消耗掉一个凭证。下面的例子可以证明。

测试程序:

public class LockSupportTest {

    public static void main(String[] args) throws InterruptedException {
        Thread parkThread = new Thread(new ParkThread());
		//为指定线程parkThread提供“许可”
		System.out.println("唤醒parkThread");
		LockSupport.unpark(parkThread);
		LockSupport.unpark(parkThread);
        parkThread.start();

    }

    static class ParkThread implements Runnable{

        @Override
        public void run() {
            System.out.println("ParkThread开始执行");
            // 等待“许可”
            LockSupport.park();
            LockSupport.park();
            System.out.println("ParkThread执行完成");
        }
    }
}

控制台执行结果:

但先调用两次park()方法再间隔调用两次unpark()方法,线程就会被唤醒。此处我们可以理解为,线程等待两个凭证,当调用一次unpark()方法提供一个凭证时就会被消耗掉,如果第二次unpark()方法的执行在第一个unpark()方法提供的凭证消耗掉之前,第二次unpark()方法不起作用;如果第二次unpark()方法的执行在第一个unpark()方法提供的凭证被消耗掉之后,第二次unpark()方法也会提供一个凭证。

我们来做一次测试,首先调用两次park()方法,随后连续调用两次unpark()方法,测试程序代码如下:

public class LockSupportTest {

    public static void main(String[] args) throws InterruptedException {
        Thread parkThread = new Thread(new ParkThread());
        parkThread.start();

        Thread.sleep(1000);
		//为指定线程parkThread提供“许可”
		System.out.println("唤醒parkThread");
		LockSupport.unpark(parkThread);
		LockSupport.unpark(parkThread);

    }

    static class ParkThread implements Runnable{

        @Override
        public void run() {
            System.out.println("ParkThread开始执行");
            // 等待“许可”
            LockSupport.park();
            LockSupport.park();
            System.out.println("ParkThread执行完成");
        }
    }
}

控制台打印结果如下:

可以看到线程是被阻塞了的。下面修改下代码,让两次unpark()方法调用有一定的时间间隔,这个时间间隔是用来让第一次unpark()方法提供的凭证消耗掉,这样使第二次unpark()方法提供的凭证起作用。测试程序代码如下:

public class LockSupportTest {

    public static void main(String[] args) throws InterruptedException {
        Thread parkThread = new Thread(new ParkThread());
        parkThread.start();

        Thread.sleep(1000);
		//为指定线程parkThread提供“许可”
		System.out.println("唤醒parkThread");
		LockSupport.unpark(parkThread);
		Thread.sleep(1);
		LockSupport.unpark(parkThread);

    }

    static class ParkThread implements Runnable{

        @Override
        public void run() {
            System.out.println("ParkThread开始执行");
            // 等待“许可”
            LockSupport.park();
            LockSupport.park();
            System.out.println("ParkThread执行完成");
        }
    }
}

控制台打印结果如下:

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

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

相关文章

【校招VIP】前端基础之post和get

考点介绍&#xff1a; get和post 是网络基础&#xff0c;也是每个前端同学绕不过去的小问题&#xff0c;但是在校招面试中很多同学在基础回答中不到位&#xff0c;或者倒在引申问题里&#xff0c;就丢分了。 『前端基础之post和get』相关题目及解析内容可点击文章末尾链接查看…

7个改变玩法规则的ChatGPT应用场景

ChatGPT因各种原因受到了广泛关注&#xff1a;ChatGPT可以充当各种改善生活改进工作的小助手&#xff0c;如内容写手、客户支持、语言翻译、编码专家等等。只需在你的聊天内容中添加适当的提示&#xff0c;人工智能将为你提供各项支持。[1] 1.ChatGPT作为内容写手 通过AI的帮助…

《Linux从练气到飞升》No.16 Linux 进程地址空间

&#x1f57a;作者&#xff1a; 主页 我的专栏C语言从0到1探秘C数据结构从0到1探秘Linux菜鸟刷题集 &#x1f618;欢迎关注&#xff1a;&#x1f44d;点赞&#x1f64c;收藏✍️留言 &#x1f3c7;码字不易&#xff0c;你的&#x1f44d;点赞&#x1f64c;收藏❤️关注对我真的…

掌握AI助手的魔法工具:解密Prompt(提示)在AIGC时代的应用「上篇」

在当今的AIGC时代&#xff0c;我们面临着越来越多的人工智能技术和应用。其中一个引人注目的工具就是Prompt&#xff08;提示&#xff09;。它就像是一种魔法&#xff0c;可以让我们与AI助手进行更加互动和有针对性的对话。那么&#xff0c;让我们一起来了解一下Prompt&#xf…

QA

1. 这是什么意思&#xff1f; label_viz[:,:,::-1] 这段代码看起来像是Python中处理图像的代码片段。让我来为您解释一下&#xff1a; 1. label_viz&#xff1a;这可能是一个二维数组&#xff08;通常是NumPy数组&#xff09;&#xff0c;代表图像上的标签或类别信息的可视化…

线程面试题-1

看的博客里面总结的线程的八股文 1、线程安全的集合有哪些&#xff1f;线程不安全的呢&#xff1f; 线程安全的&#xff1a; Hashtable&#xff1a;比HashMap多了个线程安全。 ConcurrentHashMap:是一种高效但是线程安全的集合。 Vector&#xff1a;比Arraylist多了个同步化…

Ubuntu本地快速搭建web小游戏网站,并使用内网穿透将其发布到公网上

文章目录 前言1. 本地环境服务搭建2. 局域网测试访问3. 内网穿透3.1 ubuntu本地安装cpolar内网穿透3.2 创建隧道3.3 测试公网访问 4. 配置固定二级子域名4.1 保留一个二级子域名4.2 配置二级子域名4.3 测试访问公网固定二级子域名 前言 网&#xff1a;我们通常说的是互联网&am…

SpringBoot中乐观锁的实现 (精简demo)

使用场景: 当要更新一条数据时&#xff0c;希望这条数据没有被别人更新&#xff0c;也就是说实现线程安全的数据更新 1. 数据库新增version字段, int类型, 默认值为0 2. 引入依赖 <!--mybatis拦截器--> <dependency><groupId>com.baomidou</groupId>&…

Nginx使用keepalived配置VIP

VIP常用于负载均衡的高可用&#xff0c;使用VIP可以给多个主机绑定一个IP&#xff0c;这样&#xff0c;当某个负载应用挂了之后&#xff0c;可以自动切到另一个负载。 我这里是在k8s环境中做的测试&#xff0c;集群中有6个节点&#xff0c;我给140和141两个节点配置VIP。 1. 安…

【leetcode 力扣刷题】移除链表元素 多种解法

移除链表元素的多种解法 203. 移除链表元素解法①&#xff1a;头节点单独判断解法②&#xff1a;虚拟头节点解法③&#xff1a;递归 203. 移除链表元素 题目链接&#xff1a;203.移除链表元素 题目内容&#xff1a; 理解题意&#xff1a;就是单纯的删除链表中所有值等于给定的…

Java【HTTP】什么是 Cookie 和 Session? 如何理解这两种机制的区别和作用?

文章目录 前言一、Cookie1, 什么是 Cookie2, Cookie 从哪里来3, Cookie 到哪里去4, Cookie 有什么用 二、Session1, 什么是 Session2, 理解 Session 三、Cookie 和 Session 的区别总结 前言 各位读者好, 我是小陈, 这是我的个人主页, 希望我的专栏能够帮助到你: &#x1f4d5; …

并查集 size 的优化(并查集 size 的优化)

目录 并查集 size 的优化 Java 实例代码 UnionFind3.java 文件代码&#xff1a; 并查集 size 的优化 按照上一小节的思路&#xff0c;我们把如下图所示的并查集&#xff0c;进行 union(4,9) 操作。 合并操作后的结构为&#xff1a; 可以发现&#xff0c;这个结构的树的层相对…

juc概述和Lock接口

目录 一、什么是JUC 1、JUC概述 2、进程与线程 3、线程的状态 4、wait/sleep 的区别 5、并发与并行 6、管程 7、用户线程和守护线程 二、Lock接口 1、Synchronized 使用synchronized实现售票案例 使用synchronized实现增减变量操作 2、什么是 Lock 买票例子使用lo…

如何选择 DCDC 降压型开关电源的电感

选择合适的电感是开关电源电路设计的关键之一。本文将帮助您理解电感值和电路性能之间的关系。 降压转换器&#xff08;buck converter&#xff09;&#xff0c;也称为降压转换器(step-down converter)&#xff0c;是一种开关模式稳压器&#xff08;voltage regulator&#xf…

Mac常见恶意软件再现,办公应用程序潜藏风险如何防范?

Mac电脑正受到臭名昭著的XLoader恶意软件的新变种的攻击&#xff0c;该恶意软件已被重写为在最好的MacBook上本地运行。 虽然XLoader至少从2015年开始出现&#xff0c;但在2021年发现macOS变体之前&#xff0c;它主要用于针对Windows PC。然而&#xff0c;该版本是作为Java程序…

win10系统rust串口通信实现

一、用cargo创建新工程 命令&#xff1a;cargo new comport use std::env; use std::{thread, time}; use serialport::{DataBits, StopBits, Parity, FlowControl}; use std::io::{self, Read, Write}; use std::time::Duration;fn main() -> io::Result<()> {let m…

CSS中的display属性有哪些值?它们的作用?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ CSS display 属性的不同取值和作用1. block2. inline3. inline-block4. none5. flex6. grid7. table、table-row、table-cell8. list-item9. inline-table、table-caption、table-column 等 ⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#x…

Pika Labs - 用AI工具生成短视频大片

今天我要跟大家分享如何用AI工具1分钟内生成一个短视频大片&#xff0c;效果完全不输影视大V。 只需要用一句话就可以生成视频&#xff0c;或者用一张图就能生成视频&#xff0c;这就是最新推出的AI工具Pika Labs&#xff01;被网友誉为“全球最优秀的文本生成视频AI”。 目前…

SharedPreferences详解及其ANR解决方案

关于作者&#xff1a;CSDN内容合伙人、技术专家&#xff0c; 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 &#xff0c;擅长java后端、移动开发、商业变现、人工智能等&#xff0c;希望大家多多支持。 目录 一、导读二、概览三、使用四、原理五、存在的问题六、优…

Android动态添加和删除控件/布局

一、引言 最近在研究RecyclerView二级列表的使用方法&#xff0c;需要实现的效果如下。 然后查了一些博客&#xff0c;觉得实现方式太过复杂&#xff0c;而且这种方式也不是特别受推荐&#xff0c;所以请教了别人&#xff0c;得到了一种感觉还不错的实现方式。实现的思路为&…