Java多线程(三)

news2024/11/23 11:38:38

目录

一、线程的同步(二)

同步机制释放锁的操作

不会释放锁的操作

线程的死锁问题

死锁

解决方法

Lock(锁)

使用Lock(锁)创建多线程步骤:

使用Lock解决窗口售票问题

synchronized与Lock的对比

练习

 二、线程的通信

通过例题说明线程间的通信

 wait()与notify()与notifyAll()

wait()

notify()

notifyAll()

说明

sleep()和wait()的异同

练习题

一、线程的同步(二)

同步机制释放锁的操作

1、当前线程的同步方法、同步代码块执行结束。

2、当前线程在同步代码块、同步方法中遇到break、return终止了该代码块、 该方法的继续执行。 3、当前线程在同步代码块、同步方法中出现了未处理的Error或Exception,导致异常结束。

4、当前线程在同步代码块、同步方法中执行了线程对象的wait()方法,当前线程暂停,并释放锁。

不会释放锁的操作

1、线程执行同步代码块或同步方法时,程序调用Thread.sleep()、 Thread.yield()方法暂停当前线程的执行

2、线程执行同步代码块时,其他线程调用了该线程的suspend()方法将该线程挂起,该线程不会释放锁(同步监视器)。应尽量避免使用suspend()和resume()来控制线程

线程的死锁问题

死锁

1、不同的线程分别占用对方需要的同步资源不放弃,都在等待对方放弃 自己需要的同步资源,就形成了线程的死锁

2、出现死锁后,不会出现异常,不会出现提示,只是所有的线程都处于 阻塞状态,无法继续

解决方法

1、专门的算法、原则

2、尽量减少同步资源的定义

3、尽量避免嵌套同步

Lock(锁)

从JDK 5.0开始,Java提供了更强大的线程同步机制——通过显式定义同 步锁对象来实现同步。同步锁使用Lock对象充当。

 java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的 工具。锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前应先获得Lock对象。

ReentrantLock 类实现了 Lock ,它拥有与 synchronized 相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock,可以显式加锁、释放锁。

使用Lock(锁)创建多线程步骤:

1)实例化ReentrantLock

2)调用锁定方法:lock()

3)调用解锁方法:unlock()

try{

        xxx.lock();

        //保证线程安全的代码

}finally{

        xxx.unlock();

}

使用Lock解决窗口售票问题

import java.util.concurrent.locks.ReentrantLock;


public class LockTest {
    public static void main(String[] args) {
        Window w=new Window();
        Thread t1=new Thread(w);
        Thread t2=new Thread(w);
        Thread t3=new Thread(w);
        t1.setName("窗口1");
        t2.setName("窗口2");
        t3.setName("窗口3");
        t1.start();
        t2.start();
        t3.start();
    }
}

class Window implements Runnable{
    private int ticket=100;
    //1.实例化ReentrantLock
    private ReentrantLock lock=new ReentrantLock();
    @Override
    public void run() {
        while(true) {

            try {
                lock.lock();
                if (ticket > 0) {
                    System.out.println(Thread.currentThread().getName() + ":,卖票,票号为:" + ticket);
                    ticket--;
                } else {
                    break;
                }
            }finally {
                lock.unlock();
            }
        }
    }
}

synchronized与Lock的对比

相同点:二者都可以解决线程安全问题

不同点:

  • Lock是显示锁,需手动开启和关闭锁;synchronized是隐式所,出了作用域自动释放
  • Lock只有代码块锁;synchronized有代码块锁和方法锁
  • 使用Lock锁,JVM将花费较少的时间来调度线程,性能更好。并且具有更好的扩展性(提供更所得子类)

练习

银行有一个账户。 有两个储户分别向同一个账户存3000元,每次存1000,存3次。每次存完打印账户余额。

/**
 * @auther light
 * @Description 银行有一个账户。
 * 有两个储户分别向同一个账户存3000元,每次存1000,存3次。每次存完打
 * 印账户余额
 *
 * 分析:
 * 1.是否是多线程问题? 是:两个储户线程
 * 2.是否有共享数据?  有:账户(或账户余额)
 * 3.是否有线程安全问题? 有
 * 4.需要考虑如何解决线程安全问题? 同步机制:三种方式
 * @create 2022-11-19 15:14
 */
public class AccountTest {
    public static void main(String[] args) {
        Account acct=new Account(0);
        Customer c1=new Customer(acct);
        Customer c2=new Customer(acct);
        c1.setName("甲");
        c2.setName("乙");
        c1.start();
        c2.start();
    }
}

