1. 本章学习总结
2. 书面作业
Q1.互斥访问与同步访问
** 完成题集4-4(互斥访问)与4-5(同步访问) **
1.1 除了使用synchronized修饰方法实现互斥同步访问,还有什么办法实现互斥同步访问(请出现相关代码)?
1.2 同步代码块与同步方法有何区别?
1.3 实现互斥访问的原理是什么?请使用对象锁概念并结合相应的代码块进行说明。当程序执行synchronized同步代码块或者同步方法时,线程的状态是怎么变化的?
1.4 Java多线程中使用什么关键字实现线程之间的通信,进而实现线程的协同工作?为什么同步访问一般都要放到synchronized方法或者代码块中?
1.1 在Java1.5之后,可使用java.util.concurrent.locks包提供的操作方法替代synchronized机制。
测试代码:
public class Main {
static Lock lock=new ReentrantLock();
public static void main(String[] args) {
lock.lock(); //若锁处于空闲状态,则获得锁;若其他进程获得锁,则禁用当前进程
}
public static invoke()
{
lock.unlock(); //释放锁
}
}
Lock的出现其实是同步机制的升级版,由于运行到synchronized修饰的代码块之外时,进程便自动解除block状态,因此通过Lock的使用使用户可以更*控制锁的使用。
1.2
当一个进程获得了某个object的锁时,其他进程将无法访问该obj内的同步代码块,但可以访问非同步代码块。而同步方法则是限制了进程对于obj的访问。
1.3
为了实现互斥访问,在编程中引入了互斥锁的概念,每个对象都有一个互斥锁的标记,来保证在同一时刻只能有一个进程访问该对象。
测试代码:
class Account{
private int balance;
public Account(int balance) {
this.balance = balance;
}
synchronized void deposit(int money)
{
this.balance+=money;
}
synchronized void withdraw(int money)
{
this.balance-=money;
}
public int getBalance() {
return balance;
}
}
将实现类Account中的俩方法都上锁,就意味着当某个进程调用方法时,若其他进程也调用到该方法,那么便进入block状态,当方法锁释放时重新进入running状态。
1.4
使用wait()、notify()可以用来实现线程之间的通信协作。使用synchronized关键字的原因便是防止多个线程同时访问同一对象导致的冲突。
Q2.交替执行
实验总结(不管有没有做出来)
Q3.互斥访问
3.1 修改TestUnSynchronizedThread.java源代码使其可以同步访问。(关键代码截图,需出现学号)
3.2 进一步使用执行器改进相应代码(关键代码截图,需出现学号)
参考资料:Java多线程之Executor、ExecutorService、Executors、Callable、Future与FutureTask
3.1
//201521123055
public static synchronized void addId() {
id++;
}
public static synchronized void subtractId() {
id--;
}
两个函数加上synchronized关键字即可
3.2
ExecutorService ec= Executors.newCachedThreadPool();
for(int i=0;i<3;i++)
{
ec.execute(new Adder());
}
for(int i=0;i<3;i++)
{
ec.execute(new Subtracter());
}
ec.shutdown();
ec.awaitTermination(1, TimeUnit.DAYS);
关键是调用awaitTermination()来完成join效果。
Q4.线程间的合作:生产者消费者问题
4.1 运行MyProducerConsumerTest.java。正常运行结果应该是仓库还剩0个货物。多运行几次,观察结果,并回答:结果正常吗?哪里不正常?为什么?
4.2 使用synchronized, wait, notify解决该问题(关键代码截图,需出现学号)
4.1
有时候会出现还剩十个货物,因为存取的速度不同导致的。
4.2
//201521123055
public synchronized void add(String t) {
while (repo.size() == capacity) {
try{
wait();
}catch (Exception e)
{
e.printStackTrace();
}
}
repo.add(t);
notifyAll();
}
public synchronized void remove() {
while (repo.size() == 0) {
try{
wait();
}catch (Exception e)
{
e.printStackTrace();
}
}
repo.remove(0);
notifyAll();
}
Q5.查询资料回答:什么是线程安全?(用自己的话与代码总结,写自己看的懂的作业)
线程安全就是多线程访问时,采用了加锁机制,当一个线程访问该类的某个数据时,进行保护,其他线程不能进行访问直到该线程读取完,其他线程才可使用
听起来是有点糊涂,打个比方就是,俩人都要去图书馆借同一本书,A先弄到手,那B去借肯定是借不到的,做到同一时间只有一个线程调用某一资源,就是线程安全。保证线程安全是很重要的,因为多线程中很容易对资源的调用产生问题,这时候做到唯一性是很必要的。
3. PTA实验总结及码云上代码提交记录
3.1
3.2