请选择 进入手机版 | 继续访问电脑版

马上加入IBC程序猿 各种源码随意下,各种教程随便看! 注册 每日签到 加入编程讨论群

C#教程 ASP.NET教程 C#视频教程程序源码享受不尽 C#技术求助 ASP.NET技术求助

【源码下载】 社群合作 申请版主 程序开发 【远程协助】 每天乐一乐 每日签到 【承接外包项目】 面试-葵花宝典下载

官方一群:

官方二群:

多线程编程(2)—线程安全

[复制链接]
查看1902 | 回复0 | 2019-10-24 09:49:03 | 显示全部楼层 |阅读模式

1. 题目的引出 

 线程安全归根结底可以说是内存安全,在jvm内存模子中,有一块特殊的公共内存空间,称为堆内存,进程内的所有线程都可以访问并修改其中的数据,就会造成埋伏的题目。因为堆内存空间在没有保护机制的情况下,你放进去的数据,可能被别的线程窜改。如下代码:

  1. public class ThreadSafe implements Runnable {
  2. private static int count = 0;
  3. @Override
  4. public void run() {
  5. for (int i = 0; i < 1000; i++) {
  6. count++;
  7. }
  8. }
  9. public static void main(String[] args) throws InterruptedException {
  10. ExecutorService es = Executors.newFixedThreadPool(10);
  11. for (int i = 0; i < 20; i++) {
  12. es.execute(new ThreadSafe());
  13. }
  14. es.shutdown(); //不答应添加线程到线程池,异步关闭毗连池
  15. es.awaitTermination(10L, TimeUnit.SECONDS); //等候毗连池的线程任务完成
  16. System.out.println(count);
  17. }
  18. }
复制代码

 原来渴望的值是20000,可是终极输出的结果却一点在厘革,其值总是小于即是20000,显然这是由于线程不安全造成的,多个线程并发的去访问全局变量、静态变量、文件、装备、套接字等都可能出现这种题目。


2. 线程同步的步伐

 为了和谐和配合线程之间对共享资源的访问,通常有四种方式:

   1. 临界区:访问某一段临界资源的代码片断,与共享资源雷同,但有一点差异的是,某一时间只答应一个线程去访问(对应java中的关键字 synchronized包罗的代码)。

  2. 互斥量:互斥量是一个对象,只有都拥有互斥量的对象才可以访问共享资源。而且互斥量中只有一个,通常互斥量的实现是通过锁来实现的,而且加锁操纵和开释操纵只能由同一个线程来完成。此处与临界区的区别是一段代码,通常存在与一个文件中,而互斥量是一个对象,加锁操纵息争锁操纵可以在差异的文件去编写,只要是同一个线程就好。

  3. 信号量: 信号量可以答应指定命量的线程同一时间去访问共享资源,当线程数达到了阈值后,将制止其他线程的访问,最常见的比如生产者和消耗者题目。信号量和互斥量的区别则是信号量的发出和开释可以由差异线程来完成

  4. 变乱:通过发送关照的情势来实现线程同步,可以实现差异进程中的线程同步操纵


3.饥饿与死锁

饥饿:某些线程或进程由于长期得不到资源,而总是处于停当或者阻塞状态。比方:

  ①. 该进程或线程所拥有的CPU时间片被其他线程抢占而得不到执行(通常是优先级比它高的线程或进程),不停处于停当状态。

  ②. 由于选用不适当的调理算法,导致该进程或线程长期无法得到CPU时间片,处于停当状态。

  ③. 由于唤醒的时间把握不对,唤醒线程时,所需的资源处于被锁定状态,导致线程回到阻塞状态。

死锁:两个或多个线程在执行过程中,由于相互占有对方所需的资源而又各执己见从而造成这些线程都被阻塞,若无外力的作用下,他们都将无法执行下去。比方

  ①. 进程推进顺序不合适。互相占有彼此必要的资源,同时哀求对方占有的资源,形成循环依靠的关系。

  ②. 资源不足。

  ③. 进程运行推进顺序与速率差异,也可能产生死锁。

