最近几年用Java写了一些多线程的程序,有一种需求,是建立一个公用的static类型Llit,类里面写了一些对于List的get、set、update之类的方法,而有一个线程从数据库中读取数据并维护这个List,其它有几百个几千个线程读取和修改这个List。
写本文的目的,就是为这种需求提出最佳解决方案:
一、首先了解一下同步安全问题:
对于线程安全,我们先看两个例子:
class CheckoutLane
{
public static float GetTotal(Cart cart)
{
float total = 0;
for (int i = 0; i < cart.GroceryItems.Length; i++)
{
total += cart.GroceryItems[i].Price;
Thread.Sleep(100);
}
return total;
}
}
对于上面的代码,你使用多少的线程来控制,都是安全的。因为线程之间,不会共享什么资源,唯一相同的是使用了同一个逻辑。其实这是在破坏线程的前题方面下功夫,出现线程不安全的条件都没有了,那肯定就是线程安全的。
class CheckoutLane
{
static float total;
public static float GetTotal(Cart cart)
{
total = 0;
for (int i = 0; i < cart.GroceryItems.Length; i++)
{
total += cart.GroceryItems[i].Price;
Thread.Sleep(100);
}
return total;
}
}
对于上面的这个例子,不是线程安全的,因为共享了static float total;这个资源,而各个线程都随机都被调用,可以任意修改total这个数据。这个,就正如多个收银员共享柜台,任意执行收银操作一样。
二、最简单的解决办法,加synchronized:
在加synchronized之前,首先要明白:
1、synchronized会影响代码执行效率;
2、synchronized虽然加在一个方法上,但是影响的是整个类或类的实例;
3、两个synchronized方法不能相互调用,否则会形成死锁,很好理解,A调B,B调A,A和B相互等着对方结束,但是使用了synchronized,都不能结束,程序就死在那里了。
三、还有一个办法,但有点小问题,还不成熟,想好了再说吧,欢迎拍砖。
转载于:https://blog.51cto.com/xuchenguang/422939