由于线程的执行是CPU随机调度的,比如我们开启10个线程,这10个线程并不是同时执行的,而是CPU快速的在这10个线程之间切换执行,由于切换速度极快使我们感觉同时执行罢了。

线程同步问题往往发生在多个线程调用同一方法或者操作同一变量,但是我们要知道其本质就是CPU对线程的随机调度,CPU无法保证一个线程执行完其逻辑才去调用另一个线程执行。

比如:

4个窗口同时售卖100张车票:

public static void main(String[] args) {         //测试: 卖票的动作.
//1. 因为是四个窗口, 所以需要创建四个线程对象. 给线程自定义名字
MyThread mt1 = new MyThread("窗口1");
MyThread mt2 = new MyThread("窗口2");
MyThread mt3 = new MyThread("窗口3");
MyThread mt4 = new MyThread("窗口4"); //2. 开启线程
mt1.start();
mt2.start();
mt3.start();
mt4.start();
} public class MyThread extends Thread{
//需求: 四个窗口, 卖100张票.
/*
* 思路:
* 1. 定义一个变量(tickets), 记录票数.
* 2. 因为是四个窗口 同时 卖票, 通过多线程卖票.
*/
//1. 定义一个变量(tickets), 记录票数.
private static int tickets = 100; //因为是共享数据, 所以用static修饰 //3. 使用父类的构造方法
public MyThread() {
super();
} public MyThread(String name) {
super(name);
} //2. 因为是四个窗口 同时 卖票, 通过多线程卖票.
@Override
public void run() {
/*
* 卖票的动作 的 思路:
* A: 因为不知道还有多少张票要买, 所以用while(true).
* B: 做一下越界处理. 没票就不卖了.
* C: 如果有票, 就正常的卖票即可.
*/
//A
while(true) {
//B
if (tickets < 1) {
break;
} //为了加大出现错误票的概率, 我们加入: 休眠线程的概念
try {
Thread.sleep(20);
} catch (InterruptedException e) {
e.printStackTrace();
} //C // 线程2休眠, 线程3休眠, 线程4休眠,
System.out.println(getName() + "正在出售第" + tickets-- + "张票"); /*
* 出现负数票的原因: if()判断, 休眠线程
* 假设现在卖到最后一张票了, tickets的值应该是1, 此时,
* 如果线程1抢到了资源, 会越过if(), 然后休眠, 以此类推, 四个线程都处于休眠的状态.
*
* 休眠时间过了之后, 程序继续运行.
* 假设线程1先抢到资源, 打印: 出售1号票, 然后会把tickets的值改为: 0
* 假设线程2后抢到资源, 打印: 出售0号票, 然后会把tickets的值改为: -1
* 假设线程3后抢到资源, 打印: 出售-1号票, 然后会把tickets的值改为: -2
* 假设线程4后抢到资源, 打印: 出售-2号票, 然后会把tickets的值改为: -3
*/ /*
* 出现重复值的原因: tickets--
* tickets-- 相当于 tickets = tickets - 1
* tickets-- 做了 3件事:
* A: 读值. 读取tickets的值.
* B: 改值. 将tickets的值 - 1.
* C: 赋值. 将修改后的值重新赋值给 tickets.
* 还没有来得及执行 C的动作, 此时别的线程抢走资源了, 就会出现重复值.
*
*/
}
} }

运行结果会出现下面这种一张票重复售卖或者出现卖负数票的情况:

窗口2正在出售第99张票
窗口1正在出售第98张票
窗口3正在出售第99张票
窗口3正在出售第95张票
窗口3正在出售第94张票
窗口3正在出售第93张票

所以我们可以说这种多个线程同时并发操作同一数据的时候会出现错误,是有安全问题的。

解决方案:

可使用同步代码块或同步方法来解决(synchronized )

