目录
- JVM里new对象时,堆会发生抢占吗?如何设计JVM的堆的线程安全的?
- java异常和错误的区别
- Thread使用start和run方法启动线程有什么区别
- Volatile的内存语义
- JVM内存分配担保机制
- Java序列化SerialVersion
- static、static final、final的区别
- Java为什么需要 JVM
- Java类和对象的区别
- 接口和类的区别
- UUID
- java可变参数列表的实现
- 局部性原理
- Final修饰的int、String、Map可以改变吗
- cpu的缓存一致性协议
- java程序的执行过程
- java应用cpu占用过高问题
- JDK和JRE区别
- 构造函数是否可以重载
JVM里new对象时,堆会发生抢占吗?如何设计JVM的堆的线程安全的?
会,假设JVM虚拟机上,每一次new 对象时,指针就会向右移动一个对象size的距离,一个线程正在给A对象分配内存,指针还没有来的及修改,另一个为B对象分配内存的线程,引用这之前的指针指向,这就发生了抢占,也被称为指针碰撞。
TLAB的实现是给每个线程分配私有的指针,存对象的内存空间还是给所有线程访问,其它线程无法在这个区域分配,保证堆的线程安全。Thread Local Allocation Buffer,线程本地分配缓存
JVM在内存新生代Eden Space中开辟了一小块线程私有的区域TLAB(Thread-local allocation buffer)。在Java程序中很多对象都是小对象且用过即丢,它们不存在线程共享也适合被快速GC,所以对于小对象通常JVM会优先分配在TLAB上,并且TLAB上的分配由于是线程私有,所以没有锁开销。也就是说,Java中每个线程都会有自己的缓冲区称作TLAB,在对象分配的时候不用锁住整个堆,而只需要在自己的缓冲区分配即可。
java异常和错误的区别
在java中,异常和错误同属于一个类:Throwable。Exception(异常)是应用程序中出现的可预测,可恢复的问题。Exception分为两类:非运行时异常和运行时异常。运行时异常都是 RuntimeException 类及其子类异常,如 NullPointerException、IndexOutOfBoundsException 等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般由程序逻辑错误引起。非运行时异常是指 RuntimeException 以外的异常,程序就不能编译通过。IOException和ClassNotFoundException 等以及用户自定义的 Exception 异常。Error(错误)大多表示运行应用程序时比较严重的错误。大多错误与编程者编写的程序无关,可能是代码运行时jvm出现的问题。例如OutOfMemoryError。
Thread使用start和run方法启动线程有什么区别
调用start方法启动,使用start方法才真正实现了多线程运行,因为这个时候不用等待run方法执行完成就可以继续执行下面的代码,真正实现多线程。因为thread线程有5种状态,创建-就绪-运行-阻塞-死亡这五种,调用start方法就是就绪这一步,因为这个时候线程并没有立即的执行,而是得等待,等到cpu有空闲的时候,才会执行线程里面的run方法,等run方法执行完了,线程就结束了。
如果直接使用thread执行run方法,因为run方法是thread里面的一个普通的方法,所以直接调用run方法,这个时候它是会运行在主线程中的,因为这个时候程序中只有主线程一个线程,所以如果有两个线程,都是直接调用的run方法,那么它们的执行顺序一定是顺序执行,所以这样并没有做到多线程的这种目的。
public synchronized void start() {
//这里private volatile int threadStatus = 0;初始化的时候就是0
//如果这里不为0的话就抛异常
if (threadStatus != 0)
throw new IllegalThreadStateException();
//把当前线程加入到线程组中
//private ThreadGroup group;就是这么个东西
group.add(this);
//初始化标记位未启动
boolean started = false;
try {
start0();
//标识为启动状态
started = true;
} finally {
try {
//如果没开启,标识为启动失败
if (!started) {
group.threadStartFailed(this