线程同步

问题引入
观察一面一段小程序:
  1. public class Main {
  2. private static int amount = 0;
  3. public static void main(String[] args) {
  4. System.out.println(++amount);
  5. new MyThread("thread1").start();
  6. new MyThread("thread2").start();
  7. }
  8. private static void calc(String tag){
  9. ++amount;
  10. try {
  11. Thread.sleep(1);
  12. } catch (InterruptedException e) {
  13. e.printStackTrace();
  14. }
  15. System.out.println(tag + "==>" + amount);
  16. }
  17. static class MyThread extends Thread {
  18. private String name;
  19. public MyThread(String name) {
  20. super();
  21. this.name = name;
  22. }
  23. @Override
  24. public void run() {
  25. super.run();
  26. calc(name);
  27. }
  28. }
  29. }
运行结果:
上面的代码MyThread中先将amount累加1;再睡1S,打印数据。很显然thread1睡眠之后被打断,thread2被执行才会出现这样的情况。
系统对线程的调度是有一定随机性的。当多线程操作资源时,才会出现线程安全的问题。Java 提供了一系列的方案来解决这个问题,下面将一一说明。
Synchronized关键字
同步方法块
  1. synchronized(obj){
  2. ...
  3.        //此处是同步代码块
  4. }
当线程执行此段代码时必须持有锁,否则不能执行,上面代码修改如下所示即可恢复正常:
  1. //对象锁
  2. private static Object obj = new Object();
  3. private static void calc(String tag) {
  4. //执行此代码必须持有此锁,没有锁只能等待此锁的使用线程释放锁
  5. synchronized (obj) {
  6. ++amount;
  7. try {
  8. Thread.sleep(1);
  9. } catch (InterruptedException e) {
  10. e.printStackTrace();
  11. }
  12. System.out.println(tag + "==>" + amount);
  13. }
  14. }

同步方法
此关键字还可以直接修饰方法,被修饰的方法为同步方法。同步方法的使用也必须持有锁,此时锁定的是this(当前对象),
  1. private synchronized static void calc(String tag) {
  2. // 执行此代码必须持有此锁,没有锁只能等待此锁的使用线程释放锁
  3. ++amount;
  4. try {
  5. Thread.sleep(1);
  6. } catch (InterruptedException e) {
  7. e.printStackTrace();
  8. }
  9. System.out.println(tag + "==>" + amount);
  10. }
释放锁
   synchronized修饰的代码块或者同步方法没有显性的释放锁的方法。遇到以下几种情况会释放锁。
  • 使用synchronized关键字,必须等待当前线程执行完同步代码块或者同步方法,锁被释放。
  • 遇到return、break终止了当前代码块或者同步方法,锁被释放。
  • 遇到未处理的异常导致的崩溃,当前锁被释放。
  • 线程调用wait方法,释被放锁。
下面两点不会释放锁,需要注意:
  • 程序调用Thread.sleep()、Thread.yield(),当前线程不会释放锁。
  • 其他线程调用suspend方法,将此方法挂起,当前线程不会释放锁。同时应该避免使用suspend和resume方法控制线程。
访问权限总结
访问权限原则:当前线程持有某个锁时,其他线程无法访问被同一个锁锁定的方法或者代码块;与是否在一个类中,是否是静态无关,只与锁是否被释放有关。
常见的问题如下:
  • 一个类中,有多个方法,当前线程持有某锁,其他线程无法访问此锁锁定的其他同步方法,但是可以访问其他非同步方法
  • 一个类中有多个代码块或同步方法,当前线程持有某所,其他线程可以访问其他锁锁定的同步代码块

