线程安全问题出现的原因:
多个线程共用了一个数据(共享的语句有多条,一个线程使用cpu,没有使用完,cpu被抢走了,当再次抢到cpu的时候,接着执行后面的语句,造成了错误的发生)。在java中,如果使用多个Thread启动同一个Runable的线程对象,此时这些线程会共享Runable对象中的属性,就容易出现线程安全问题。
线程m1和m2共享对象m;其中m对象中的id属性被共享,在运行时m1,m2;争抢运行,出现其中一方在修改id但尚未赋值的情况下另一方已经进行修改并重新赋值,之后此方将之前未赋值的值赋给id,导致id的值错乱,出现线程安全问题。
线程安全问题的解决:
java中使用加锁的方式解决线程安全问题;目前加锁方式有两种:synchronized关键字和Lock锁。
synchronized
synchronized是Java提供的一个并发控制的关键字,作用于对象上;用法有两种:使用synchronized语句块直接加在所需语句上;直接加在方法上。
synchronized语句块:
synchronized(任意对象) {
需要加锁的代码
}
两次调用对象最终属性id两轮1000次自增,结果2000正确,未出现线程安全问题。
Lock锁
在jdk1.5后新增Lock锁,同样用于解决线程安全问题;
用法:
lock()方法:上锁
unlock()方法:释放锁
(可以放在try{}finally{}语句块中保证释放锁操作的执行)
synchronized和Lock的区别:
synchronized和Lock都用来为进程加锁,防止多个线程情况下出现线程安全问题。
区别:
1.synchronized是java的一个关键字,属于jvm层面;Lock是一个接口。
2.释放:synchronized加锁的语句时,线程调用完成对应语句就会释放锁;
Lock锁需要通过.unlock()方法释放锁。
3.锁的获取:两个线程时,线程A先抢到锁,线程B会等待,线程A如果阻塞,线程B会一直等待;Lock锁可以通过tryLock()方法判断锁的状态,可以避免死锁。
4.锁类型:synchronized可重入、非公平、不可中断;Lock锁:可重入、可公平、可中断。