一些制止死锁的步伐:

  1. 不要在锁地区内涵加把锁,即不要在开释锁之前竞争其他锁。

  2. 减小锁粒度,即减小线程加锁的范围,真正必要的时间再去加锁。

  3. 顺序访问共享资源。

  4. 设置超机遇制,凌驾指定时间则程序返回错误。

  5. 竞争锁期间,答应程序被制止。


4.代码层面解决线程安全

 解决线程安全主要考虑三方面:

  1. 可见性:当多个线程并发的读写某个共享资源的时间,每个线程总是可以取到该共享资源的最新数据。

  2. 原子性:某线程对一个或者多个共享资源所做的连续串写操纵不会被制止,在此期间不会有其他线程同时对这些共享资源举行写操纵。

  3. 有序性:单个线程内的操纵必须是有序的。

通常原子性都可以得到保证,题目的病端就出在可见性和原子性。

4.1 可见性的题目

  如下实例程序,按通常的明白来说,当主线程等候一秒后,把flag的值修改为true后,另外一个线程应该可以感知到,然后跳过while循环,直接打印出后面的数据,可是最结果却不停卡在了while循环里。

  1. public class Thread4 implements Runnable{
  2. private static boolean flag = false;
  3. @Override
  4. public void run() {
  5. System.out.println("waiting for data....");
  6. while (!flag);
  7. System.out.println("cpying with data");
  8. }
  9. public static void main(String[] args) throws InterruptedException {
  10. Thread4 thread4 = new Thread4();
  11. Thread t = new Thread(thread4);
  12. t.start();
  13. Thread.sleep(1000);
  14. flag = true;
  15. }
  16. }
  17. /* output
  18. * waiting for data....
  19. */
复制代码

  主要的原因是java程序在jvm上运行的时间,该程序所占用的内存分为两类主内存和工作内存(逻辑上的内存,实际上是cpu的寄存器和高速缓存,因为,cpu在盘算的时间,并不总是从内存读取数据,它的数据读取顺序优先级是:寄存器-高速缓存-内存。CPU和内存之间通过总线举行)。也就是在主线程中启动另一个线程t会开辟出一个新的工作内存,与主线程的工作内存相互独立,且线程之间无法直接通讯,只能去主内存读取全局变量,而线程t中做while判断的时间并不会去读取主内存的flag值,致使线程t无法被感知到flag在其他线程被改变,可以做一个测试,如今把run函数改成如下情势:  

  1. public class Thread4 implements Runnable{
  2. private volatile static boolean flag = false;
  3. @Override
  4. private volatile static boolean flag = false;
  5. public void run() {
  6. System.out.println("waiting for data....");
  7. while (!flag);
  8. System.out.println("cpying with data");
  9. }
  10. //....
  11. }
  12. /* output
  13. * waiting for data....
  14. * false
  15. * ...
  16. * false
  17. * cpying with data
  18. */
复制代码

 为了感知其他线程中一些全局变量值的厘革,而且制止频仍去测试主内存中的数据厘革,保证线程之间的可见性,可以利用volatile关键字去修饰全局变量,如下:

  1. public void run() {
  2. System.out.println("waiting for data....");
  3. /* Notice
  4. 假如在while循环里加上System.out.println(flag);语句,则不会利用本工作内存的flag数据,
  5. 而是重新去主内存加载数据
  6. */
  7. while (!flag){
  8. System.out.println(flag); //测试,可以做到线程的可见性
  9. }
  10. System.out.println("cpying with data");
  11. }
  12. /* output
  13. * waiting for data....
  14. * false
  15. * ...
  16. * false
  17. * cpying with data
  18. */
复制代码

volatile关键字借助MESI一致性协议,会在工作内存(CPU的寄存器等)与主内存毗连的总线上创建一道总线嗅探机制,一旦发现其他线程修改了主内存中的某个全局变量(即图中橙灰色线条读取的数据以及写回的数据),就会让其他工作线程中从主内存拷贝出来的副本变量失效(即图中紫色的线条读取的数据),从而会使左边的线程重新去读取数据(即图中红色的线条读取的数据)。如下图:

