非线程安全和线程安全
非线程安全:其实会在多个线程对同一个对象中的实例变量进行并发访问时产生,结果就是脏读。
线程安全:获得的实例变量的值经过同步处理,不会出现脏读的现象。
一个方法中的变量不存在非线程安全问题,永远都是线程安全的。这是因为方法内部的变量是私有特性造成的。
代码如下解读
public class HasSelfPrivateNum { public void addI(String username) { try { int num = 0;//注意该变量在方法内部 if (username.equals("a")) { num = 100; System.out.println("a set over!"); Thread.sleep(2000); } else { num = 200; System.out.println("b set over!"); } System.out.println(username + " num=" + num); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
public class ThreadA extends Thread { private HasSelfPrivateNum numRef; public ThreadA(HasSelfPrivateNum numRef) { super(); this.numRef = numRef; } @Override public void run() { super.run(); numRef.addI("a"); } }
public class ThreadB extends Thread { private HasSelfPrivateNum numRef; public ThreadB(HasSelfPrivateNum numRef) { super(); this.numRef = numRef; } @Override public void run() { super.run(); numRef.addI("b"); } }
public class Run { public static void main(String[] args) { HasSelfPrivateNum numRef = new HasSelfPrivateNum(); ThreadA athread = new ThreadA(numRef); athread.start(); ThreadB bthread = new ThreadB(numRef); bthread.start(); } }
把上面的代码改成如下:不加同步出现脏读,加上没有问题
结论:两个线程访问同一个对象中的同步方法时一定是线程安全的
public class HasSelfPrivateNum { //该变量放在方法的外面 不安全 private int num = 0; //加上synchronized后多个线程并发访问,只能排队访问资源,不会出现线程安全问题 public synchronized void addI(String username) { try { if (username.equals("a")) { num = 100; System.out.println("a set over!"); Thread.sleep(2000); } else { num = 200; System.out.println("b set over!"); } System.out.println(username + " num=" + num); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
再对上面的main()代码改造。
public class Run { public static void main(String[] args) { //两个线程访问两个不同的实例的相同的同步方法,结果异步执行 HasSelfPrivateNum numRef1 = new HasSelfPrivateNum(); HasSelfPrivateNum numRef2 = new HasSelfPrivateNum(); ThreadA athread = new ThreadA(numRef1); athread.start(); ThreadB bthread = new ThreadB(numRef2); bthread.start(); } }
总结:关键字synchronized获得的锁都是对象锁,不是说的吧一段代码或者方法当做锁,上面的案例就是那个线程先执行带synchronized关键字的方法,那个线程就持有该方法所属对象的锁Lock,其他线程只能等待,前提是多个线程访问同一个对象。这里不做synchronized锁的是对象代码阐述。