线程安全可定义为:
当多个线程访问某个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的。也就是说线程能按照我们想要处理的结果进行数据处理,就是线程安全的。
如下代码,我们预想的是每次启动一个线程,count+1,即 出现 1,2,3,4,5的情况,通过对如下累加运算代码的分析,可做线程安全和非安全的对比。
import java.util.concurrent.atomic.AtomicInteger;
/**
* synchronized:可以在任意对象及方法上加锁,而加锁的这段代码称为"互斥区"或"临界区"
* @author lcx
*
*/
public class MyThread extends Thread{
private int count = 0;
//synchronized 加锁
public synchronized void run(){
count++;
System.out.println(this.currentThread().getName() + " count = "+ count);
}
public static void main(String[] args) {
/**
* 分析:当多个线程访问myThread的run方法时,以排队的方式进行处理(这里排对是按照CPU分配的先后顺序而定的),
* 一个线程想要执行synchronized修饰的方法里的代码:
* 1 尝试获得锁
* 2 如果拿到锁,执行synchronized代码体内容;拿不到锁,这个线程就会不断的尝试获得这把锁,直到拿到为止,
* 而且是多个线程同时去竞争这把锁。(也就是会有锁竞争的问题)
*/
MyThread myThread = new MyThread();
Thread m1 = new Thread(myThread,"m1");
Thread m2 = new Thread(myThread,"m2");
Thread m3 = new Thread(myThread,"m3");
Thread m4 = new Thread(myThread,"m4");
Thread m5 = new Thread(myThread,"m5");
m1.start();
m2.start();
m3.start();
m4.start();
m5.start();
}
}
如图所示,当run方法不加锁时,并未达到我们的预期,所以是线程非安全的,执行结果如下
如图所示,当run方法加锁时,虽然线程顺序是不定的,但是达到了我们的预期,所以是线程安全的,执行结果如下