094903yiu9c98407xq0wei.png

  固然解决了原子性题目,可是volatile关键字不支持原子性操纵,如下程序:

  1. public class Thread5 implements Runnable {
  2. private static volatile int count = 0;
  3. @Override
  4. public void run() {
  5. for (int i = 0; i < 10000; i++) {
  6. count++;
  7. }
  8. }
  9. public static void main(String[] args) throws InterruptedException {
  10. ExecutorService es = Executors.newFixedThreadPool(10);
  11. for (int i = 0; i < 20; i++) {
  12. es.execute(new Thread5());
  13. }
  14. es.shutdown(); //不答应添加线程,异步关闭毗连池
  15. es.awaitTermination(10L, TimeUnit.SECONDS); //等候毗连池的线程任务完成
  16. System.out.println(count);
  17. }
  18. }
  19. /* output
  20. * 175630
  21. */
复制代码

4.2 原子性题目

 针对原子性题目,我们可以利用熟悉的synchronized关键字,synchronized关键字最主要有以下3种应用方式:

  • 修饰实例方法,作用于当前实例加锁,进入同步代码前要得到当前实例的锁

  • 修饰静态方法,作用于当前类对象加锁,进入同步代码前要得到当前类对象的锁

  • 修饰代码块,指定加锁对象,对给定对象加锁,进入同步代码库前要得到给定对象的锁。

部门示例代码如下:

  1. public class Thread5 implements Runnable {
  2. private static int count = 0;
  3. public synchronized static void add() {
  4. count++;
  5. }
  6. @Override
  7. public void run() {
  8. for (int i = 0; i < 1000000; i++) {
  9. // add()
  10. synchronized (Thread5.class){
  11. count++;
  12. }
  13. }
  14. }
  15. public static void main(String[] args) throws InterruptedException {
  16. ExecutorService es = Executors.newFixedThreadPool(10);
  17. for (int i = 0; i < 20; i++) {
  18. es.execute(new Thread5());
  19. }
  20. es.shutdown(); //不答应添加线程,异步关闭毗连池
  21. es.awaitTermination(10L, TimeUnit.SECONDS); //等候毗连池的线程任务完成
  22. System.out.println(count);
  23. }
  24. }
  25. /* output
  26. * 20000000
  27. */
复制代码

然而synchronized是一种灰心锁,具有强烈的独占和排他特性,它频仍的加锁和开释锁操纵会使程序的效率低下。与灰心锁相对是一种乐观锁操纵CAS(CompareAndSwap),乐观锁就是每次去取数据的时间都乐观的以为数据不会被修改,因此这个过程不会上锁,但是在更新的时间会判断一下在此期间的数据有没有更新,假如没有更新则去修改,否则失败。可是上面这种 操纵会出现ABA(A-B-A,中途被改变,但最后又改回原值)的题目,

针对上面的题目,java中可以利用Atomic,它的包名为java.util.concurrent.atomic。这个包内里提供了一组原子变量的操纵类(通过值加版本号的方式去解决ABA题目),这些类可以保证在多线程环境下,当某个线程在执行atomic的方法时,不会被其他线程打断,不停比及该方法执行完成(详细的API文档可以检察参考文献第5点)。

  1. public class ThreadSafe implements Runnable {
  2. private static AtomicInteger count = new AtomicInteger(0);
  3. @Override
  4. public void run() {
  5. for (int i = 0; i < 10000; i++) {
  6. count.getAndAdd(1);
  7. }
  8. }
  9. public static void main(String[] args) throws InterruptedException {
  10. ExecutorService es = Executors.newFixedThreadPool(10);
  11. for (int i = 0; i < 20; i++) {
  12. es.execute(new ThreadSafe());
  13. }
  14. es.shutdown(); //不答应添加线程,异步关闭毗连池
  15. es.awaitTermination(10L, TimeUnit.SECONDS); //等候毗连池的线程任务完成
  16. System.out.println(count);
  17. }
  18. }
  19. /* output
  20. * 200000
  21. */
复制代码

5. 其他方法解决线程同步

