JAVA学习第二十六课(多线程(五))- 多线程间的通信问题
一、线程间的通信
实例代码:
需求是:输入一个姓名和性别后,就输出一个姓名和性别
class Resource
{
String name;
String sex ;
}
class Input implements Runnable
{
Resource r;
Input(Resource r)
{
this.r = r;
}
public void run()
{
int x = 0;
while(true)
{
synchronized (r)
{
if(x==0)
{
r.name = "BLF";
r.sex = "male";
}
else {
r.name = "妮妮妮妮";
r.sex = "female";
}
x = (++x)%2;
}
}
}
} class Output implements Runnable
{
Resource r;
public Output(Resource r) {
// TODO Auto-generated constructor stub
this.r = r;
}
public void run()
{
while(true)
{
synchronized (r) //输入和输出都是应用同一个锁
{
System.out.println(r.name+"..."+r.sex);
}
}
}
}
public class Main
{
public static void main(String[] args)
{
Resource r = new Resource();//共享同一资源
Input in = new Input(r);
Output out = new Output(r);
Thread t1 = new Thread(in);
Thread t2 = new Thread(out);
t1.start();
t2.start();
}
}
上述代码尽管攻克了。多线程处理同一资源问题,可是出现了一个问题就是,打印非常多一个人名和性别后再打印还有一个,无法实现交替输出。原因:输入线程获取运行权后,不会运行一次,输入姓名和性别,输入线程还拥有运行权。一直赋值。等到切换到输出运行权后,输出最后输入的姓名和性别。可是输出线程也不会仅仅输出一次,一直输出,所以出现打印非常多同一姓名和性别的问题
二、线程的等待/唤醒机制:
用一个boolean值推断是否有数据,有就不放。没有就放
输入线程运行时推断是否有数据,有。就释放运行权,再释放运行资格,进入冻结状态。输出线程运行。输出后boolean值置为false
Input:
if(flag)//有值
wait();//当前冻结,切换输出线程
flag = true;
notify(); Output:
if(!flag)
wait();
flag = false;//flag置为假。输出完成
notify();//唤醒输入线程
等待/唤醒机制:
方法:
1.wait();//使线程进入冻结状态,CPU释放运行权和运行资格,被wait()的线程会被存储到线程池中
2.notify();//唤醒线程池中的随意的一个线程
3.notifyAll();//唤醒全部线程,使之处于运行状态或暂时堵塞状态,总之,使其具备运行资格
这些方法必须定义在同步中。这些方法是用于操作线程状态的方法,所以必须明白在哪个锁上的线程
(wait()A锁的线程。仅仅能用A锁的notify唤醒A锁的线程。
)
了解:而wait()等方法是定义在Object类中的。这些方法是监视器的方法。而监视器呢。就能够理解为锁,控制哪个锁所属下的线程,而锁又能够是随意的,而全部的类都是继承于Object类的,所以wait()等类是定义在Object里
下述代码就攻克了上述代码的问题,实现了输入输出线程的交替运行
class Resource extends Object
{
String name;
String sex ;
boolean flag = false;
}
class Input implements Runnable
{
Resource r;
Input(Resource r)
{
this.r = r;
}
public void run()
{
int x = 0;
while(true)
{
synchronized (r)
{
if(r.flag)
{
try {
r.wait();
} catch (Exception e) {
// TODO: handle exception
} } if(x==0)
{
r.name = "BLF";
r.sex = "male";
}
else {
r.name = "妮妮妮妮";
r.sex = "female";
}
r.flag = true;
r.notify();
x = (++x)%2;
}
}
}
} class Output implements Runnable
{
Resource r;
public Output(Resource r) {
// TODO Auto-generated constructor stub
this.r = r;
}
public void run()
{ while(true)
{
synchronized (r)
{
if(!r.flag)
{
try {
r.wait();
} catch (Exception e) {
// TODO: handle exception
}
} System.out.println(r.name+"..."+r.sex);
r.flag = false;
r.notify();
}
}
}
}
public class Main
{
public static void main(String[] args)
{ Resource r = new Resource();
Input in = new Input(r);
Output out = new Output(r);
Thread t1 = new Thread(in);
Thread t2 = new Thread(out);
t1.start();
t2.start();
}
}
三:等待唤醒机制代码优化
资源中的成员为了可控,应该为私有的。对外提供方法
所以同步的操作。就在Resource类中运行,应用同步函数
class Resource extends Object
{
private String name;
private String sex ;
private boolean flag = false;
public synchronized void set(String name,String sex)
{
// TODO Auto-generated constructor stub
if(flag)
{
try {
this.wait();
} catch (Exception e) {
// TODO: handle exception }
} this.name = name;
this.sex = sex; flag = true;
this.notify();
}
public synchronized void out()
{
if(!flag)
{
try {
this.wait();
} catch (Exception e) {
// TODO: handle exception
}
System.out.println(name+" : "+sex);
flag = false;
this.notify();
}
}
}
class Input implements Runnable
{
Resource r;
Input(Resource r)
{
this.r = r;
}
public void run()
{
int x = 0;
while(true)
{
if(x==0)
{
r.set("BLF", "male");
}
else
{
r.set("妮妮妮妮", "female");
}
x = (++x)%2; }
}
}
class Output implements Runnable
{
Resource r;
public Output(Resource r) {
// TODO Auto-generated constructor stub
this.r = r;
}
public void run()
{
while(true)
{
r.out();
}
}
}
public class Main
{
public static void main(String[] args)
{ Resource r = new Resource();
Input in = new Input(r);
Output out = new Output(r);
Thread t1 = new Thread(in);
Thread t2 = new Thread(out);
t1.start();
t2.start();
}
}
JAVA学习第二十六课(多线程(五))- 多线程间的通信问题的更多相关文章
- JAVA学习第二十九课(经常使用对象API)- String类
多线程告一段落,開始经常使用对象API的涉及,背也要背下来!.! 日后开发,遇见最多的对象是文字,也就是字符串 String类 字符串是一个特殊对象 字符串一旦初始化就不能够被改变 一.特点 publ ...
- Python学习第二十六课——PyMySql(python 链接数据库)
Python 链接数据库: 需要先安装pymysql 包 可以设置中安装,也可以pip install pymysql 安装 加载驱动: import pymysql # 需要先安装pymysql 包 ...
- 风炫安全WEB安全学习第二十六节课 XSS常见绕过防御技巧
风炫安全WEB安全学习第二十六节课 XSS常见绕过防御技巧 XSS绕过-过滤-编码 核心思想 后台过滤了特殊字符,比如说
- 201671010140. 2016-2017-2 《Java程序设计》java学习第十六周
java学习第十六周-并发 本周,学习了Java中线程,并发的知识,在老师的带领下,进行了对知识的理解学习,以及对实验的运行讲解,对这一块内容掌握的还可以,在自主编程中,也能够完成.线, ...
- NeHe OpenGL教程 第二十六课:反射
转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线 ...
- 风炫安全web安全学习第二十八节课 CSRF攻击原理
风炫安全web安全学习第二十八节课 CSRF攻击原理 CSRF 简介 跨站请求伪造 (Cross-Site Request Forgery, CSRF),也被称为 One Click Attack 或 ...
- 风炫安全WEB安全学习第二十三节课 利用XSS获取COOKIE
风炫安全WEB安全学习第二十三节课 利用XSS获取COOKIE XSS如何利用 获取COOKIE 我们使用pikachu写的pkxss后台 使用方法: <img src="http:/ ...
- 风炫安全WEB安全学习第二十节课 反射型XSS讲解
风炫安全WEB安全学习第二十节课 反射型XSS讲解 反射性xss演示 原理讲解 如果一个应用程序使用动态页面向用户显示错误消息,就会造成一种常见的XSS漏洞.通常,该页面会使用一个包含消息文本的参数, ...
- Java学习第二十四天
1:多线程(理解) (1)JDK5以后的针对线程的锁定操作和释放操作 Lock锁 (2)死锁问题的描述和代码体现 (3)生产者和消费者多线程体现(线程间通信问题) 以学生作为资源来实现的 资源类:St ...
随机推荐
- Python标准库——collections模块的Counter类
1.collections模块 collections模块自Python 2.4版本开始被引入,包含了dict.set.list.tuple以外的一些特殊的容器类型,分别是: OrderedDict类 ...
- Ubuntu服务器安装node
查看Ubuntu系统的是32位还是64位 harvey@harvey:/Application$ uname -m #x86_64表示这是64位的系统 x86_64 2. 在nodejs官网ht ...
- js实现的联想输入
以前也写过一个jQuery的这种插件,但是很多地方根本不用jQuery,这个功能也有很多其它库支持,但是为了用这个功能而加载很多js插件,这样效率明显下降了很多,而且这个东西平时也很常用,所以一决心自 ...
- javascript三种嵌入方式
什么是JavaScript? JavaScript是运行在浏览器端的脚步语言,JavaScript主要解决的是前端与用户交互的问题,包括使用交互与数据交互,JavaScript是浏览器解释执行的. J ...
- css深入理解之border
1. border-width border-width不支持百分比,类似的还有outline,box-shadow,text-shadow等 border-width支持关键字:thin(1px, ...
- 如何php防止XSS攻击
什么是XSS:这里通俗的讲,就像是SQL注入一样,XSS攻击也可以算是对HTML和JS的一种注入.你本来希望得到是从用户那得到一段有用的文本文字,但用户提交给你的却是别有用心的可执行javascrip ...
- int 与 String 与 char 之间的互相转换
int 转 String: //方式一: int i1 = 888; String s1 = Integer.toString(i1); //方式二: int i2 = 888; String s2 ...
- HDU 6319.Problem A. Ascending Rating-经典滑窗问题求最大值以及COUNT-单调队列 (2018 Multi-University Training Contest 3 1001)
2018 Multi-University Training Contest 3 6319.Problem A. Ascending Rating 题意就是给你长度为k的数列,如果数列长度k<n ...
- 51nod 最长公共子序列问题(动态规划)(LCS)(递归)
最长公共子序列问题 输入 第1行:字符串A 第2行:字符串B (A,B的长度 <= 1000) 输出 输出最长的子序列,如果有多个,随意输出1个. 输入示例 abcicba abdkscab 输 ...
- 2、Django实战第2天:app设计
app设计:五大模块 users 用户管理 courses 课程管理 organization 机构和教师管理 operation 用户操作管理 新建上面4个模块的app manage.py@mxon ...