由于线程的执行是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. Android Broadcast 和 BroadcastReceiver的权限机制

    在Android应用开发中,有时会遇到以下两种情况, 1. 一些敏感的广播并不想让第三方的应用收到 : 2. 要限制自己的Receiver接收某广播来源,避免被恶意的同样的ACTION的广播所干扰. ...

  2. js 扩展实例

    //扩展实例1 字符串中首字符转大写 var test=' this is '; String.prototype.mytrim=function(){ var re=/^\s+(.*?)\s+$/; ...

  3. 【Python系统学习】基础篇

    这次真的是最后一次了!第三次滚Python的基础.走了太多弯路.认真一点!菜鸟! 教程 转义字符 \ 可以转义很多字符,比如\n表示换行,\t表示制表符,字符\本身也要转义,所以\\表示的字符就是\ ...

  4. HDU - 3949 :XOR(线性基,所有集合的不同异或和中,求从小到大第K个)

    XOR is a kind of bit operator, we define that as follow: for two binary base number A and B, let C=A ...

  5. Protobuf-net 应用

    什么是ProtoBuf-net Protobuf是google开源的一个项目,是基于二进制的类似于XML,JSON这样的数据表示语言,用户数据序列化反序列化,google声称google的数据通信都是 ...

  6. 使用.NET Remoting开发分布式应用——基于租约的生存期

    一.概述 知名类型的SingleCall对象可以在客户程序的方法调用之后被垃圾收集器清理掉,因为它没有保持状态,属于无状态的.而客户激活的类型的对象和知名类型的SingleTon对象都属于生存期长的对 ...

  7. easyui datagrid 单元格加进度条(亲测可用)

    {field: 'DataItemNum', title: '数据完整度', width: 100, formatter: function (v, r, i) { var p = (v / 27) ...

  8. 集群/分布式环境下,Session处理策略

    前言 在搭建完集群环境后,不得不考虑的一个问题就是用户访问产生的session如何处理.如果不做任何处理的话,用户将出现频繁登录的现象.比如集中中存在A.B两台服务器,用户在第一次访问网站是,Ngin ...

  9. Linux安装微信

    地址:http://www.toutiao.com/i6362126617556288001/#6649976-tsina-1-90079-4471e2b057b5019ad452c722f04bba ...

  10. Goclipse on Eclipse

    Goclipse插件安装地址 https://goclipse.github.io/releases gocode.godef都可以通过Download快速安装 但guru容易失败:package g ...