a. 自旋锁

 线程循环反复查抄变量是否可用,在这一过程中线程不停保持执行(RUNNABLE),因此是一种忙等候,不像关键字synchronized一样,一旦发现不能访问,则处于线程处于阻塞状态(BLOCKED)。

  1. public class Thread6 implements Runnable{
  2. private static final Lock lock = new ReentrantLock();
  3. private volatile static int count = 0;
  4. @Override
  5. public void run() {
  6. for (int i = 0; i < 1000000; i++){
  7. lock.lock();
  8. count++;
  9. lock.unlock();
  10. }
  11. }
  12. static void test(ExecutorService es) throws InterruptedException {
  13. for (int i = 0; i < 20; i++) {
  14. es.execute(new Thread6());
  15. }
  16. es.shutdown(); //不答应添加线程,异步关闭毗连池
  17. es.awaitTermination(10L, TimeUnit.SECONDS); //等候毗连池的线程任务完成
  18. System.out.println(count);
  19. }
  20. public static void main(String[] args) throws InterruptedException {
  21. ExecutorService es = Executors.newFixedThreadPool(20);
  22. test(es);
  23. }
  24. }
复制代码

 假如在利用lock的时间包罗了try...catch...语句,要注意的是lock 必须在 finally 块中开释。否则,假如受保护的代码将抛出异常,锁就有可能永久得不到开释!

 与Lock类同一个包java.util.concurrent.locks下另有一种读写分离的锁ReentrantReadWriteLock类,读写锁维护了一对锁,一个读锁和一个写锁。一样平常情况下,读写锁的性能都会比排它锁好,因为大多数场景读是多于写的。在读多于写的情况下,读写锁可以或许提供比排它锁更好的并发性和吞吐量。

  

  1. public class RWTest {
  2. private static final Map<String, Object> map = new HashMap<String, Object>();
  3. private static final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
  4. private static final Lock readLock = lock.readLock();
  5. private static final Lock writeLock = lock.writeLock();
  6. public static final Object get(String key) {
  7. readLock.lock();
  8. try {
  9. return map.get(key);
  10. } finally {
  11. readLock.unlock();
  12. }
  13. }
  14. public static final Object put(String key, Object value) {
  15. writeLock.lock();
  16. try {
  17. return map.put(key, value);
  18. } finally {
  19. writeLock.unlock();
  20. }
  21. }
  22. public static final void clear() {
  23. writeLock.lock();
  24. try {
  25. map.clear();
  26. } finally {
  27. writeLock.unlock();
  28. }
  29. }
  30. }
复制代码

 利用自旋锁Lock也提供了Condition来实现线程间的状态关照的,可以根据实际情况去唤醒某个线程(与后面的wait差异,是随机的)或者所有线程。可以通过lock.newCondition()来获取得Condition实例,可以根据实际需求创建多个实例。

  1. public class Thread9 {
  2. public static ReentrantLock lock=new ReentrantLock();
  3. public static Condition condition =lock.newCondition();
  4. public static void main(String[] args) {
  5. new Thread(){
  6. @Override
  7. public void run() {
  8. lock.lock();//哀求锁
  9. try{
  10. System.out.println(Thread.currentThread().getName()+"==》进入等候");
  11. condition.await();//设置当火线程进入等候
  12. }catch (InterruptedException e) {
  13. e.printStackTrace();
  14. }finally{
  15. lock.unlock();//开释锁
  16. }
  17. System.out.println(Thread.currentThread().getName()+"==》继续执行");
  18. }
  19. }.start();
  20. new Thread(){
  21. @Override
  22. public void run() {
  23. lock.lock();//哀求锁
  24. try{
  25. System.out.println(Thread.currentThread().getName()+"=》进入");
  26. Thread.sleep(2000);//苏息2秒
  27. condition.signal();//随机唤醒等候队列中的一个线程
  28. System.out.println(Thread.currentThread().getName()+"苏息竣事");
  29. }catch (InterruptedException e) {
  30. e.printStackTrace();
  31. }finally{
  32. lock.unlock();//开释锁
  33. }
  34. }
  35. }.start();
  36. }
  37. }
  38. /*output
  39. *Thread-0==》进入等候
  40. *Thread-1=》进入
  41. *Thread-1苏息竣事
  42. *Thread-0==》继续执行
  43. */
复制代码