//A
while(true) {
//while循环中的代码就是一次完成的卖票过程, 之所以会出现非法值的票,
//原因是因为: 某个线程在卖票期间, 被别的线程抢走CPU资源了.
//其实解决方案很简单: 在某一个线程卖(一次)票期间, 别的线程不能干预. synchronized (MyThread.class) {
//要加锁的代码
//B
if (tickets < 1) {
break;
} //为了加大出现错误票的概率, 我们加入: 休眠线程的概念
try {
Thread.sleep(20);
} catch (InterruptedException e) {
e.printStackTrace();
} //C // 线程2休眠, 线程3休眠, 线程4休眠,
System.out.println(getName() + "正在出售第" + tickets-- + "张票");
} }

综上所述,我们发现如果多个线程同时操作同一数据的情况,使用同步代码块或方法同步就可以解决了。

那么我们下面说一下同步代码块和同步方法的锁对象:

非静态方法同步对应的锁是this

public class Demo {
   //同步代码块
public void method1() {
synchronized (this) {
System.out.print("山");
System.out.print("东");
System.out.print("张");
System.out.print("学");
System.out.print("友");
System.out.print("\r\n");
}
}
    //同步方法
  public synchronized void method2() {
System.out.print("s");
System.out.print("d");
System.out.print("z");
System.out.print("x");
System.out.print("y");
System.out.print("\r\n");
}
}

静态方法同步对应的锁是当前类的字节码文件对象:

public class Demo {
  //静态方法代码块同步
public static void method1() {
synchronized (Demo.class) {
System.out.print("山");
System.out.print("东");
System.out.print("张");
System.out.print("学");
System.out.print("友");
System.out.print("\r\n");
}
}
   //同步静态方法 
  public static synchronized void method2() {
System.out.print("s");
System.out.print("d");
System.out.print("z");
System.out.print("x");
System.out.print("y");
System.out.print("\r\n");
}
}

这里说明一下:

/*
* 静态同步方法 和 非静态同步方法的锁对象
* 静态方法:
锁对象: 该类的字节码文件对象. 也就是上面的Demo.class
非静态方法:
锁对象: this
*/

总结:普通同步方法的锁对象是this,静态同步方法的锁对象是类的字节码文件对象

   当前类实例对象,同步代码块锁可以自己定义,只要保证操作同一数据的线程使用的是同一把锁即可。

java基础之多线程三:多线程并发同步的更多相关文章

  1. java基础解析系列(三)---HashMap

    java基础解析系列(三)---HashMap java基础解析系列 java基础解析系列(一)---String.StringBuffer.StringBuilder java基础解析系列(二)-- ...

  2. Java基础系列3:多线程超详细总结

    该系列博文会告诉你如何从入门到进阶,一步步地学习Java基础知识,并上手进行实战,接着了解每个Java知识点背后的实现原理,更完整地了解整个Java技术体系,形成自己的知识框架. 1.线程概述 几乎所 ...

  3. 【Java基础】【25多线程(下)&GUI】

    25.01_多线程(单例设计模式)(掌握) 单例设计模式:保证类在内存中只有一个对象. 如何保证类在内存中只有一个对象呢? (1)控制类的创建,不让其他类来创建本类的对象.private (2)在本类 ...

  4. Java提高班(三)并发中的线程同步与锁

    乐观锁.悲观锁.公平锁.自旋锁.偏向锁.轻量级锁.重量级锁.锁膨胀...难理解?不存的!来,话不多说,带你飙车. 上一篇介绍了线程池的使用,在享受线程池带给我们的性能优势之外,似乎也带来了另一个问题: ...

  5. Java基础(七)——多线程

    一.概述 1.介绍 Java VM 启动的时候会有一个进程Java.exe,该进程中至少有一个线程负责Java程序的执行.而且这个线程运行的代码存在于main方法中,该线程称之为主线程.其实从细节上来 ...

  6. 【Java基础】【24多线程(上)】

    24.01_多线程(多线程的引入)(了解) 1.什么是线程 线程是程序执行的一条路径, 一个进程中可以包含多条线程 多线程并发执行可以提高程序的效率, 可以同时完成多项工作 2.多线程的应用场景 红蜘 ...

  7. JAVA基础学习-集合三-Map、HashMap,TreeMap与常用API

    森林森 一份耕耘,一份收获 博客园 首页 新随笔 联系 管理 订阅 随笔- 397  文章- 0  评论- 78  JAVA基础学习day16--集合三-Map.HashMap,TreeMap与常用A ...

  8. Java基础知识点(三)

    前言:准备将Java基础知识点总结成一个系列,用于平常复习并加深理解.每篇尽量做到短小精悍,便于阅读. 1.Math类中相关函数 Math.floor(x):返回不大于x的最大整数.eg:Math.f ...

  9. Java基础学习笔记: 多线程,线程池,同步锁(Lock,synchronized )(Thread类,ExecutorService ,Future类)(卖火车票案例)

    多线程介绍 学习多线程之前,我们先要了解几个关于多线程有关的概念.进程:进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能. 线 ...

  10. Java 并发和多线程(三) 多线程的代价 [转]

    原文链接:http://tutorials.jenkov.com/java-concurrency/costs.html 作者:Jakob Jenkov     翻译:古圣昌        校对:欧振 ...

