JDK中Thread.State类的几种状态

线程的生命周期









线程的安全问题(同步与互斥)

方法一:同步代码块

多个线程的同步监视器(锁)必须的是同一把,任何一个类的对象都可以

  1. synchronized(同步监视器){
  2. } //说明:操作共享数据的代码,即为需要被同步的代码(对共享数据的操作)

对于Thread实现和Runnable实现,同步锁需要区别

1.Thread下的同同步锁

  1. class WinThread extends Thread{
  2. /*
  3. 方式一 继承Thread
  4. */
  5. private static int ticket=100; //需要设置静态变量
  6. private static Object ticketLock=new Object(); //同步监视器,静态,因为要是共享唯一变量
  7. @Override
  8. public void run() {
  9. while(true)
  10. {
  11. synchronized (ticketLock){
  12. if (ticket > 0)
  13. {
  14. System.out.println(this.getName() + "卖票: " + ticket);
  15. ticket--;
  16. }
  17. else break;
  18. }
  19. }
  20. }
  21. }

2.Runnable下的同步锁

  1. class winRunnable implements Runnable{
  2. private int ticket=100; //不用设置成静态变量
  3. private Object obj=new Object(); //声明一个对象为锁
  4. @Override
  5. public void run() {
  6. while(true)
  7. {
  8. try {
  9. sleep(100);
  10. } catch (InterruptedException e) {
  11. e.printStackTrace();
  12. }
  13. synchronized(obj){ //互斥操作
  14. if (ticket > 0)
  15. {
  16. System.out.println(Thread.currentThread().getName() + "卖票: " + ticket);
  17. ticket--;
  18. }
  19. else break;
  20. }
  21. }
  22. }
  23. }

Thread下可以考虑用WinThread.class当锁,而Runnable下可以考虑用this当锁(总之要分析是不是同一个锁)

自己的坑:不能把共享的数据当作锁(例如WinThread下的int ticket改成Integer后当锁),这里还是会有问题

方法二:同步方法(代码略)

多声明一个对共享数据数据操作的方法,在方法的前面添加声明synchronized,而不是直接在run()前面加

1.同步方法中也涉及到同步锁,但锁是默认的

2.非静态的同步方法,锁是this

静态的同步方法,锁是当前类本身——xxx.class

  1. * 同步方法synchronized实现runnable接口
  2. * private synchronized void 方法(){//普通同步方法的同步监视器是this
  3. * //需要互斥的代码
  4. * }
  1. * 同步方法synchronized实现继承Thread
  2. * 因为不同线程是Thread类的不同对象,所以一般要用静态
  3. * private static synchronized void 方法(){//静态同步方法默认同步监视器是类本身
  4. * //需要互斥的代码
  5. * }

同步的局限性:操作同步代码时,只能有一个线程参与,其他线程等待,相当于互斥量为1

单例模式的同步问题

死锁问题

方法二:Lock(锁) ---JDK5.0新增

  1. class LockRunnable implements Runnable{
  2. private int ticket=100; //不用设置成静态变量
  3. private ReentrantLock lock=new ReentrantLock(true); //true为选择FIFO的公平
  4. @Override
  5. public void run() {
  6. while(true)
  7. {
  8. try {
  9. try {
  10. sleep(100);
  11. } catch (InterruptedException e) {
  12. e.printStackTrace();
  13. }
  14. lock.lock();
  15. if (ticket > 0)
  16. {
  17. System.out.println(Thread.currentThread().getName() + "卖票: " + ticket);
  18. ticket--;
  19. }
  20. else break;
  21. }finally {
  22. lock.unlock();
  23. }
  24. }
  25. }
  26. }

面试题:synchronized 与 Lock 的区别

相同:二者都可以解决线程安全问题

不同:synchronized机制在执行完相应的同步代码之后,自动的释放同步监视器

Lock需要手动的启动同步 .lock() ,同时结束同步也需要手动 .unlock()