class Account{
    double balance;
    public Account( double balance){
        this.balance=balance;

    }
    //存钱
    public synchronized void  deposit(double amt){
        if(amt>0){
            balance+=amt;
//            try {
//                Thread.sleep(1000);
//            } catch (InterruptedException e) {
//                throw new RuntimeException(e);
//            }
            System.out.println(Thread.currentThread().getName()+":存钱成功,余额为:"+balance);
        }

    }
}

class Customer extends Thread{
    Account acct;
    public Customer(Account acct){
        this.acct=acct;
    }

    @Override
    public void run() {
        for(int i=0;i<3;i++){
            acct.deposit(1000);
        }
    }
}

运行结果如下:

 二、线程的通信

操作同一个资源时多个线程之间不断切换执行时所发出的信号

通过例题说明线程间的通信

线程通信例子:使用各个线程交替打印1-100

* 涉及到三个方法:
* 1.wait():一旦执行此方法,当前线程就进入阻塞状态,并释放同步监视器
* 2.notify():一旦执行此方法,就会唤醒被wait的一个线程,如果有多个线程被wait,就唤醒优先级高的那个
* 3.notifyAll():一旦执行此方法就会唤醒所有被wait的线程
*
* 说明:
* 1.wait()、notify()、notifyAll()这三个方法只能使用在同步代码块或同步方法中
* 2.wait()、notify()、notifyAll()这三个方法的调用者必须是同步代码块或同步方法中的同步监视器。
* 3.wait()、notify()、notifyAll()这三个方法是定义在java.lang.Object类中的

 

public class ThreadCommunicationTest {
    public static void main(String[] args) {
        Number n=new Number();
        Thread n1=new Thread(n);
        Thread n2=new Thread(n);
        n1.setName("线程1");
        n2.setName("线程2");
        n1.start();
        n2.start();

    }
}

