线程安全之Synchronized关键字

 之前我讲了关于 线程基础方面的相关知识,本篇文章将会带着大家来学习下线程安全相关的知识。

成都创新互联服务项目包括金寨网站建设、金寨网站制作、金寨网页制作以及金寨网络营销策划等。多年来,我们专注于互联网行业,利用自身积累的技术优势、行业经验、深度合作伙伴关系等,向广大中小型企业、政府机构等提供互联网行业的解决方案,金寨网站推广取得了明显的社会效益与经济效益。目前,我们服务的客户以成都为中心已经辐射到金寨省份的部分城市,未来相信会继续扩大服务区域并继续获得客户的支持与信任!

1 多线程下为什么会存在线程安全问题

线程的合理使用能够提升程序的处理性能,一是能够利用多核 CPU 来实现线程的并行执行,二是线程的异步化执行能够提高系统的吞吐量。

虽然线程有这些优点,但同时也带来了很多问题。比如说:

1.1 共享变量带来的安全性问题

先来看个图:

 

一个变量 i ,如果线程 A 或者线程 B 单独访问并且修改变量 i 的值没有任何问题,那如果并行的修改变量 i ,那就会有安全性问题。

然后用代码来模拟一下这种场景,为了更好的看到效果,我用100个线程:

 
 
 
 
  1. public class ThreadDemo1 { 
  2.  
  3.     private static int i = 0; 
  4.  
  5.     public static void inc() { 
  6.         try { 
  7.             Thread.sleep(1); 
  8.         } catch (InterruptedException e) { 
  9.             e.printStackTrace(); 
  10.         } 
  11.         i++; 
  12.     } 
  13.  
  14.     public static void main(String[] args) throws InterruptedException { 
  15.         for (int i = 0; i < 100; i++) { 
  16.             new Thread(() -> ThreadDemo1.inc()).start(); 
  17.         } 
  18.         Thread.sleep(1000); 
  19.         System.out.println("运行结果" + i); 
  20.     } 

输出结果:

88

这个输出结果是不固定的,第一次可能是 88 ,第二次可能是 87 ,这个结果就和我们预期的结果不一致(预期结果是100),所以一个对象是否是线程安全的,取决于它是否会被多个线程访问,以及程序中是如何去使用这个对象的。如果 多个线程访问同一个共享对象,在不需额外的同步以及调用端代码不用做其他协调的情况下,这个共享对象的状态 依然是正确的(正确性意味着这个对象的结果与我们预期 规定的结果保持一致),那说明这个对象是线程安全的。

对于线程安全性,本质上是管理对于数据状态的访问,而且这个这个状态通常是共享的、可变的。共享:是指这个 数据变量可以被多个线程访问;可变:指这个变量的值在 它的生命周期内是可以改变的。

2.如何保证线程并行的数据安全性-Synchroinzed

针对上面那种情况,我们该如何解决这种问题呢?首先想到的就是加锁,并且这种锁必须是互斥的。比如上面的图片的例子,如果线程A在修改 i 的值时,线程 B 就不能去修改 i 的值。也就是说并行去修改共享变量的值会有线程安全性问题,那么我们不让你并行,不就解决了这个问题嘛。所以java提供了 Synchroinzed 关键字。

2.1 Synchroinzed 的基本认识

Synchroinzed 很早就有了,只是之前是重量级锁,所以很好有人使用。在 javaSE 1.6 对Synchroinzed进行了优化引入了偏向锁和轻量级锁。所以在并发量不高的情况还是推荐使用 Synchroinzed 来加锁。为什么是并发量不高的情况推荐使用,因为并发量高的情况 Synchroinzed 会升级为重量级锁。

2.2 Synchroinzed 的三种加锁方式

  1. 修饰实例方法,锁是当前实例对象 ,进入同步代码前要获得当前实例的锁
  2. 修饰静态方法,锁是当前类的class对象 ,进入同步代码前要获得当前类对象的锁
  3. 修饰代码块,锁是括号里面的对象,对给定对象加锁,进入同步代码库前要获得给定对象的锁。

看下简单的代码

 
 
 
 
  1. public class SynchroinzedDemo { 
  2.  
  3.     /** 
  4.      * 对静态方法加锁 
  5.      */ 
  6.     public static synchronized void test(){} 
  7.     /** 
  8.      * 对实例方法加锁 
  9.      */ 
  10.     public synchronized void test1(){} 
  11.     /** 
  12.      * 对代码块加锁 
  13.      */ 
  14.     public void test2(){ 
  15.         synchronized(this){} 
  16.     } 

然后我们将上面的例子实现 synchronized 加锁:

 
 
 
 
  1. public class ThreadDemo1 { 
  2.  
  3.     private static int i = 0; 
  4.  
  5.     public static void inc() { 
  6.         synchronized (ThreadDemo1.class){ 
  7.             try { 
  8.                 Thread.sleep(1); 
  9.             } catch (InterruptedException e) { 
  10.                 e.printStackTrace(); 
  11.             } 
  12.             i++; 
  13.         } 
  14.     } 
  15.  
  16.     public static void main(String[] args) throws InterruptedException { 
  17.         for (int i = 0; i < 100; i++) { 
  18.             new Thread(() -> ThreadDemo1.inc()).start(); 
  19.         } 
  20.         Thread.sleep(1000); 
  21.         System.out.println("运行结果" + i); 
  22.     } 

运行结果:

运行结果100

完美的解决共享变量并行修改带来的线程安全问题。

3 总结

本文带着大家了解了一下线程的安全性问题和解决线程安全性问题的 synchronized 关键字的用法。后面的并发编程系列会讲解更多的解决线程安全性的方法。敬请期待!

文章名称:线程安全之Synchronized关键字
文章起源:http://www.mswzjz.cn/qtweb/news15/225565.html

攀枝花网站建设、攀枝花网站运维推广公司-贝锐智能,是专注品牌与效果的网络营销公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 贝锐智能