b. wait.notify.notifyAll

 在关键字synchronized的线程同步机制,调用线程的sleep,yield方法时,线程并不会让出对象锁,但是调用wait却差异,线程自动开释其占有的对象锁,同时不会去申请对象锁,当线程被唤醒的时间,它才再次去申请竞争对象的锁(该关键字通常只与synchronized团结利用)。notify()唤醒在等候该对象同步锁的线程(只唤醒一个,假如有多个在等候),注意的是在调用此方法的时间,并不能确切的唤醒某一个等候状态的线程,而是由JVM确定唤醒哪个线程,而且不是按优先级。而notifyAll()则是唤醒所有等候的线程。

  1. public class Thread8 implements Runnable {
  2. private int num;
  3. private Object lock;
  4. public Thread8(int num, Object lock) {
  5. this.num = num;
  6. this.lock = lock;
  7. }
  8. public void run() {
  9. try {
  10. while (true) {
  11. synchronized (lock) {
  12. lock.notifyAll();
  13. lock.wait();
  14. System.out.println(num);
  15. }
  16. }
  17. } catch (InterruptedException e) {
  18. e.printStackTrace();
  19. }
  20. }
  21. public static void main(String[] args) {
  22. final Object lock = new Object();
  23. Thread thread1 = new Thread(new Thread8(1, lock));
  24. Thread thread2 = new Thread(new Thread8(2, lock));
  25. thread1.start();
  26. thread2.start();
  27. }
  28. }
  29. /* output
  30. * 交替输出1,2,1,2,1,2......
  31. */
复制代码

6.并发编程—CountDownLatch、CyclicBarrier、Semaphore和fork/join框架

1. CountDownLatch

CountDownLatch实现的是一个倒序计数器,可以通过调用它的countDown实现计数器减一和await方法来阻塞当火线程:

  1. public class Thread10 {
  2. public static void main(String[] args) throws InterruptedException {
  3. int count = 20;
  4. final CountDownLatch cdl = new CountDownLatch(count);
  5. ExecutorService es = Executors.newCachedThreadPool();
  6. for (int i = 0; i < count; i++) {
  7. es.execute(new Runnable() {
  8. @Override
  9. public void run() {
  10. try {
  11. System.out.println(cdl.getCount());
  12. }finally {
  13. cdl.countDown();
  14. }
  15. }
  16. });
  17. }
  18. cdl.await();
  19. es.shutdown();
  20. System.out.println("主线程如今才竣事: count = "+cdl.getCount());
  21. }
  22. }
复制代码

2.CyclicBarrier

即回环栅栏,是一种可重用的线程阻塞器,它将率先到达栅栏的这些线程阻塞(调用await()方法),直到指定命量的线程都到达该处,这些线程将会被全部开释。

  1. public class Thread11 implements Runnable{
  2. private int num;
  3. private static CyclicBarrier cb = new CyclicBarrier(6); //指定栅栏的等候线程数
  4. public Thread11(int num){
  5. this.num = num;
  6. }
  7. @Override
  8. public void run() {
  9. try {
  10. Thread.sleep(1000*num); //等候指定命量时间后到达栅栏处
  11. System.out.println(Thread.currentThread().getName() +" is coming..");
  12. cb.await(10L, TimeUnit.SECONDS);
  13. System.out.println("continue....");
  14. } catch (Exception e) {
  15. e.printStackTrace();
  16. }
  17. }
  18. public static void main(String[] args) {
  19. ExecutorService es = Executors.newCachedThreadPool();
  20. for (int i = 0; i < 8; i++) {
  21. es.execute(new Thread11(i));
  22. }
  23. es.shutdown();
  24. }
  25. }
  26. /*
  27. *pool-1-thread-1 is coming..
  28. *pool-1-thread-2 is coming..
  29. *pool-1-thread-3 is coming..
  30. *continue....
  31. *continue....
  32. *continue....
  33. *pool-1-thread-4 is coming..
  34. *超时异常错误(指定时间内线程数量仍然到达)
  35. */
复制代码