Java多线程—线程同步(单信号量互斥)的更多相关文章

  1. Java多线程——线程同步

    在之前,已经学习到了线程的创建和状态控制,但是每个线程之间几乎都没有什么太大的联系.可是有的时候,可能存在多个线程多同一个数据进行操作,这样,可能就会引用各种奇怪的问题.现在就来学习多线程对数据访问的 ...

  2. Java多线程 - 线程同步

    多线程操作同一个对象时,容易引发线程安全问题.为了解决线程安全问题,Java多线程引入了同步监视器. 同步代码块 同步代码块语法格式如下: synchronized(obj){ //此处的代码即为同步 ...

  3. JAVA多线程线程同步问题

    线程同步 在多线程的编程环境下,可能看着没有问题的代码在运行几千上万或者更多次后,出现了一些看着很奇怪的问题,出现这样的问题的原因就是可能会有两个或者更多个线程进入了同一块业务处理代码中导致了判断失效 ...

  4. Java中线程同步锁和互斥锁有啥区别?看完你还是一脸懵逼?

    首先不要钻概念牛角尖,这样没意义. 也许java语法层面包装成了sycnchronized或者明确的XXXLock,但是底层都是一样的.无非就是哪种写起来方便而已. 锁就是锁而已,避免多个线程对同一个 ...

  5. Java多线程:线程同步与关键字synchronized

    一.同步的特性1. 不必同步类中所有的方法, 类可以同时拥有同步和非同步方法.2. 如果线程拥有同步和非同步方法, 则非同步方法可以被多个线程自由访问而不受锁的限制. 参见实验1:http://blo ...

  6. Java多线程——线程之间的同步

    Java多线程——线程之间的同步 摘要:本文主要学习多线程之间是如何同步的,如何使用volatile关键字,如何使用synchronized修饰的同步代码块和同步方法解决线程安全问题. 部分内容来自以 ...

  7. Java多线程-线程的同步(同步方法)

    线程的同步是保证多线程安全访问竞争资源的一种手段.线程的同步是Java多线程编程的难点,往往开发者搞不清楚什么是竞争资源.什么时候需要考虑同步,怎么同步等等问题,当然,这些问题没有很明确的答案,但有些 ...

  8. Java多线程的同步控制记录

    Java多线程的同步控制记录 一.重入锁 重入锁完全可以代替 synchronized 关键字.在JDK 1.5 早期版本,重入锁的性能优于 synchronized.JDK 1.6 开始,对于 sy ...

  9. Java多线程——线程的创建方式

    Java多线程——线程的创建方式 摘要:本文主要学习了线程的创建方式,线程的常用属性和方法,以及线程的几个基本状态. 部分内容来自以下博客: https://www.cnblogs.com/dolph ...

随机推荐

  1. 详细描述一下 Elasticsearch 搜索的过程?

    想了解 ES 搜索的底层原理,不再只关注业务层面了. 解答: 搜索拆解为"query then fetch" 两个阶段. query 阶段的目的:定位到位置,但不取. 步骤拆解如下 ...

  2. spring源码-扩展点

    /** * @Author quan * @Date 2020/11/13 * 扩展原理 * BeanPostProcessor bean后置处理器,bean创建对象初始化前后进行拦截工作 * * * ...

  3. web.xml---配置文件概要

    web.xml分发器: case1: springMvc的分发器: 作用:将匹配上的请求交由springMvc处理,路径会继续到达springMvc的处理器映射器 <servlet> &l ...

  4. producer内存管理分析

    1 概述 kafka producer调用RecordAccumulator#append来将消息存到本地内存.消息以TopicPartition为key分组存放,每个TopicPartition对应 ...

  5. SpringAop实现原理及代理模式

    Spring Aop的原理 Spring的AOP就是通过动态代理实现的.当为某个Bean或者某些Bean配置切面时,Spring会为其创建代理对象,当调用该对象的某个方法时,实际是调用生成的代理类的对 ...

  6. visual studio 2019工具里添加开发中命令提示符的方法

    最新新装了visual studio 2019,发现默认的没有开发者命令提示符 现将添加步骤描述如下: 从VS2019菜单选择"Tools",然后选择"外部工具" ...

  7. FR9833 32V转5V

  8. ionic的checkbox分析

    之前分析了一个原生的checkbox,趁热打铁分析ionic的自带checkbox. html <label class="checkbox"> <input t ...

  9. PYthon窗口学习之用异步请求解决Treeview列表插入大量数据反应慢的解决办法

    当列表插入大量数据时,经常会等一会才显示数据 异步请求就将每一个插入语句并发运行,从而提高插入速度 代码: # 显示结果 def insert_result(table, info): def ins ...

  10. win10 Celery异步任务报错: Task handler raised error: ValueError('not enough values to unpack (expected 3, got 0)

    示例代码如下: from celery import Celery app = Celery('tasks', backend='redis://×××:6379/1', broker='redis: ...