随机推荐

  1. OLE剪切板与拖拽

    https://www.xuebuyuan.com/1074399.html https://blog.csdn.net/uda1985/article/details/6179801

  2. C#调用EasyPusher推送到EasyDarwin实现视频流中转

    本文转自:http://www.cnblogs.com/kangkey/p/6772863.html 最近在公司项目中,遇到需要将内网的监控视频信息,在外网进行查看,最终通过查阅资料,发现EasyDa ...

  3. 你所不知道的,Java 中操作符的秘密?

    在 Java 编程的过程中,我们对数据的处理,都是通过操作符来实现的.例如,用于赋值的赋值操作符.用于运算的运算操作符等.用于比较的比较操作符,还包括逻辑操作符.按位操作符.移位操作符.三元操作符等等 ...

  4. .net 系列化与反序列化(转载)

    .net序列化及反序列化 转载自:http://www.cnblogs.com/Tim_Liu/archive/2010/11/09/1872587.html 序列化是指一个对象的实例可以被保存,保存 ...

  5. vue中使用axios发送请求

    我们知道,vue2.0以后,vue就不再对vue-resource进行更新,而是推荐axios,而大型项目都会使用 Vuex 来管理数据,所以这篇博客将结合两者来发送请求 1.安装axios cnpm ...

  6. InnoSetup使用笔记

    今天用InnoSetup做安装包时,因为要装的驱动区分32位.64位,64位系统中要安装32位+64位驱动. 想在脚本中进行判断.折腾一阵,终于搞定: 参考了:http://379910987.blo ...

  7. 【工作】to-do-list

    当你不确定的时候,你就把你所在的工作做好,所在的你不愿意的行业做好,所谓的自由选择,它本身不自由的,不自由过程当中,如何你把它做好,就做人生的一个经历,人生的一个积累.-- 王石 TODO

  8. 如何加快MyEclipse的启动速度

    学习java开发的朋友对Myeclipse应该不陌生,MyEclipse企业级工作平台(MyEclipseEnterprise Workbench ,简称MyEclipse)是对EclipseIDE的 ...

  9. Intent详解以及实例

    Android中统一用Intent来封装程序的“调用意图“.不管程序想启动一个Activity,一个Servicer,还是一个BroadcastReceiver.使用Intent提供了一个统一的编程模 ...

  10. 关于altera的fft核使用问题记录

    altera的fft核使用比较特别,今天我做了一下仿真,发现一些问题,现做记录如下: 1,ip配置 parameters选项卡主要是fft变换的长度和数据长度,旋转因子长度,需要注意的是“Twiddl ...