一,理论基础
为什么要使用多线程?
无非就是两点,提高CPU的处理效率,然后就是提高响应速度。
因为现在大多数CPU都是多核处理器了,所以我们在使用多线程的时候,每一个CPU处理一个线程,这样就使CPU的处理效率大大提升,而一个CPU处理一个线程之后再处理
另外一个效率就慢了。
提高响应速度的话就是,比如我们在订外卖的时候,他需要经过多个过程,如果是一个一个完了才是另一个的话,就会使响应速读变慢,然后我们使用多线程的话,可以使响应速度得到提高。
多线程的不安全性?
对于多线程的不安全性就是,多个线程可见的数据是不一样的。
造成这种不安全性的根源是什么?
原因有三个反别是原子性,可见性,有序性
可见性的原因是本地内存,我们有两个线程,线程1与线程2,当我们线程1,对变量更改了以后,对于线程1的本地内存是更改了,但是原内存是没有更改的,所以在线程2中那个变量还是之前的数据
原子性就是要么都执行,要么都不执行。这可能是上下文切换导致的,比如说两个线程1,2。线程1刚执行到第一个阶段的时候,就切换到线程2,就导致了本来应该线程1全部执行完毕之后才执行线程2,导致了结果错误。
然后就是有序性,编码优化带来的有序性问题
并行和并发有什么区别?
并发就是在多个任务在同一个CPU上,分时间段执行,在逻辑上是一同执行的
并行就是多个任务分到不同的CPU上,同时执行。
什么是Java的内存模型?
Java的内存模型简称JMM,他可以让线程对共享变量的改变使另外的线程可见,它还可以表示线程与内存之间的抽象关系,线程之间的共享变量存储在主内存中,对其中一个线程更改共享变量时,对存储在本地内存中,但是本地内存时JMM的一个抽象概念,并不真实存在。
volatile关键字的作用?
对于可见性来说,Java提供的volatile关键字是保证可见性和禁止指令重排,怎么保证可见性,是因为在更改线程之间的共享数据时候之前是存入线程的本地内存,经过volatile关键字修饰后,存入到主内存中,这就使得所有线程都可见,保证了可见性。
还有作用就是与CAS结合保证了原子性。
volatile可以使一个非原子性操作变为原子性操作吗?
并不能,volatile是保证线程可见性和禁止指令级排序的作用,但是它可以保证32位的double类型和long类型的原子性
volatile变量和atomic原子类变量有什么区别?
volatile变量的写操作是发生在读操作之前的,所以对count++,这个操作就是不具有原子性的。
而对于atomic原子类变量的话,是可以让种操作具有原子性的,所以就在原有数据是加一
Final关键字的用法
修饰变量,赋值之后就无法改变
修饰方法之后就无法被重写
修饰类之后,无法被继承
修饰参数,无法改变参数所应用的对象
所以被Final关键字修饰后的都是编译期常量吗
并不是就比如说随机数吗,只有执行完之后才是常量