线程安全

线程同步与异步

线程同步如同一群人上公交车,排着队的一个上完后接着下一个。

线程异步如同一群人上公交车,不排队,谁抢到谁上去,各上各的。

 

线程安全

多个线程并发读写一个临界资源时候会发生”线程并发安全问题”。常见的临界资源有:多线程共享实例变量、静态公共变量。

比如一个 ArrayList 类,在添加一个元素的时候,它可能会有两步来完成:1. 在 Items[Size] 的位置存放此元素;2. 增大 Size 的值。

如果是在多线程情况下,比如有两个线程,线程 A 先将元素存放在位置 0。但是此时 CPU 调度线程A暂停,线程 B 得到运行的机会。线程B也向此 ArrayList 添加元素,因为此时 Size 仍然等于 0 (注意哦,我们假设的是添加一个元素是要两个步骤哦,而线程A仅仅完成了步骤1),所以线程B也将元素存放在位置0。然后线程A和线程B都继续运行,都增加 Size 的值。

那好,我们来看看 ArrayList 的情况,元素实际上只有一个,存放在位置 0,而 Size 却等于 2。这就是“线程不安全”了。

 

线程安全解决办法

把异步的操作变为同步操作。解决线程安全问题是让多个线程同步的读写临界资源。相对应的效率也降低了。通过牺牲性能来解决安全问题。

加锁synchronized关键字

1)synchronized修饰方法

public synchronized void synMethod() {
     //方法体
}

这时,一次只能有一个线程进入该方法,其他线程要想在此时调用该方法,只能排队等候,当前线程(就是在synchronized方法内部的线程)执行完该方法后,别的线程才能进入。

2)synchronized块

当一个方法被修饰后,该方法变为同步方法,虽然保证了代码的执行安全,到那时效率低下。我们实际上只需要将方法需要同步的代码片段加锁,这样可以缩小同步范围,从而提高代码的运行效率。     

synchronized(同步监视器){
    //需要同步的代码片段
}

同步监视器就是一个对象,任何对象都可以。但要保证一点,多个线程看到的。应该是”同一个对象’。一般情况下使用this即可。

public void run() {
    synchronized (this) {
    System.out.println(Thread.currentThread().getName());
  }
}

 

线程安全类与线程不安全类

StringBuffer是同步的 synchronized append(),StringBuilder不是同步的

Vector和Hashtable、是同步的,ArrayList和HashMap不是同步的

Collections.synchronizedList();//将list集合转变成线程安全的

Collections.synchronizedMap();

List<String> list = new ArrayList<String>();

list = Collections.synchronizedList(list);

posted on 2017-03-20 14:35  芹derella  阅读(178)  评论(0)    收藏  举报