最好不适用线程的子线程,直接调用线程,然后用rannable接口

然后如果要公用一个参数,就是公用资源的时候,一定要在run方法的前面加上synchronized

例子

猫和狗喝同一杯水(共用资源的问题)

/**
* @author nienie
*
*/ public class test { /**
* @param args
*/
public static void main(String[] args) {
// TODO 自动生成的方法存根
test1 test2 = new test1();
int n=10;
test2.setW(n);
System.out.println("一共" +n +"克水,dog和cat每次喝1克水");
Thread dog,cat;
dog = new Thread(test2);
cat = new Thread(test2);
dog.setName("dog");
cat.setName("cat");
dog.start();
cat.start();
} }
public class test1 implements Runnable{
int w = 10;
public void setW(int n){
w = n;
}
public synchronized void run() {
String name = Thread.currentThread().getName();
while(name.equals("dog")){
w = w-1;
System.out.println(name+" 喝水,剩下 "+w+ " 克"); notify();
try {
wait();
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
if(w==0) System.exit(0);
}
while(name.equals("cat")){
w = w-1;
System.out.println(name+" 喝水,剩下 "+w + " 克");
notify();
try {
wait();
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
if(w==0) System.exit(0);
}
}
}

运行结果

一共10克水,dog和cat每次喝1克水
dog 喝水,剩下 9 克
cat 喝水,剩下 8 克
dog 喝水,剩下 7 克
cat 喝水,剩下 6 克
dog 喝水,剩下 5 克
cat 喝水,剩下 4 克
dog 喝水,剩下 3 克
cat 喝水,剩下 2 克
dog 喝水,剩下 1 克
cat 喝水,剩下 0 克

这里要注意的是一定要有那个synchronized,这个的作用相当于一个锁,锁定某一个线程先做好,但是如果直接使用,有一个矛盾,就是无法实现两个线程交替地使用,这个会导致等这个线程结束了之后,然后再执行第二个线程,哪怕有sleep也不行的,所以这里就灵活的使用wait()方法,这个函数会将这个线程停止,然后让出cpu,让cpu执行其他线程,但是有一点要注意的是,在执行了wait之后,要使用notify方法激活另外的线程,不然的话,就会导致全部的线程都停止,然后程序就停下来了。

没有synchronized的情况

public class test1 implements Runnable{
int w = 10;
public void setW(int n){
w = n;
}
public void run() {
String name = Thread.currentThread().getName();
while(name.equals("dog")){
w = w-1;
System.out.println(name+" 喝水,剩下 "+w+ " 克");
notify();
try {
Thread.sleep(1000);
wait();
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
if(w==0) System.exit(0);
}
while(name.equals("cat")){
w = w-1;
System.out.println(name+" 喝水,剩下 "+w + " 克");
notify();
try {
Thread.sleep(1000);
wait();
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
if(w==0) System.exit(0);
}
}
} 运行结果:
一共10克水,dog和cat每次喝1克水
dog 喝水,剩下 9 克
Exception in thread "dog" cat 喝水,剩下 8 克
Exception in thread "cat" java.lang.IllegalMonitorStateException
at java.lang.Object.notify(Native Method)
at test1.run(test1.java:29)
at java.lang.Thread.run(Unknown Source)
java.lang.IllegalMonitorStateException
at java.lang.Object.notify(Native Method)
at test1.run(test1.java:16)
at java.lang.Thread.run(Unknown Source)

## 书本上面没有使用synchronized

/**
* @author nienie
*
*/ public class Example12_2 { /**
* @param args
*/
public static void main(String[] args) {
// TODO 自动生成的方法存根
House house = new House();
house.serWater(10);
Thread dog,cat;
dog = new Thread(house);
cat = new Thread(house);
dog.setName("dog");
cat.setName("cat");
dog.start();
cat.start();
} }
public class House implements Runnable{
int waterAmount;
public void serWater(int w){
waterAmount = w;
}
public void run() {
int m = 1;
while(true){
if(waterAmount <=0){
return;
}
waterAmount = waterAmount -m;
System.out.println("剩 "+waterAmount+" 克");
try{
Thread.sleep(2000);
}
catch (Exception e) {}
}
}
}
运行结果:
剩 9 克
剩 8 克
剩 7 克
剩 7 克
剩 6 克
剩 6 克
剩 5 克
剩 5 克
剩 4 克
剩 4 克
剩 3 克
剩 3 克
剩 2 克
剩 2 克
剩 0 克
剩 0 克

明显的出现了很多重复的,所以这里就要用到synchronized

书本上使用synchronized的另外一个例子

/**
* @author nienie
*
*/ public class Example12_5 { /**
* @param args
*/
public static void main(String[] args) {
// TODO 自动生成的方法存根
Bank bank = new Bank();
bank.setMoney(200);
Thread accountant,cashier;
accountant = new Thread(bank);
cashier = new Thread(bank);
accountant.setName("会计");
cashier.setName("出纳");
accountant.start();
cashier.start();
}
} public class Bank implements Runnable{
int money = 200;
public void setMoney(int n){
money = n;
}
public void run() {
// TODO 自动生成的方法存根
if(Thread.currentThread().getName().equals("会计"))
saveOtTake(300);
else if(Thread.currentThread().getName().equals("出纳"))
saveOtTake(150);
}
public synchronized void saveOtTake(int amount) {
// TODO 自动生成的方法存根
if(Thread.currentThread().getName().equals("会计")){
for (int i=1;i<3;i++){
money = money + amount/3;
System.out.println(Thread.currentThread().getName()+" 存入 "+amount/3+" ,账上有 "+money+" 万,休息一会再存");
try{
Thread.sleep(1000);
}
catch (InterruptedException e) {}
}
}
else if(Thread.currentThread().getName().equals("出纳")){
for (int i=1;i<3;i++){
money = money - amount/3;
System.out.println(Thread.currentThread().getName()+" 取出 "+amount/3+" ,账上有 "+money+" 万,休息一会再取");
try{
Thread.sleep(1000);
}
catch (InterruptedException e) {}
}
}
}
} 运行结果:
会计 存入 100 ,账上有 300 万,休息一会再存
会计 存入 100 ,账上有 400 万,休息一会再存
会计 存入 100 ,账上有 500 万,休息一会再存
出纳 取出 50 ,账上有 450 万,休息一会再取
出纳 取出 50 ,账上有 400 万,休息一会再取
出纳 取出 50 ,账上有 350 万,休息一会再取

这里也是共用一个资源,但是有一个问题就是,这里的是要会计一直工作完,然后出纳才可以做,就是无论里面有多少钱,只要会计还没有将钱存完,出纳就无法拿钱,哪怕里面的钱足够,这样子就起不到两个线程交替运行了,所以这里还要修改一下,使会计和出纳交替运行,只要里面够钱,那出纳就可以用,然后拿了之后,会计继续存钱(就是在会计休息的时候,出纳可以拿钱,出纳休息的时候,会计也可以存钱,然后对那个money也有一个锁,不会出现资源混乱。)

对Bank Class进行修改

public synchronized void saveOtTake(int amount) {
// TODO 自动生成的方法存根
if(Thread.currentThread().getName().equals("会计")){
for (int i=1;i<=3;i++){
money = money + amount/3;
System.out.println(Thread.currentThread().getName()+" 存入 "+amount/3+" ,账上有 "+money+" 万,休息一会再存");
notify();
try{
Thread.sleep(1000);
wait();
}
catch (InterruptedException e) {}
}
}
else if(Thread.currentThread().getName().equals("出纳")){
for (int i=1;i<=3;i++){
money = money - amount/3;
System.out.println(Thread.currentThread().getName()+" 取出 "+amount/3+" ,账上有 "+money+" 万,休息一会再取");
notify();
try{
Thread.sleep(1000);
wait();
}
catch (InterruptedException e) {}
}
}
System.exit(0);
}
运行结果:
会计 存入 100 ,账上有 300 万,休息一会再存
出纳 取出 50 ,账上有 250 万,休息一会再取
会计 存入 100 ,账上有 350 万,休息一会再存
出纳 取出 50 ,账上有 300 万,休息一会再取
会计 存入 100 ,账上有 400 万,休息一会再存
出纳 取出 50 ,账上有 350 万,休息一会再取

对java的Thread的理解的更多相关文章

  1. Java线程Thread的状态解析以及状态转换分析 多线程中篇(七)

    线程与操作系统中线程(进程)的概念同根同源,尽管千差万别. 操作系统中有状态以及状态的切换,Java线程中照样也有. State 在Thread类中有内部类 枚举State,用于抽象描述Java线程的 ...

  2. 转载:java中Thread.sleep()函数使用

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  3. Java类加载机制的理解

    算上大学,尽管接触Java已经有4年时间并对基本的API算得上熟练应用,但是依旧觉得自己对于Java的特性依然是一知半解.要成为优秀的Java开发人员,需要深入了解Java平台的工作方式,其中类加载机 ...

  4. 沉淀再出发:关于java中的AQS理解

    沉淀再出发:关于java中的AQS理解 一.前言 在java中有很多锁结构都继承自AQS(AbstractQueuedSynchronizer)这个抽象类如果我们仔细了解可以发现AQS的作用是非常大的 ...

  5. 多线程系列之 java多线程的个人理解(二)

    前言:上一篇多线程系列之 java多线程的个人理解(一) 讲到了线程.进程.多线程的基本概念,以及多线程在java中的基本实现方式,本篇主要接着上一篇继续讲述多线程在实际项目中的应用以及遇到的诸多问题 ...

  6. Java 内省(Introspector)深入理解

    Java 内省(Introspector)深入理解 一些概念: 内省(Introspector) 是Java 语言对 JavaBean 类属性.事件的一种缺省处理方法. JavaBean是一种特殊的类 ...

  7. Java并发--Thread类详情

    以下是本文的目录大纲: 一.线程的状态 二.上下文切换 三.Thread类中的方法 转载原文链接:http://www.cnblogs.com/dolphin0520/p/3920357.html 一 ...

  8. java.lang.Thread类详解

    java.lang.Thread类详解 一.前言 位于java.lang包下的Thread类是非常重要的线程类,它实现了Runnable接口,今天我们来学习一下Thread类,在学习Thread类之前 ...

  9. java之结合代码理解synchronized关键字

    为了保证数据的一致性即实现线程的安全性,java虚拟机提供了同步和锁机制.synchronized关键字是最基本的互斥同步手段.除此之外,还可以使用java.util.concurrent包中的重入锁 ...

随机推荐

  1. 用Markdown写微信公众号文章

    目前微信公众号的编辑器是不支持Markdown语法的,那怎么办呢? 有一款叫Markdown Here的插件可以解决这个问题(支持Chrome.Firefox.Safari). 官方网站:http:/ ...

  2. linux scp远程拷贝文件及文件夹

    [http://www.jb51.net/LINUXjishu/73131.html] 1.拷贝本机/home/administrator/test整个目录至远程主机192.168.1.100的/ro ...

  3. git添加本地仓库与远程仓库连接

    在本地建立一个文件夹,需要与远程git仓库进行连接,具体方法: <1>首先进入所在文件目录执行:  git init 初始化git,紧接着 git  add . git commit -m ...

  4. CSS预处理器之Less详解

    本文最初发表于博客园,并在GitHub上持续更新前端的系列文章.欢迎在GitHub上关注我,一起入门和进阶前端. 以下是正文. CSS 预处理器 为什么要有 CSS 预处理器 CSS基本上是设计师的工 ...

  5. 三、scrapy后续

    CrawlSpiders 通过下面的命令可以快速创建 CrawlSpider模板 的代码: scrapy genspider -t crawl tencent tencent.com 我们通过正则表达 ...

  6. MysqL碎片整理优化

    先来说一下什么是碎片,怎么知道碎片有多大! 简单的说,删除数据必然会在数据文件中造成不连续的空白空间,而当插入数据时,这些空白空间则会被利用起来.于是造成了数据的存储位置不连续,以及物理存储顺序与理论 ...

  7. 使用canvas编写时间轴插件

    使用canvas编写时间轴插件 背景 项目中有一个视频广场的功能,需要一个时间轴类似视频播放中进度条功能一样显示录像情况,并且可以点击.拖动.放大缩小展示时间轴,获取到时间轴的某个时间.原来的时间轴是 ...

  8. dubbox系列【一】——dubbox简介

    1.dubbox是什么? dubbox是当当网开源的开源分布式服务框架,基于阿里巴巴dubbo. 1个框架 + 2个方案:分布式服务框架 + RPC远程调用方案 + SOA服务治理方案. 2.dubb ...

  9. uva140

    全排列回溯剪枝. 题目数据很水.记录当前最小带宽,边回溯边计算当前序列最大的距离(也就是带宽),如果当前带宽超过了当前的最小带宽就剪枝. 注意下,数据读入时的字符串处理. AC代码 #include& ...

  10. 在SpringBoot中使用FluentValidator验证插件

    前言 在我们编写项目的时候,在controller中往往离不开对一些数据的校验.这里并不是说对于这些数据业务上面的校验,而是对这些数据进行空校验或者是长度校验等. 有些时候校验可以省略,根据业务的需要 ...