3.Semaphore信号量

  信号量用于保护对一个或多个共享资源的访问,其内部维护一个计数器,用来只是当前可以访问共享资源的数量。可以通过tryAcquire去实行获取答应,还可以通过availablePermits()方法得到可用的答应数量,而acquire/release则是获取/开释答应。

  1. public class Thread12 implements Runnable {
  2. private static SecureRandom random= new SecureRandom();
  3. private static Semaphore semaphore = new Semaphore(3, true);
  4. @Override
  5. public void run() {
  6. try {
  7. semaphore.acquire();
  8. System.out.println(Thread.currentThread().getName() + " got permission...");
  9. Thread.sleep(random.nextInt(10000));
  10. semaphore.release();
  11. System.out.println(Thread.currentThread().getName() + " released permission...");
  12. } catch (InterruptedException e) {
  13. e.printStackTrace();
  14. }
  15. }
  16. public static void main(String[] args) {
  17. ExecutorService es = Executors.newCachedThreadPool();
  18. for (int i = 0; i < 6; i++) {
  19. es.execute(new Thread12());
  20. }
  21. es.shutdown();
  22. }
  23. }
复制代码

4.fork/join框架

 Fork/Join框架提供了的一个用于并行执行任务的框架,充实利用了CPU资源,把大任务分割成多少个小任务,终极汇总每个小任务结果后得到大任务结果的框架。(只供Java7利用)

Fork/Join利用两个类:

  • ForkJoinTask:我们要利用ForkJoin框架,必须起首创建一个ForkJoin任务。它提供在任务中执行fork()和join()操纵的机制,Fork/Join框架提供了以下两个子类:

    • RecursiveAction:用于没有返回结果的任务。

    • RecursiveTask :用于有返回结果的任务。

  • ForkJoinPool :ForkJoinTask必要通过ForkJoinPool来执行,任务分割出的子任务会添加到当前工作线程所维护的双端队列中。

ForkJoinPool与其他类型的ExecutorService的差异之处主要在于利用工作盗取,每个线程都有自己的双端任务队列,线程在一样平常情况下会从队列头去获取任务,当某个线程任务队列的为空的时间,它会实行从其他线程的任务队列的尾部去“盗取”任务来执行。

  1. public class Thread13 extends RecursiveTask<Integer> {
  2. private int start;
  3. private int end;
  4. public Thread13(int start, int end) {
  5. this.start = start;
  6. this.end = end;
  7. }
  8. @Override
  9. protected Integer compute() {
  10. int m = 1000; //每个线程盘算的范围大小
  11. int s = start, n = end; //每个线程盘算的起始地点
  12. int r = 0; //算和的变量
  13. List<ForkJoinTask<Integer>> it = new ArrayList<ForkJoinTask<Integer>>();
  14. while (s <= n) {
  15. if (n - s < m) {
  16. for (int i = s; i <= n; i++) {
  17. r += i;
  18. }
  19. } else {
  20. n = Math.min(s + m - 1, n); //得到一个新的start
  21. it.add(new Thread13(s, n).fork()); //得到每一个范围[如(0,999)]参加一个线程
  22. }
  23. s = n + 1;
  24. n = end;
  25. }
  26. for (ForkJoinTask<Integer> t : it) {
  27. r += t.join();
  28. }
  29. return r;
  30. }
  31. public static void main(String[] args) throws ExecutionException, InterruptedException {
  32. ForkJoinPool fjp = new ForkJoinPool();
  33. int s = 1, n = 10001;
  34. Future<Integer> rs = fjp.submit(new Thread13(s, n));
  35. System.out.println(rs.get());
  36. }
  37. }
  38. /* output
  39. * 50015001
  40. */
复制代码

  


参考文献

  1. 庞永华. Java多线程与Socket:实战微服务框架[M].电子工业出版社.2019.3

  2. Executors类中创建线程池的几种方法的分析

  3. 知乎——假如你这样答复“什么是线程安全”,面试官都会对你刮目相看

  4. 知乎——Java线程内存模子,线程、工作内存、主内存

  5. Java进阶——Java中的Atomic原子特性

  6. 深入明白Java并发之synchronized实现原理

  7. Java的wait(), notify()和notifyAll()利用小结

  8. java多线程-07-Lock和Condition

  9. Java并发编程:CountDownLatch、CyclicBarrier和Semaphor







来源:https://www.cnblogs.com/helloworldcode/p/11728321.html
C#论坛 www.ibcibc.com IBC编程社区
C#
C#论坛
IBC编程社区
*滑块验证:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则