三、线程同步之Sysnchronized关键字的更多相关文章

  1. 多线程,线程同步,synchronized关键字的用法

    一.什么是多线程 Java多线程实现方式主要有四种:继承Thread类.实现Runnable接口.实现Callable接口通过FutureTask包装器来创建Thread线程.使用ExecutorSe ...

  2. MFC线程(三):线程同步事件(event)与互斥(mutex)

    前面讲了临界区可以用来达到线程同步.而事件(event)与互斥(mutex)也同样可以做到. Win32 API中的线程事件 HANDLE hEvent = NULL; void MainTestFu ...

  3. Java:多线程,线程同步,synchronized关键字的用法(同步代码块、非静态同步方法、静态同步方法)

    关于线程的同步,可以使用synchronized关键字,或者是使用JDK 5中提供的java.util.concurrent.lock包中的Lock对象.本文探讨synchronized关键字. sy ...

  4. java笔记--关于线程同步(7种同步方式)

    关于线程同步(7种方式) --如果朋友您想转载本文章请注明转载地址"http://www.cnblogs.com/XHJT/p/3897440.html"谢谢-- 为何要使用同步? ...

  5. JAVA中线程同步的方法(7种)汇总

    同步的方法: 一.同步方法 即有synchronized关键字修饰的方法. 由于java的每个对象都有一个内置锁,当用此关键字修饰方法时, 内置锁会保护整个方法.在调用该方法前,需要获得内置锁,否则就 ...

  6. java 多线程总结篇3之——生命周期和线程同步

    一.生命周期 线程的生命周期全在一张图中,理解此图是基本: 线程状态图 一.新建和就绪状态 当程序使用new关键字创建了一个线程之后,该线程就处于新建状态,此时它和其他的Java对象一样,仅仅由Jav ...

  7. Java:多线程,线程同步,同步锁(Lock)的使用(ReentrantLock、ReentrantReadWriteLock)

    关于线程的同步,可以使用synchronized关键字,或者是使用JDK 5中提供的java.util.concurrent.lock包中的Lock对象.本文探讨Lock对象. synchronize ...

  8. java中线程同步的几种方法

    1.使用synchronized关键字 由于java的每个对象都有一个内置锁,当用此关键字修饰方法时, 内置锁会保护整个方法.在调用该方法前,需要获得内置锁,否则就处于阻塞状态. 注: synchro ...

  9. java笔记--关于线程同步(5种同步方式)【转】

    为何要使用同步?     java允许多线程并发控制,当多个线程同时操作一个可共享的资源变量时(如数据的增删改查),      将会导致数据不准确,相互之间产生冲突,因此加入同步锁以避免在该线程没有完 ...

随机推荐

  1. Celery的实践指南

    http://www.cnblogs.com/ToDoToTry/p/5453149.html Celery的实践指南   Celery的实践指南 celery原理: celery实际上是实现了一个典 ...

  2. checkbox全选与非全选之间的切换

    <div id="congras_area"> <input type="checkbox" name="" id=&qu ...

  3. PHP图片加文字水印和图片水印方法(鉴于李老师博客因没加水印被盗,特搜集的办法。希望能有用!)

    $dst_path = 'dst.jpg'; //创建图片的实例 $dst = imagecreatefromstring(file_get_contents($dst_path)); //打上文字 ...

  4. React的井字过三关(1)

    React的井字过三关(1) 本文系React官方教程的Tutorial: Intro To React的笔记.由笔者用ES5语法改写. 在本篇笔记中,尝试用React构建一个可交互的井字棋游戏. 开 ...

  5. [Java] Spring MVC 知识点

    云图: @Service 用于标注业务层组件. 在 Spring-servlet 配置xml中,component-scan 标签默认情况下自动扫描指定路径下的包(含所有子包),将带有@Compone ...

  6. 更换域名后的数据库sql的执行命令

    原来域名为trz.lqzcw.com 更改成 www.trzbearing.com UPDATE wp_options SET option_value = replace(option_value, ...

  7. yum 操作复习

    RPM与YUM是配合使用的. rpm负责从网站或指定的文件路径,获取到rpm软件包.也就是说你要安装什么服务或软件,就要先找到rpm包,下载下来.也就是通常说的配置yum源. 啥是yum源.你下载下的 ...

  8. KMP学习之旅

    说起kmp就要从字符串的匹配说起,下面我们谈谈字符串的匹配 给定一个原字符串:bababababababababb,再给定一个模式串:bababb,求模式串是否在源字符串中出现 最简单的方法就是遍历源 ...

  9. 关于TCP中的MSS

    MSS 是TCP选项中最经常出现,也是最早出现的选项.MSS选项占4byte.MSS是每一个TCP报文段中数据字段的最大长度,注意:只是数据部分的字段,不包括TCP的头部.TCP在三次握手中,每一方都 ...

  10. HTML <input type="file">上传文件——结合asp.net的一个文件上传示例

    HTML的代码:(关键是要在form里设置enctype="multipart/form-data",这样才能在提交表单时,将文件以二进制流的形式传输到服务器) 一. <fo ...