zoukankan      html  css  js  c++  java
  • 线程同步机制

    1.线程安全问题的概述

    ============================================================================================================== 

    2.线程安全问题的代码实现

    RunnableImpl.java

    package com.itheima.demo06.ThreadSafe;
    /*
    实现卖票案例
    */
    public class RunnableImpl implements Runnable{
    //定义一个多个线程共享的票源
    private int ticket = 100;


    //设置线程任务:卖票
    @Override
    public void run() {
    //使用死循环,让卖票操作重复执行
    while(true){
    //先判断票是否存在
    if(ticket>0){
    //提高安全问题出现的概率,让程序睡眠
    try {
    Thread.sleep(10);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }

    //票存在,卖票 ticket--
    System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
    ticket--;
    }
    }
    }
    }

    ---------------------------------------------------------------------------------------------------------------------------------------------------

    Demo01Ticket.java

    package com.itheima.demo06.ThreadSafe;
    /*
    模拟卖票案例
    创建3个线程,同时开启,对共享的票进行出售
    */
    public class Demo01Ticket {
    public static void main(String[] args) {
    //创建Runnable接口的实现类对象
    RunnableImpl run = new RunnableImpl();
    //创建Thread类对象,构造方法中传递Runnable接口的实现类对象
    Thread t0 = new Thread(run);
    Thread t1 = new Thread(run);
    Thread t2 = new Thread(run);
    //调用start方法开启多线程
    t0.start();
    t1.start();
    t2.start();
    }
    }

     ===========================================================================

    3.线程安全问题产生的原理

     ==================================================================================================================

    4.解决线程安全问题_同步代码块

     

     

     =======================================================================================================

    5.同步技术的原理

     =========================================================================================================

    6.解决线程安全问题_同步方法

     

     

     RunnableImpl.java

    package com.itheima.demo08.Synchronized;
    /*
    卖票案例出现了线程安全问题
    卖出了不存在的票和重复的票

    解决线程安全问题的二种方案:使用同步方法
    使用步骤:
    1.把访问了共享数据的代码抽取出来,放到一个方法中
    2.在方法上添加synchronized修饰符

    格式:定义方法的格式
    修饰符 synchronized 返回值类型 方法名(参数列表){
    可能会出现线程安全问题的代码(访问了共享数据的代码)
    }
    */
    public class RunnableImpl implements Runnable{
    //定义一个多个线程共享的票源
    private static int ticket = 100;


    //设置线程任务:卖票
    @Override
    public void run() {
    System.out.println("this:"+this);//this:com.itheima.demo08.Synchronized.RunnableImpl@58ceff1
    //使用死循环,让卖票操作重复执行
    while(true){
    payTicketStatic();
    }
    }

    /*
    静态的同步方法
    锁对象是谁?
    不能是this
    this是创建对象之后产生的,静态方法优先于对象
    静态方法的锁对象是本类的class属性-->class文件对象(反射)
    */
    public static /*synchronized*/ void payTicketStatic(){
    synchronized (RunnableImpl.class){
    //先判断票是否存在
    if(ticket>0){
    //提高安全问题出现的概率,让程序睡眠
    try {
    Thread.sleep(10);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }

    //票存在,卖票 ticket--
    System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
    ticket--;
    }
    }

    }

    /*
    定义一个同步方法
    同步方法也会把方法内部的代码锁住
    只让一个线程执行
    同步方法的锁对象是谁?
    就是实现类对象 new RunnableImpl()
    也是就是this
    */
    public /*synchronized*/ void payTicket(){
    synchronized (this){
    //先判断票是否存在
    if(ticket>0){
    //提高安全问题出现的概率,让程序睡眠
    try {
    Thread.sleep(10);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }

    //票存在,卖票 ticket--
    System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
    ticket--;
    }
    }

    }
    }

    -------------------------------------------------------------------------------------------------------------------------------------------------------

    Demo01Ticket.java

    package com.itheima.demo08.Synchronized;

    /*
    模拟卖票案例
    创建3个线程,同时开启,对共享的票进行出售
    */
    public class Demo01Ticket {
    public static void main(String[] args) {
    //创建Runnable接口的实现类对象
    RunnableImpl run = new RunnableImpl();
    System.out.println("run:"+run);//run:com.itheima.demo08.Synchronized.RunnableImpl@58ceff1
    //创建Thread类对象,构造方法中传递Runnable接口的实现类对象
    Thread t0 = new Thread(run);
    Thread t1 = new Thread(run);
    Thread t2 = new Thread(run);
    //调用start方法开启多线程
    t0.start();
    t1.start();
    t2.start();
    }
    }

    ========================================================================================================

    7.静态同步方法

     =================================================================================================================

    8.解决线程安全问题_Lock锁

     

     RunnableImpl.java

    package com.itheima.demo09.Lock;

    import java.util.concurrent.locks.Lock;
    import java.util.concurrent.locks.ReentrantLock;

    /*
    卖票案例出现了线程安全问题
    卖出了不存在的票和重复的票

    解决线程安全问题的三种方案:使用Lock锁
    java.util.concurrent.locks.Lock接口
    Lock 实现提供了比使用 synchronized 方法和语句可获得的更广泛的锁定操作。
    Lock接口中的方法:
    void lock()获取锁。
    void unlock() 释放锁。
    java.util.concurrent.locks.ReentrantLock implements Lock接口


    使用步骤:
    1.在成员位置创建一个ReentrantLock对象
    2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
    3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
    */
    public class RunnableImpl implements Runnable{
    //定义一个多个线程共享的票源
    private int ticket = 100;

    //1.在成员位置创建一个ReentrantLock对象
    Lock l = new ReentrantLock();

    //设置线程任务:卖票
    @Override
    public void run() {
    //使用死循环,让卖票操作重复执行
    while(true){
    //2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
    l.lock();

    //先判断票是否存在
    if(ticket>0){
    //提高安全问题出现的概率,让程序睡眠
    try {
    Thread.sleep(10);
    //票存在,卖票 ticket--
    System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
    ticket--;
    } catch (InterruptedException e) {
    e.printStackTrace();
    }finally {
    //3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
    l.unlock();//无论程序是否异常,都会把锁释放
    }
    }
    }
    }

    /*//设置线程任务:卖票
    @Override
    public void run() {
    //使用死循环,让卖票操作重复执行
    while(true){
    //2.在可能会出现安全问题的代码前调用Lock接口中的方法lock获取锁
    l.lock();

    //先判断票是否存在
    if(ticket>0){
    //提高安全问题出现的概率,让程序睡眠
    try {
    Thread.sleep(10);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }

    //票存在,卖票 ticket--
    System.out.println(Thread.currentThread().getName()+"-->正在卖第"+ticket+"张票");
    ticket--;
    }

    //3.在可能会出现安全问题的代码后调用Lock接口中的方法unlock释放锁
    l.unlock();
    }
    }*/
    }

  • 相关阅读:
    巧用系统命令看是否中毒
    从80年代初到2003年末中国出现的街头骗术
    万象数据库新密码世界```和原来的不同了
    换网关bat
    给自己的电脑做一个O盘 -隐藏自己私密的东
    一个小恶 搞 病 毒
    诊断卡常见代码
    路由跟踪命令的用法和技巧
    block,inline和inline-block的区别
    Jquery easyui开启行编辑模式增删改操作
  • 原文地址:https://www.cnblogs.com/curedfisher/p/12449105.html
Copyright © 2011-2022 走看看