class Number implements Runnable{
    private int i=1;
    @Override
    public void run() {
        synchronized (this) {
                for (; i <= 100; ) {
                    notify();
                    try {
                        Thread.sleep(10);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                    System.out.println(Thread.currentThread().getName() + ":打印输出:" + i);
                    i++;
                    try {
                        wait();
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
    }
}

部分运行结果为:

 wait()与notify()与notifyAll()

wait()

令当前线程挂起并放弃CPU、同步资源并等待,使别的线程可访问并修改共享资源,而当前线程排队等候其他线程调用notify()或notifyAll()方法唤醒,唤醒后等待重新获得对监视器的所有权后才能继续执行。

notify()

唤醒正在排队等待同步资源的线程中优先级最高者结束等待

notifyAll()

唤醒正在排队等待资源的所有线程结束等待.

说明

  • wait()、notify()、notifyAll()这三个方法只能使用在同步代码块或同步方法中
  • .wait()、notify()、notifyAll()这三个方法的调用者必须是同步代码块或同步方法中的同步监视器。
  • wait()、notify()、notifyAll()这三个方法是定义在java.lang.Object类中的

sleep()和wait()的异同

相同点:一旦执行方法,都可以使得当前线程进入阻塞状态

不同点:

  • 两个方法声明的位置不同:sleep()声明在Thread类中;wait()声明在Object类中
  • 调用的范围和要求不同:sleep()可以生命在任何需要的场景下;wait()只能声明在同步代码块或同步方法中
  • 关于是否释放同步监视器:如果两个方法必须使用在同步代码块或同步方法中,sleep()不会释放同步监视器;wait()会释放同步监视器 

练习题

生产者(Productor)将产品交给店员(Clerk),而消费者(Customer)从店员处 取走产品,店员一次只能持有固定数量的产品(比如:20),如果生产者试图 生产更多的产品,店员会叫生产者停一下,如果店中有空位放产品了再通 知生产者继续生产;如果店中没有产品了,店员会告诉消费者等一下,如 果店中有产品了再通知消费者来取走产品。

问题分析:

* 1、是否是多线程问题:是,生产者线程,消费者线程

* 2、是否有共享数据:是,店员(或产品)

* 3、如何解决多线程问题:同步机制,三种方法

* 4、是否涉及线程通信:是

public class ProductTest {
    public static void main(String[] args) {
        Clerk c=new Clerk();
        Producer p1=new Producer(c);
        Customer c1=new Customer(c);
        Thread t1=new Thread(p1);
        Thread t2=new Thread(c1);
        t1.setName("生产者1");
        t2.setName("消费者1");
        t1.start();
        t2.start();


    }
}

class Clerk {
    private int product=0;
    public  synchronized void produceProduct() {
        //生产者生产产品

            if(product<20){
                product++;
                System.out.println(Thread.currentThread().getName()+":开始生产第 "+product+" 个产品");
                notify();
            }else{
                try {
                    wait();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }



    }


    public  synchronized void consumeProduct() {
       //消费者消费产品

            if(product>0){
                System.out.println(Thread.currentThread().getName()+":开始消费第 "+product+" 个产品");
                product--;
                notify();
            }else{
                try {
                    wait();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }


    }
}

class Producer implements Runnable{
    Clerk clerk;
    public Producer(Clerk clerk){
        this.clerk=clerk;
    }
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName()+"开始生产产品...");
        while(true){
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            clerk.produceProduct();
        }

    }
}

class Customer implements Runnable{
    Clerk clerk;
    public Customer(Clerk clerk){
        this.clerk=clerk;
    }
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName()+"开始消费产品...");
        while(true){
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            clerk.consumeProduct();
        }

    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/25871.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

优化器-SQL语句分析与优化

一、连接-配置优化 1.1 连接数过多问题 有时会碰到Mysql&#xff1a;error 1040&#xff1a;Too many connection的错误。原因&#xff1a;超过了服务端设置的最大并发连接数。 1.2 从两个方面解决问题 服务端&#xff0c;增加服务端可用连接数&#xff1b;客户端&#xff0…

如何在视频上添加水印?建议收藏这些方法

小伙伴们平时会刷短视频吗&#xff1f;那你们会不会自己也在平台上&#xff0c;发布一些自己剪辑的短视频呢&#xff1f;在网上发布的视频&#xff0c;很容易被一些不安好心的人&#xff0c;直接窃取&#xff0c;所以为视频添加自己的水印显得尤为重要。那你们知道如何给视频加…

sqli-labs/Less-49

欢迎界面还是以sort作为注入点 首先判断属于数字型还是字符型 输入如下 sortrand() 页面从没有变化 说明属于字符型 然后输入1 发现没有报错信息 不能使用报错注入 只能通过结果去反映处你的注入是对是错 首先输入1-- 成功回显 说明注入类型就是属于单引号字符型 然后接…

Towards Class-Oriented Poisoning Attacks Against Neural Networks 论文笔记

#论文笔记# 1. 论文信息 论文名称Towards Class-Oriented Poisoning Attacks Against Neural Networks作者Bingyin Zhao会议/出版社WACV 2022pdf&#x1f4c4;在线pdf代码无 基于类别的 availability attacks&#xff0c;不同于原本的 availability attacks 只考虑降低模型的…

VMware虚拟机中的Linux通过NAT模式共享主机网卡实现与外部设备通信

目前遇到的使用场景&#xff1a; 需要VMware虚拟机中linux通过PC端的物理网卡与外界其他设备或PC进行通信&#xff0c;因此需要配置虚拟机中linux的通信链路。 1.设置PC端IP网络 如果要实现虚拟机被局域网其它机子访问到&#xff0c;那么这里我们要选择桥架模式&#xff0c;具…

Oracle实验五Sql语句

每一句插入都要带commit提交&#xff0c;不然会出现很多报错 直接从WPS实验报告里复制过来的&#xff0c;可能有中英文标点问题 实验目的 &#xff08;1&#xff09; 掌握数据的插入&#xff08;INSERT&#xff09;、 修改&#xff08;UPDATE&#xff09; 和删除&#xff08;D…

现代密码学导论-1-导论

目录 1.1 密码学和现代密码学 1.2 私钥加密 1.2.1 私钥加密的两个广泛应用 1.2.2 加密的语法 1.2.3 柯克霍夫原则 1.1 密码学和现代密码学 经典密码学(20世纪80年代以前)和现代密码学之间的另一个非常重要的区别与它的采用有关。历史上&#xff0c;密码学的主要消费者是军…

Android App开发实战项目之电子书架的实现(附源码 简单易懂 可直接使用)

需要图片集和源码请点赞关注收藏后评论区留言~~~ 一、需求描述 在手机上浏览电子书的浏览体验跟阅读纸质书差不多&#xff0c;翻页过程仍旧呈现纸张翻转的视觉特效&#xff0c;让读者看起来赏心悦目。总结一下&#xff0c;手机阅读无非是要具有两大功能点&#xff1a;其一为书…

【GlobalMapper精品教程】025:影像数据集的建立与巧妙使用

GlobalMapper影像数据集类似于金字塔,作用是提高大量影像的加载与显示速度,还可批量进行一系列设置。本文的配套数据为data025.rar。 文章目录 1. 建立影像数据集2. 影像数据集的使用1. 建立影像数据集 (1)点击【文件】→【创建新地图目录】。 (2)选择影像数据集存放路径…

Doker学习笔记1(狂神)

虚拟机技术缺点&#xff1a; 1.资源占用十分多 2.冗余步骤多 3.启动很慢&#xff01; 容器化技术 我们去安装docker&#xff1a; 我们先保证我们的虚拟机是可以使用的。 环境查看&#xff1a; 系统内核是3.10以上的 系统版本&#xff1a; 我们用的是centOS7虚拟机。 然后…

微服务学习笔记(二)

文章目录Spring Cloud Eureka1.Spring Cloud Eureka 简介2.Spring Cloud Eureka 和 Zookeeper 的区别2.1 什么是 CAP 原则&#xff08;面试&#xff09;2.2 分布式特征3.Spring Cloud 其他注册中心3.1 Consul3.2 Nacos4.Spring Cloud Eureka 快速入门4.1 搭建 Eureka-server4.1…

【ELM回归预测】探路者优化极限学习机回归预测【含Matlab源码 2231期】

⛄一、探路者算法简介 提出的一种新兴的智能优化算法&#xff0c;该算法的思想起源于群体动物的狩猎行为&#xff0c;种群中的个体分为探路者和跟随者两种角色。算法的寻优过程模拟了种群寻找食物的探索过程&#xff0c;利用探路者、跟随者两种角色不同的位置更新方式以及角色…

蓝牙血压计PCBA硬件解决方案

蓝牙血压计是利用现代电子技术与血压间接测量原理进行血压测量的医疗设备。家庭医疗保健已成为现代人的医疗保健时尚。过去人们测量血压必须到医院才行&#xff0c;而今只要拥有了蓝牙血压计&#xff0c;坐在家里便可随时监测血压的变化&#xff0c;如发现血压异常便可及时去医…

odoo14 | odoo中domain的复杂写法

本片文章主要讲述domain中复杂业务需求的逻辑构思&#xff0c;关于doamin的使用位置会在另一篇文章中体现。 二叉树与波兰式 在讲述domain的使用前先讲解一下科班生必学的《数据结构与算法》中二叉树遍历与波兰式的内容&#xff0c;如果你会二叉树先序遍历与波兰式转换请直接…

从零开始的深度学习之旅(2)

目录深层神经网络1. 异或门问题1.1 异或代码实现2.神经网络的层2.1 去除激活函数的异或门2.2 使用sigmoid函数的异或门3.从0实现深度神经网络的正向传播深层神经网络 1. 异或门问题 在第一篇的博客中,我们使用代码实现了与门 import torch X torch.tensor([[1,0,0],[1,1,0]…

通过实战总结的 使用GoFrame小技巧

文章目录gf gen dao设置参数可传可不传model作为结构体类型模型关联添加数据主程序如下&#xff1a;gomeGoods.MainImgs的定义&#xff1a;批量插入数据配置插件自动生成service总结一起学习有朋友问我能不能搞一个GoFrame技巧篇&#xff0c;让新手少踩坑的那种。今天他来了&am…

数据库-范式例题

目录 1、请简述满足1NF、2NF和3NF的基本条件。并完成下题:某信息一览表如下:其是否满足3NF,若不满足将其化为符合3NF的关系。 解:     1NF: 属性都是不可分割的数据项 2NF: 不存在部分函数依赖,存在传递函数依赖。 2NF: 不存在

算法复杂度分析中的渐近分析(基于输入大小)

为什么要进行性能分析&#xff1f; 有许多重要的事情需要注意&#xff0c;例如用户友好性、模块化、安全性、可维护性等。为什么要担心性能&#xff1f;答案很简单&#xff0c;只有当我们有性能时&#xff0c;我们才能拥有上述所有东西。因此&#xff0c;性能就像货币&#xf…

磷酸化多肽Asp-Arg-Val-Tyr(PO3H2)-Ile-His-Pro-Phe、129785-85-9

在血管平滑肌细胞中有多种作用&#xff0c;包括正常动脉的收缩&#xff0c;培养的细胞或病变血管的肥大或增生等。 编号: 200676 中文名称: 八肽DRV-pTyr-IHPF CAS号: 129785-85-9 单字母: H2N-DRV-pTyr-IHPF-OH 三字母: H2N-Asp-Arg-Val-Tyr(PO3H2)-Ile-His-Pro-Phe-COOH 氨基…

MySQL——进阶

第1章 存储引擎 1.1 MySQL体系结构 1.2 存储引擎介绍 定义&#xff1a;存储引擎就是存储数据、建立索引、更新/查询数据等技术的实现方式 。存储引擎是基于表的&#xff0c;而不是基于库的&#xff0c;所以存储引擎也可被称为表类型。我们可以在创建表的时候&#xff0c;来指定…