总结线程安全问题的原因和解决方案

目录

一、什么是线程安全?

二、线程不安全带来的问题举例:

三、线程不安全的原因总结:

四、解决方案 

1、同步代码块

2、同步方法

3、静态同步方法

4、加锁Lock解决问题


一、什么是线程安全?

        在操作系统中,因为线程的调度是随机的(抢占式执行),正是因为这中随机性,才会让代码中产生很多bug 如果认为是因为这样的线程调度才导致代码产生了bug,则认为线程是不安全的, 如果这样的调度,并没有让代码产生bug,我们则认为线程是安全的

二、线程不安全带来的问题举例:

售票问题:

public class Test {
    private static int ticketCount=1;
    public static void main(String[] args) {
        //t1模拟售票窗口一
        Thread t1= new Thread(()-> {
            while(ticketCount<100) {
                System.out.println((Thread.currentThread().getName()+"正在卖第:"+ticketCount++ +"张票"));
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        //t2模拟售票窗口二
        Thread t2 = new Thread(()-> {
            while(ticketCount<100) {
                System.out.println((Thread.currentThread().getName()+"正在卖第:"+ticketCount++ +"张票"));
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        });
        t1.setName("窗口1");
        t2.setName("窗口2");

        t1.start();
        t2.start();
    }
}

来看输出结果所出现的问题:

         看标红的地方,出现了两个窗口售卖同一张票的情况,这就是多线程所导致的线程安全问题

三、线程不安全的原因总结:

1、抢占式执行

————多个线程的调度执行过程,可以视为是“全随机”的

2、多个线程修改同一个变量

3、修改操作不是原子的

原子性:

         定义: 即一个操作或者多个操作 要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行。

        原子性是拒绝多线程操作的,不论是多核还是单核,具有原子性的量,同一时刻只能有一个线程来对它进行操作。简而言之,在整个操作过程中不会被线程调度器中断的操作,都可认为是原子性。例如 a=1是原子性操作,但是a++和a +=1就不是原子性操作。Java中的原子性操作包括:

(1)基本类型的读取和赋值操作,且赋值必须是值赋给变量,变量之间的相互赋值不是原子性操作。

(2)所有引用reference的赋值操作

(3)java.concurrent.Atomic.* 包中所有类的一切操作

4、内存可见性问题

可见性:

        定义:指当多个线程访问同一个变量时,一个线程修改了这个变量的值,其他线程能够立即看得到修改的值。

        在多线程环境下,一个线程对共享变量的操作对其他线程是不可见的。Java提供了volatile来保证可见性,当一个变量被volatile修饰后,表示着线程本地内存无效,当一个线程修改共享变量后他会立即被更新到主内存中,其他线程读取共享变量时,会直接从主内存中读取。

5、指令重排序

有序性:
        定义:即程序执行的顺序按照代码的先后顺序执行。

        Java内存模型中的有序性可以总结为:如果在本线程内观察,所有操作都是有序的;如果在一个线程中观察另一个线程,所有操作都是无序的。前半句是指“线程内表现为串行语义”,后半句是指“指令重排序”现象和“工作内存主主内存同步延迟”现象。

        在Java内存模型中,为了效率是允许编译器和处理器对指令进行重排序,当然重排序不会影响单线程的运行结果,但是对多线程会有影响。Java提供volatile来保证一定的有序性。最著名的例子就是单例模式里面的DCL(双重检查锁)。另外,可以通过synchronized和Lock来保证有序性,synchronized和Lock保证每个时刻是有一个线程执行同步代码,相当于是让线程顺序执行同步代码,自然就保证了有序性。

四、解决方案 

        通常我们使用同步(关键字为synchronized)来解决这种由于多线程同时操作共享数据带来的线程安全问题。
        同步可以理解为:我们将多条操作共享数据的语句代码包成一个整体,让某个线程执行时其他线程不能执行。
        同步方案包括三种方式,它们对应的锁对象是不一样的。另外我们可以通过加锁来同步代码块,解决安全问题。
因此常用的解决方案有四种。
注意:
   同步可以解决问题的根本原因就在于锁对象上,因此要避免线程安全问题,多个线程必须使用同一个锁对象,否则,不能解决问题

1、同步代码块

格式:synchronized(对象) {
                需要被同步的代码;
            }

这里的锁对象可以是任意对象

利用该方法优化后如下:

public class Test1 {
    private static int ticketCount=1;
    private static Object object= new Object();
    public static void main(String[] args) {

        //t1模拟售票窗口一
        Thread t1= new Thread(()-> {
            while(ticketCount<100) {
                synchronized (object) {
                    System.out.println((Thread.currentThread().getName() + "正在卖第:" + ticketCount++ + "张票"));
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        });
         //t2模拟售票窗口二
        Thread t2 = new Thread(()-> {
            while(ticketCount<100) {
                synchronized (object) {
                    System.out.println((Thread.currentThread().getName() + "正在卖第:" + ticketCount++ + "张票"));
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        });
        t1.setName("窗口1");
        t2.setName("窗口2");

        t1.start();
        t2.start();
    }
}

2、同步方法

 格式:把同步(synchronized)加在方法上。

这时的锁对象是this

 利用该方法优化后如下:

public class Test2 {
    private static int ticketCount=1;
    public static void main(String[] args) {
        //t1模拟售票窗口一
        Thread t1= new Thread(()-> {
            while(ticketCount<100) {
                sellTicket();
            }
        });
        //t2模拟售票窗口二
        Thread t2 = new Thread(()-> {
            while(ticketCount<100) {
                sellTicket();
            }
        });
        t1.setName("窗口1");
        t2.setName("窗口2");

        t1.start();
        t2.start();
    }
    public static synchronized void sellTicket() {
        System.out.println((Thread.currentThread().getName()+"正在卖第:"+ticketCount++ +"张票"));
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }
}

3、静态同步方法

 格式:将同步加在静态方法上

此时的锁对象为当前类的字节码文件对象

public class Test3 {
    private static int ticketCount=1;
    public static void main(String[] args) {
        //t1模拟售票窗口一
        Thread t1= new Thread(()-> {
            while(ticketCount<100) {
                //同步代码块实现同步.这里设置的锁对象是该类的字节码文件对象
                synchronized (Test3.class) {
                    sellTicket3();
                }
            }

        });
        Thread t2 = new Thread(()-> {
            while(ticketCount<100) {
                synchronized (Test3.class) {
                    sellTicket3();
                }
            }
        });
        t1.setName("窗口1");
        t2.setName("窗口2");

        t1.start();
        t2.start();
    }
    public static synchronized void sellTicket3 () {
        System.out.println((Thread.currentThread().getName() + "正在卖第:" + ticketCount++ + "张票"));
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }
}

4、加锁Lock解决问题

要用lock和unlock包裹起来才能保证线程安全

public class Test4 {
    private static int ticketCount=1;
    private static Lock lock= new ReentrantLock();
    public static void main(String[] args) {
        //t1模拟售票窗口一
        Thread t1= new Thread(()-> {
            while(ticketCount<100) {
                lock.lock();
                System.out.println((Thread.currentThread().getName()+"正在卖第:"+ticketCount++ +"张票"));
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                lock.unlock();
            }
        });
        Thread t2 = new Thread(()-> {
            while(ticketCount<100) {
                lock.lock();
                System.out.println((Thread.currentThread().getName()+"正在卖第:"+ticketCount++ +"张票"));
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                lock.unlock();
            }
        });
        t1.setName("窗口1");
        t2.setName("窗口2");

        t1.start();
        t2.start();
    }
}

本期到这了先,下期见!!!(关于Java中线程安全的类,后续补充) 


版权声明:本文为LYJbao原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。