xyan

导航

Java第十次作业

(一)学习总结
1.用思维导图对java多线程的学习内容进行总结。
参考资料: XMind。

2.下面是一个单线程实现的龟兔赛跑游戏。

public class TortoiseHareRace {
    public static void main(String[] args) {
        int totalStep = 10;
        int tortoiseStep = 0;
        int hareStep = 0;
        boolean[] flags = {true,false};
        System.out.println("龟兔赛跑开始了...");
        while(tortoiseStep < totalStep && hareStep < totalStep){
            tortoiseStep++;
            System.out.println("乌龟跑了"+tortoiseStep+"步...");
            boolean isHareSleep = flags[((int)(Math.random()*10))%2];
            if(isHareSleep){
                System.out.println("兔子睡着了zzzz");
            }else{
                hareStep += 2;
                System.out.println("兔子跑了"+hareStep+"步...");
            }
        }       
    }
}

阅读程序,采用实现Runnable接口的方式用多线程实现这个小游戏。下面给出主线程类,补充Tortoise(乌龟)线程类和Hare线程类。

public class TortoiseHareRace { 
    public static void main(String[] args) {
        Tortoise tortoise = new Tortoise(10);
        Hare hare = new Hare(10);
        Thread tortoiseThread = new Thread(tortoise);
        Thread hareThread = new Thread(hare);
System.out.println("龟兔赛跑开始了...");
        tortoiseThread.start();
        hareThread.start();
    }
}

Tortoise(乌龟)线程类:

public class Tortoise implements Runnable {
	public Tortoise(int i) {
		this.totalStep=i;
	}
	public int totalStep;
	public int tortoiseStep = 0;
    boolean[] flags = {true,false};
	public void run() {
		while(tortoiseStep < totalStep){
            tortoiseStep++;
            System.out.println("乌龟跑了"+tortoiseStep+"步...");
		}
	}
}

Hare线程类:

public class Hare implements Runnable{
	public Hare(int i) {
		this.totalStep=i;
	}
	public int totalStep;
	public int hareStep = 0;
    boolean[] flags = {true,false};
	public void run() {
		while(hareStep < totalStep){
			boolean isHareSleep = flags[((int)(Math.random()*10))%2];
			if(isHareSleep){
                System.out.println("兔子睡着了zzzz");
            }else{
                hareStep += 2;
                System.out.println("兔子跑了"+hareStep+"步...");
            }
		}
		
	}
}

3.下面的程序是模拟了生产者——消费者问题,生产者生产10个数,消费者依次消费10个数,运行程序,看结果是否正常?存在什么问题?说明原因。使用synchronized, wait, notify解决程序出现的问题。写出修改的部分程序即可。

class Consumer implements Runnable {
    private Clerk clerk;
    public Consumer(Clerk clerk) {
        this.clerk = clerk;
    }
    public void run() {
        System.out.println("消费者开始消耗整数......");
        // 消耗10个整数
        for(int i = 1; i <= 10; i++) {
            try {
                 // 等待随机时间
                Thread.sleep((int) (Math.random() * 3000));
            }
            catch(InterruptedException e) {
                e.printStackTrace();
            }              
            clerk.getProduct();// 从店员处取走整数
        }
    }
 }class Producer implements Runnable {
    private Clerk clerk;
    public Producer(Clerk clerk) {
        this.clerk = clerk;
    }
    public void run() {
        System.out.println( "生产者开始生产整数......");
        // 生产1到10的整数
        for(int product = 1; product <= 10; product++) {
            try {
                Thread.sleep((int) Math.random() * 3000);
            }
            catch(InterruptedException e) {
                e.printStackTrace();
            }
           clerk.setProduct(product); // 将产品交给店员
        }
    } 
}public class ProductTest {
    public static void main(String[] args) {
        Clerk clerk = new Clerk();
        Thread consumerThread = new Thread(new Consumer(clerk)); 
        Thread producerThread = new Thread(new Producer(clerk)); 
        consumerThread.start(); 
        producerThread.start(); 
    }
}class Clerk {
    private int product = -1; // -1 表示目前没有产品 
     // 这个方法由生产者呼叫
    public void setProduct(int product) {
        this.product = product; 
        System.out.printf("生产者设定 (%d)%n", this.product);      
    } 
    // 这个方法由消费者呼叫
    public int getProduct() {          
        int p = this.product; 
        System.out.printf("消费者取走 (%d)%n", this.product);      
        return p; 
    } 
}

运行结果是:先是生产者生产了从一到十,然后是消费者连续取走十个十。结果不正常,存在的问题是线程没有等待,也就没有了线程的交替运行。
修改后的程序:

class Clerk {
	private boolean flag=true;  //起始状态先生产后消费
	//flag = true  可以生产不能取走     flag = false 可以取走不能生成
    private int product = -1; // -1 表示目前没有产品 
     // 这个方法由生产者呼叫
    public synchronized void setProduct(int product) {
    	while(this.flag == false){
    		try {
    			wait();
            } catch (InterruptedException e) {  e.printStackTrace();   }
        }
    	if(this.flag == true){
    		this.product = product; 
            System.out.printf("生产者设定 (%d)%n", this.product);  
    	}
        this.flag = false;
        notify();
            
    } 
    // 这个方法由消费者呼叫
    public synchronized int getProduct() {          
        int p = this.product; 
        while(this.flag == true){
            try {
                wait();
            } catch (InterruptedException e) {   e.printStackTrace();  }
        }
        if(this.flag == false){
        	System.out.printf("消费者取走 (%d)%n", this.product); 
        }
        this.flag = true;
        notify();
        return p;
    } 
}

4.其他需要总结的内容。
(二)实验总结
实验内容:
1.模拟三个老师同时分发80分作业,每个老师相当于一个线程。
程序设计思路:写一个发卷子的线程类,和一个主线程类,实现的是Runnable接口,用关键字synchronized 来实现三位老师的线程同步进行。
类图结构:

2.模拟一个银行存款的程序。假设有两个储户都去银行往同一个账户进行存款,一次存100,每人存三次。要求储户每存一次钱,账户余额增加100,并在控制台输出当前账户的余额。
完成实验内容,代码上传到码云,对完成实验内容过程中遇到的问题、解决方案和思考等进行归纳总结,注意代码中必须有必要的注释。

程序设计思路:建一个银行类给该用户进行账户初始化,建一个存储的线程类,还有主线程类,通过for进行控制线程运行几次。
类图结构:

问题1:无法控制多线程中的一个线程的运行次数。
原因:线程是随机的执行。
解决方案:

        for(i=0;i<3;i++){
			Chuhu mt=new Chuhu();
			new Thread(mt,"第1名储户存储完,").start();
		}
		for(i=0;i<3;i++){
			Chuhu mt=new Chuhu();
			new Thread(mt,"第2名储户存储完,").start();
		}

(三)代码托管(务必链接到你的项目)
https://git.oschina.net/hebau_cs15/java-cs01lxy.git

码云commit历史截图
上传实验项目代码到码云,在码云项目中选择“统计-commits”,设置搜索时间段,搜索本周提交历史,并截图。

posted on 2017-05-30 18:35  xyan  阅读(96)  评论(0编辑  收藏  举报