volatile 关键字的作用有那些?
- volatile 可以用来修饰变量,不能修饰方法或代码块。用于保证变量在内存中的可见性和防止指令重排。它只能保证可见性,不能保证复合操作的原子性。
- 保证变量的内存可见性就是当一个线程修改了某个变量时,其他线程能够立即看到这个变量的修改。
volatile 会发出 lock 指令对当前 cpu 缓存进行上锁,这样就会让当前 cpu 独占缓存,让其他 cpu 的此缓存段失效,因为加了 lock,接下来操作的指令会独占数据写回主存和缓存,其他 cpu 因为缓存失效就会去主存读数据重新写入缓存,就保证了不同 cpu 之间的缓存一致性
- 防止指令重排就是 volatile 变量的写操作在 JVM 执行时不会发生指令重排。确保写操作在读操作之前。
volatile 变量禁止指令重排序。针对 volatile 修饰的变量,在读写操作指令前后会插入内存屏障,指令重排序时不能把后面的指令重排序到内存屏障.
volatile 与synchronized 的对比?
- 机制和用途:
- synchronized 用于提供线程间的同步机制。当一个线程进入由 synchronized 修饰的方法或代码块时,会获取一个监视器锁。这保证了在同一时间只有一个线程能执行这段代码。从而保证了数据的一致性和线程的安全性。
- volatile 用于修饰变量。volatile 的主要作用是保证变量的可见性。即当一个线程修改了变量后,其他线程能够立即看到这个变量的修改。此外,它还可以防止重排序。但是 volatile 不保证复合操作的原子性。
总而言之。synchronized 关键字确保了多个线程访问资源的同步性。volatile 关键字确保变量在多个线程的可见性。
- 原子性:
- synchronized 修饰的代码是原子性的。代码在执行过程中不能被其他线程打断。
- volatile 只有单次读写操作是原子性的。不保证复合操作(如自增、自减)的原子性。



JDK8有哪些新特性?
- lambda 表达式:允许通过更简洁的方式编写匿名函数。
- 允许在接口中定义默认和静态方法。
- Stream API:提供了一种声明式的方法对集合进行操作。支持各种操作,如过滤、映射、排序、归约等
- 函数式接口:提供了一系列函数式接口,如 Consumer、Predicate、Function 等。
- 新的日期和时间 API:引入了 java.time 包。提供了更加清晰易用的日期和时间处理方式
- 方法引用:通过方法名称来引用方法,而不是执行方法。允许以更简洁的方式传递方法作为参数。如 System.out::println 等