JAVA之旅(十三)——线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this
JAVA之旅(十三)——线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this
我们继续上个篇幅接着讲线程的知识点
一.线程的安全性
当我们开启四个窗口(线程)把票陆陆续续的卖完了之后,我们要反思一下,这里面有没有安全隐患呢?在实际情况中,这种事情我们是必须要去考虑安全问题的,那我们模拟一下错误
package com.lgl.hellojava;
import javax.security.auth.callback.TextInputCallback;
//公共的 类 类名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 需求:简单的卖票程序,多个线程同时卖票
*/
MyThread myThread = new MyThread();
Thread t1 = new Thread(myThread);
Thread t2 = new Thread(myThread);
Thread t3 = new Thread(myThread);
Thread t4 = new Thread(myThread);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
/**
* 卖票程序
*
* @author LGL
*
*/
class MyThread implements Runnable {
// 票数
private int tick = 100;
@Override
public void run() {
while (true) {
if (tick > 0) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("卖票:" + tick--);
}
}
}
}
我们输出的结果
这里出现了0票,如果你继续跟踪的话,你会发现,还会出现-1,-2之类的票,这就是安全隐患,那原因是什么呢?
- 当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一个部分,还没有执行完,另外一个线程参与了执行,导致共享数据的错误
解决办法:对多条操作共享数据的语句,只能让一个线程都执行完再执行过程中其他线程不可以参与运行
JAVA对多线程的安全问题提供了专业的解决办法,就是同步代码块
synchronized(对象){
//需要同步的代码
}
那我们怎么用呢?
package com.lgl.hellojava;
//公共的 类 类名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 需求:简单的卖票程序,多个线程同时卖票
*/
MyThread myThread = new MyThread();
Thread t1 = new Thread(myThread);
Thread t2 = new Thread(myThread);
Thread t3 = new Thread(myThread);
Thread t4 = new Thread(myThread);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
/**
* 卖票程序
*
* @author LGL
*
*/
class MyThread implements Runnable {
// 票数
private int tick = 100;
Object oj = new Object();
@Override
public void run() {
while (true) {
synchronized(oj){
if (tick > 0) {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("卖票:" + tick--);
}
}
}
}
}
这样,就输出
二.多线程同步代码块
我们为什么可以这样去同步线程?
对象如同锁,持有锁的线程可以在同步中执行,没有执行锁的线程即使获取了CPU的执行权,也进不去,因为没有获取锁,我们可以这样理解
- 四个线程,哪一个进去就开始执行,其他的拿不到执行权,所以即使拿到了执行权,也进不去,这个同步能解决线程的安全问题
但是,同步是有前提的
- 1.必须要有两个或者两个以上的线程,不然你同步也没必要呀
- 2.必须是多个线程使用同一锁
必须保证同步中只能有一个线程在运行
但是他也有一个弊端:那就是多个线程都需要判断锁,较为消耗资源
三.多线成同步函数
我们可以写一段小程序,来验证这个线程同步的问题,也就是说我们看看下面这段程序是否有安全问题,有的话,如何解决?
package com.lgl.hellojava;
//公共的 类 类名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 需求:银行里有一个金库 有两个人要存钱300
*/
MyThread myThread = new MyThread();
Thread t1 = new Thread(myThread);
Thread t2 = new Thread(myThread);
t1.start();
t2.start();
}
}
/**
* 存钱程序,一次100
* @author LGL
*
*/
class MyThread implements Runnable {
private Bank b = new Bank();
@Override
public void run() {
for (int i = 0; i < 3; i++) {
b.add(100);
}
}
}
/**
* 银行
* @author LGL
*
*/
class Bank {
private int sum;
public void add(int n) {
sum = sum + n;
System.out.println("sum:" + sum);
}
}
当你执行的时候你会发现
这里是没错的,存了600块钱,但是,这个程序是有安全隐患的
如何找到问题?
- 1.明确哪些代码是多线成运行代码
- 2.明确共享数据
- 3.明确多线成运行代码中哪些语句是操作共享数据的
那我们怎么找到安全隐患呢?我们去银行的类里面做些认为操作
/**
* 银行
* @author LGL
*
*/
class Bank {
private int sum;
public void add(int n) {
sum = sum + n;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("sum:" + sum);
}
}
让他sleep一下你就会发现
这样的话,我们就可以使用我们的同步代码了
/**
* 银行
*
* @author LGL
*
*/
class Bank {
private int sum;
Object j = new Object();
public void add(int n) {
synchronized (j) {
sum = sum + n;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("sum:" + sum);
}
}
}
这样代码就可以同步了
哪些代码该同步,哪些不该同步,你一定要搞清楚,根据上面的3个条件
大家有没有注意到,函数式具有封装代码的特定,而我们所操作的同步代码块也是有封装代码的特性,拿这样的话我们就可以换一种形式去操作,那就是写成函数的修饰符
/**
* 银行
*
* @author LGL
*
*/
class Bank {
private int sum;
public synchronized void add(int n) {
sum = sum + n;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("sum:" + sum);
}
}
这样也是OK的
四.同步函数的锁是this
既然我们学习了另一种同步函数的写法,那我们就可以把刚才的买票小例子进一步封装一下了
package com.lgl.hellojava;
//公共的 类 类名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 需求:简单的卖票程序,多个线程同时卖票
*/
MyThread myThread = new MyThread();
Thread t1 = new Thread(myThread);
Thread t2 = new Thread(myThread);
Thread t3 = new Thread(myThread);
Thread t4 = new Thread(myThread);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
/**
* 卖票程序
*
* @author LGL
*
*/
class MyThread implements Runnable {
// 票数
private int tick = 100;
@Override
public synchronized void run() {
while (true) {
if (tick > 0) {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread()+"卖票:" + tick--);
}
}
}
}
但是这样做,你却会发现一个很严重的问题,那就是
永远只有0线程在执行卖票
那是因为我们并没有搞清楚需要同步哪一个代码段,我们应该执行的只是里面的那两段代码,而不是整个死循环,所以我们得封装个函数进行线程同步
package com.lgl.hellojava;
//公共的 类 类名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 需求:简单的卖票程序,多个线程同时卖票
*/
MyThread myThread = new MyThread();
Thread t1 = new Thread(myThread);
Thread t2 = new Thread(myThread);
Thread t3 = new Thread(myThread);
Thread t4 = new Thread(myThread);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
/**
* 卖票程序
*
* @author LGL
*
*/
class MyThread implements Runnable {
// 票数
private int tick = 100;
@Override
public void run() {
while (true) {
show();
}
}
private synchronized void show() {
if (tick > 0) {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread() + "卖票:" + tick--);
}
}
}
这样输出解决了
问题是被解决了,但是随之问题也就来了
- 同步函数用的是哪一个锁呢?
函数需要被对象调用,那么函数都有一个所属对象的引用,就是this,所以同步函数所引用的锁是this,我们来验证一下,我们把程序改动一下
使用两个线程来卖票,一个线程在同步代码块中,一个线程在同步函数中,都在执行卖票动作
package com.lgl.hellojava;
//公共的 类 类名
public class HelloJJAVA {
public static void main(String[] args) {
/**
* 需求:简单的卖票程序,多个线程同时卖票
*/
MyThread myThread = new MyThread();
Thread t1 = new Thread(myThread);
Thread t2 = new Thread(myThread);
// Thread t3 = new Thread(myThread);
// Thread t4 = new Thread(myThread);
t1.start();
myThread.flag = false;
t2.start();
// t3.start();
// t4.start();
}
}
/**
* 卖票程序
*
* @author LGL
*
*/
class MyThread implements Runnable {
// 票数
private int tick = 100;
Object j = new Object();
boolean flag = true;
@Override
public void run() {
if (flag) {
while (true) {
synchronized (j) {
if (tick > 0) {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread() + "code:"
+ tick--);
}
}
}
} else {
while (true) {
show();
}
}
}
private synchronized void show() {
if (tick > 0) {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread() + "show:" + tick--);
}
}
}
当我们运行的时候就发现
他只在show中进行,那是为什么呢?因为主线程开启的时候瞬间执行,我们要修改一下,让线程1开启的时候,主线程睡个10毫秒试试
t1.start();
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
myThread.flag = false;
t2.start();
这样输出的结果貌似是交替进行
但是所知而来的,是0票,这说明这个线程不安全,我们明明加了同步啊,怎么还是不安全呢?因为他用的不是同一个锁,一个用Object,一个是用this的锁,我们再改动一下,我们把Object更好为this,这样输出
现在就安全,也正确了
好的,我们本篇幅就先到这里了,我们下篇也继续讲线程
如果有兴趣,可以加入群:555974449
JAVA之旅(十三)——线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this的更多相关文章
- 线程执行synchronized同步代码块时再次重入该锁过程中抛异常,是否会释放锁
一个线程执行synchronized同步代码时,再次重入该锁过程中,如果抛出异常,会释放锁吗? 如果锁的计数器为1,抛出异常,会直接释放锁: 那如果锁的计数器为2,抛出异常,会直接释放锁吗? 来简单测 ...
- java高并发系列 - 第10天:线程安全和synchronized关键字
这是并发系列第10篇文章. 什么是线程安全? 当多个线程去访问同一个类(对象或方法)的时候,该类都能表现出正常的行为(与自己预想的结果一致),那我们就可以所这个类是线程安全的. 看一段代码: pack ...
- Java多线程-线程的同步(同步代码块)
对于同步,除了同步方法外,还可以使用同步代码块,有时候同步代码块会带来比同步方法更好的效果. 追其同步的根本的目的,是控制竞争资源的正确的访问,因此只要在访问竞争资源的时候保证同一时刻只能一个线程访问 ...
- 彻底理解线程同步与同步代码块synchronized
public class Demo { public static synchronized void fun1(){ } public synchronized void fun2(){ } pub ...
- Java精通并发-同步方法访问标志与synchronized关键字之间的关系
继续基于上一次https://www.cnblogs.com/webor2006/p/11428811.html来研究synchronized关键字在字节码中的表现,在上一次文末提出了一个这样的问题: ...
- 对象及变量的并发访问(同步方法、同步代码块、对class进行加锁、线程死锁)&内部类的基本用法
主要学习多线程的并发访问,也就是使得线程安全. 同步的单词为synchronized,异步的单词为asynchronized 同步主要就是通过锁的方式实现,一种就是隐式锁,另一种是显示锁Lock,本节 ...
- Java进阶(四十三)线程与进程的区别
Java进阶(四十三)线程与进程的区别 1.线程的基本概念 概念:线程是进程中执行运算的最小单位,是进程中的一个实体,是被系统独立调度和分派的基本单位,线程自己不拥有系统资源,只拥有一点在运行中必 ...
- java线程学习之synchronized关键字
关键字synchronized的作用是实现线程间的同步.它的任务是对同步的代码加锁.一个代码块同时只能有同一个线程进行读和写操作,从而保证线程间是安全的. 线程安全的概念是:当多个线程访问某一个类(对 ...
- java 线程 (三)线程并发的安全性 同步代码块
package cn.sasa.demo1; import java.util.concurrent.ExecutionException; public class ThreadDemo { pub ...
随机推荐
- PHP Filter 函数
PHP Filter 简介 PHP 过滤器用于对来自非安全来源的数据(比如用户输入)进行验证和过滤. 安装 Filter 函数是 PHP 核心的组成部分.无需安装即可使用这些函数. PHP Filte ...
- android自定义View之3D索引效果
效果图: 我的小霸王太卡了. 最近工作比较忙,今天搞了一下午才搞出来这个效果,这种效果有很多种实现方式,最常见的应该是用贝塞尔曲线实现的.今天我们来看另一种不同的实现方式,只需要用到 canvas.s ...
- Android简易实战教程--第三十八话《自定义通知NotifiCation》
上一篇小案例,完成了一个普通的通知,点击通知启动了一个活动.但是那里的通知没有加入些"靓点",这一篇就给它加入自定义的布局,完成自定义的通知. 应用:比如QQ音乐为例,当点击音乐播 ...
- 如何获得Android手机的软件安装列表
Android的PackageManager类用于检索目前安装在设备上的应用软件包的信息.你可以通过调用getpackagemanager()得到PackageManager类的一个实例.对查询和操作 ...
- Gradle 1.12用户指南翻译——第五十二章. Maven 插件
本文由CSDN博客貌似掉线翻译,其他章节的翻译请参见:http://blog.csdn.net/column/details/gradle-translation.html翻译项目请关注Github上 ...
- Android Multimedia框架总结(二十三)MediaCodec补充及MediaMuxer引入(附案例)
请尊重分享成果,转载请注明出处,本文来自逆流的鱼yuiop,原文链接:http://blog.csdn.net/hejjunlin/article/details/53729575 前言:前面几章都是 ...
- Android基于JsBridge封装的高效带加载进度的WebView
Tamic http://blog.csdn.net/sk719887916/article/details/52402470 概述 从去年4月项目就一直用起了JsBridge,前面也针对jsBrid ...
- C语言如何在两个文件中访问同一个全局变量
方法一: 不使用头文件. 1.c 中 int var; 2.c 中 extern int var; 方法二: 使用头文件. 1.c 中 int var; 不必添加#include "1.h& ...
- SpriteKit给游戏弹跳角色添加一个高度标示器
这是一个类似于跳跃涂鸦的小游戏,主角不断吃能量球得到跳跃能量向更高的地方跳跃,如果图中碰到黑洞就挂了- 在游戏调试过程中如果能实时知道主角的高度就好了,这将有助于程序猿动态的判断游戏胜败逻辑. 你可以 ...
- Android源码浅析(四)——我在Android开发中常用到的adb命令,Linux命令,源码编译命令
Android源码浅析(四)--我在Android开发中常用到的adb命令,Linux命令,源码编译命令 我自己平时开发的时候积累的一些命令,希望对你有所帮助 adb是什么?: adb的全称为Andr ...