线程安全

1.定义

多线程访问共享数据,会产生线程安全问题。

2.代码模拟

卖票Ticked类:

package com.lanyue.day22;

public class Person {

    public static void main(String[] args) {

        Car one = new Car("宝马");

        one.start();

        Driver two = new Driver();
new Thread(two).start(); new Thread(new Runnable() {
@Override
public void run() {
for(int i = 0;i < 5;i++){ try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("第二个司机第" + i + "次报数中。");
}
}
}).start();
}
}

卖票窗口Window类

package com.lanyue.day23;

public class Windows {

    public static void main(String[] args) {

        Ticked runnable = new Ticked();
Thread one = new Thread(runnable);
Thread two = new Thread(runnable);
Thread three = new Thread(runnable); one.start();
two.start();
three.start();
}
}

程序执行图

3.解决方法

当使用多个线程对同一个资源有写操作时,就容易出现线程安全问题。为解决这个问题,Java提供了同步机制synchronized来解决这个问题。

那么如何实现所谓的同步机制呢?有三个方法:

1.同步代码块

synchronized用于方法中的某个区块中,表示只对这个区块的资源实行互斥访问。

示例:

synchronized (同步锁){
//代码块.

注意:

(1)同步代码块中的锁对象可以是任意对象。

(2)必须保证多个线程使用的锁对象是同一个对象。

示例代码:

package com.lanyue.day23;

public class Ticked implements Runnable {

    public Integer ticks = 100;
public Object obj = new Object();
public boolean state = true; @Override
public void run() { while(state){ synchronized (obj){ if(ticks-- > 0){ System.out.println(Thread.currentThread().getName() + "窗口卖出了第" + (100 - ticks) + "张票"); }else{ state = false;
}
}
}
}
}

2.同步方法

(1)普通同步锁方法:

只需要在方法前加一个修饰符 synchronized 即可(本质上也是利用锁对象锁定的,这个锁是this)。

格式:

权限修饰符 synchronized 返回值类型 方法名(参数){

方//法体

示例代码:

package com.lanyue.day23;

public class TickedTwo implements Runnable{

    public Integer ticks = 100;
public boolean state = true; @Override
public void run(){ while(state){ view();
}
} public synchronized void view(){ if(ticks-- > 0){ System.out.println(Thread.currentThread().getName() + "窗口卖出了第" + (100 - ticks) + "张票"); }else{ state = false;
}
}
}

(3)静态同步方法

本质也是利用锁对象,这个锁对象是class文件对象

格式:

权限修饰符 static synchronized 返回值类型 方法名(参数){

//方法体。

示例代码:

package com.lanyue.day23;

public class TickedThree implements Runnable{

    public static Integer ticks = 100;
public static boolean state = true; @Override
public void run() { while(state){ view();
}
} public static synchronized void view(){ if(ticks-- > 0){ System.out.println(Thread.currentThread().getName() + "窗口卖出了第" + (100 - ticks) + "张票"); }else{ state = false;
}
}
}

3.锁机制

Lock接口

ReentrantLock类是Lock接口的实现类。

使用步骤:

(1)在成员变量位置创建 ReentrantLock 对象。

(2)在可能发生线程安全问题的代码前调用ReentrantLock 对象的lock()方法锁住共享数据。

(3)在可能发生线程安全问题的代码后调用ReentrantLock  对象的unlock()方法解锁共享资源。

代码示例:

package com.lanyue.day23;

import java.util.concurrent.locks.ReentrantLock;

public class TickedFour implements Runnable {

    public ReentrantLock myLock = new ReentrantLock();
public Integer ticks = 100;
public boolean state = true; @Override
public void run() { while(state){ myLock.lock();
if(ticks-- > 0){ System.out.println(Thread.currentThread().getName() + "窗口卖出了第" + (100 - ticks) + "张票"); }else{ state = false;
} myLock.unlock();
}
}
}

Java第二十四天,线程安全的更多相关文章

  1. “全栈2019”Java第二十四章:流程控制语句中决策语句switch下篇

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  2. 学JAVA第二十四天,Set集合与StringBuilder

    下面的内容需要慢慢看,因为,我的语言表达能力不是很好 首先说Set把,Set集合是一个无序且不允许重复的集合,而且查找效率也是快的可怕的. 但是,有些时候,我们必须要用储存多个相同的值时,Set也是可 ...

  3. “全栈2019”Java多线程第二十四章:等待唤醒机制详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  4. javaSE第二十四天

    第二十四天    363 1:多线程(理解)    363 (1)JDK5以后的Lock锁    363 A:定义    363 B:方法:    364 C:具体应用(以售票程序为例)    364 ...

  5. Gradle 1.12用户指南翻译——第二十四章. Groovy 插件

    其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...

  6. Java 1.ExecutorService四种线程池的例子与说明

    1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? new Thread(new Runnable() { @Override public void run() { ...

  7. NeHe OpenGL教程 第二十四课:扩展

    转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线 ...

  8. Python第二十四天 binascii模块

    Python第二十四天 binascii模块 binascii用来进行进制和字符串之间的转换 import binascii s = 'abcde' h = binascii.b2a_hex(s) # ...

  9. SQL注入之Sqli-labs系列第二十四关(二阶注入)

    开始挑战第二十四关(Second Degree Injections) 0x1 前言 SQL注入一般分为两类:一阶SQL注入(普通SQL注入),二阶SQL注入 .二次注入不是注入两次的意思,请不要混淆 ...

随机推荐

  1. Linux启动nginx时报错nginx: [emerg] getpwnam("nginx") failed

    编译时指定了用户而没有创建用户导致报错 解决: 查看你添加的用户是什么, [root@localhost nginx]# sbin/nginx -Vnginx version: nginx/1.10. ...

  2. juery 弹出框

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  3. js 任意值变化封装

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  4. What is the difference between shades and shadows?

    Shade is the darkness of an object not in direct light, while shadows are the silhouette of an objec ...

  5. oracle去除重复数据与oracle分页

    一.去除oracle中重复数据,可以使用rowid列,rowid列是一个伪列,该列在数据库中灭一个表中都有,但是我们查询数据库的时候,默认都没有给我们返回这一列,这一列用来区分数据库中的每一行时间,可 ...

  6. Python3之turtle的基本用法#Python学习01#

    一.turtle基本语法 1.导入turtle 模块import turtle 2.显示箭头turtle.showturtle() 3.写字符串turtle.write("因小米" ...

  7. CodeForces 196B Infinite Maze

    Infinite Maze time limit per test 2 seconds memory limit per test 256 megabytes input standard input ...

  8. JUC 中提供的限流利器-Semaphore(信号量)

    在 JUC 包下,有一个 Semaphore 类,翻译成信号量,Semaphore(信号量)是用来控制同时访问特定资源的线程数量,它通过协调各个线程,以保证合理的使用公共资源.Semaphore 跟锁 ...

  9. TensorFlow-Bitcoin-Robot:一个基于 TensorFlow LSTM 模型的 Bitcoin 价格预测机器人。

    简介 TensorFlow-Bitcoin-Robot:一个基于 TensorFlow LSTM 模型的 Bitcoin 价格预测机器人. 文章包括一下几个部分: 1.为什么要尝试做这个项目? 2.为 ...

  10. 面试指南」JS 模块化、组件化、工程化相关的 15 道面试题

    JS 模块化.组件化.工程化相关的 15 道面试题 1.什么是模块化? 2.简述模块化的发展历程? 3.AMD.CMD.CommonJS 与 ES6 模块化的区别? 4.它们是如何使用的? 5.exp ...