1. 分别设计一个Account类,和两个任务类 Company和Bank: (1)Account类拥有账户余额属性和余额增减操作;(2)Company任务类能增加Account对象中的余额;(3)Bank任务类能减少(或使用)Account对象中的余额(比如用于基金投资)。要求使用同一account对象来分别创建两个Company线程对象和两个Bank线程对象,通过两次测试来完成以下两任务(实验报告中 截两个图对应以下两个任务):
1) 出现Account对象余额不一致问题(同步问题);
2)使用同步代码块或同步方法或Lock对象来解决同步问题。
package com.join; class Bank implements Runnable { private Account account; Bank(Account account) { this.account = account; } @Override public void run() { for (int i = 0; i != 5; ++i) { account.decreaseBalance(10); } } } class Company implements Runnable { private Account account; Company(Account account) { this.account = account; } @Override public void run() { for (int i = 0; i != 5; ++i) { account.increaseBalance(10); } } } public class Account { private double balance; private Account(double balance) { this.balance = balance; } void increaseBalance(double increment) { double b = balance; b = balance + increment; balance = b; } void decreaseBalance(double decrement) { double b = balance; b = balance - decrement; balance = b; } private synchronized double getBalance() { return balance; } public static void main(String[] args) { Account account = new Account(100); Company company = new Company(account); Bank bank = new Bank(account); Thread t1 = new Thread(company); Thread t2 = new Thread(company); Thread t3 = new Thread(bank); Thread t4 = new Thread(bank); // 或者给方法加锁 try { t1.start(); t1.join(); t2.start(); t2.join(); t3.start(); t3.join(); t4.start(); t4.join(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("最终余额为:" + account.getBalance()); } }
2. 设计一个生产电脑和搬运电脑类,要求生产一台电脑就搬走一台电脑,如果没有新的电脑生产出来,则搬运工要等待新电脑产出;如果生产出的电脑没有搬走,则要等待电脑搬走之后在生产,并统计出生产的电脑数量。
package com.my.test; class Producer extends Thread { int count = 0; boolean hasNewComputer = false; public Producer(String name) { super(name); } private synchronized void produceComputer() throws InterruptedException { while (hasNewComputer) { wait(); } Thread.sleep(3000); hasNewComputer = true; System.out.println(Thread.currentThread().getName() + ":生产第" + ++count + "台电脑"); notify(); } public synchronized void consumeComputer(int count) throws InterruptedException { while (!hasNewComputer) { wait(); } Thread.sleep(3000); hasNewComputer = false; System.out.println(Thread.currentThread().getName() + ":第" + count + "台电脑被消费"); notify(); } @Override public void run() { while (true) { try { produceComputer(); } catch (InterruptedException e) { e.printStackTrace(); } } } } class Consumer extends Thread { int count = 0; Producer producer; public Consumer(String name, Producer producer) { super(name); this.producer = producer; } @Override public void run() { while (true) { try { producer.consumeComputer(++count); } catch (InterruptedException e) { e.printStackTrace(); } } } } public class Main { public static void main(String[] args) { Producer producer = new Producer("producer"); Consumer consumer = new Consumer("consumer", producer); producer.start(); consumer.start(); } }