最近几年用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,都不能结束,程序就死在那里了。

三、还有一个办法,但有点小问题,还不成熟,想好了再说吧,欢迎拍砖。