201521123117 《Java程序设计》第11周学习总结
1. 本周学习总结
1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容。
2. 书面作业
本次PTA作业题集多线程
1.互斥访问与同步访问
完成题集4-4(互斥访问)与4-5(同步访问)
1.1 除了使用synchronized修饰方法实现互斥同步访问,还有什么办法实现互斥同步访问(请出现相关代码)?
可以使用Lock对象和Condition对象来实现互斥同步访问。(注意使用Lock对象时一定要unlock)
//201521123117
public void deposit(int money){
Lock.lock();
try{
setBalance(this.balance+money);
}finally{
Lock.unlock();
}
}
public void withdraw(int money){
Lock.lock();
try{
try{
while(this.getBalance()<=0){
condition.await();
condition.signal();
}
}catch(Exception e){
System.out.println(e);
}
setBalance(this.balance-money);
if(balance<0)
throw new IllegalStateException(balance+"");
}finally{
Lock.unlock();
}
}
1.2 同步代码块与同步方法有何区别?
(1)同步方法, 即有synchronized关键字修饰的方法。由于java的每个对象都有一个内置锁,当用此关键字修饰方法时,内置锁会保护整个方法。在调用该方法前,需要获得内置锁,否则就处于阻塞状态。
(2)同步代码块,即有synchronized关键字修饰的语句块。被该关键字修饰的语句块会自动被加上内置锁,从而实现同步。
实现同步是一种高开销的操作,所以一般使用同步代码块比使用同步方法效率更高一点。
1.3 实现互斥访问的原理是什么?请使用对象锁概念并结合相应的代码块进行说明。当程序执行synchronized同步代码块或者同步方法时,线程的状态是怎么变化的?
实现互斥访问的原理:每一个对象都有一个监视器锁,当线程获取到此对象的监视器锁的所有权时,此监视器锁就会被锁定,不能在被其他线程所占用,直到此线程执行完毕,监视器锁将释放,其他线程才能再次争夺监视器锁的所有权。例如:
class Counter {
private static int id = 0;
public synchronized static void addId() {
id++;
}
public synchronized static void subtractId() {
id--;
}
public static int getId() {
return id;
}
}
在这段代码中,线程1获得对象的监视器锁的所有权,获得id的值为0,然后执行addId(),id自增,此时id的值变为1,线程1执行完毕,id的监视器锁释放;线程2会获得id的监视器锁的所有权,获得id的值为1,然后执行subtractId(),id自减,此时id的值变为0,线程2执行完毕,id的监视器锁释放。所有线程执行完毕后输出id的值为0。
线程的状态变化:多个线程运行—>多个线程竞争对象的监视器锁—>一个线程获得监视器锁的所有权,运行,其余线程等待—>一个线程结束,其余线程可竞争对象监视器锁—>直到所有线程运行结束输出结果
1.4 Java多线程中使用什么关键字实现线程之间的通信,进而实现线程的协同工作?为什么同步访问一般都要放到synchronized方法或者代码块中?
wait()、notify()/notifyAll()实现了线程之间的协同工作;
当多个线程同时工作时,若没有将同步访问一般都要放到synchronized方法或者代码块中,线程之间会因为争夺同一个资源导致发生冲突,线程运行会出现混乱,最后结果会出现错误。
2.交替执行
实验总结(不管有没有做出来)
答:第一,实验中的大问题就是代码写不出来,只能上网去查资料,然后老师上课的讲解,问舍友(最主要),然后一遍一遍的试一遍一遍的过。
第二:其中会出现一些前几周也会出现的代码格式上的小问题,这些小问题比较好解决。
3.互斥访问
3.1 修改TestUnSynchronizedThread.java源代码使其可以同步访问。(关键代码截图,需出现学号)
此题只需加上synchronized关键字即可
//201521123117
class Counter {
private static int id = 0;
public synchronized static void addId() {
id++;
}
public synchronized static void subtractId() {
id--;
}
public static int getId() {
return id;
}
}
结果截图:
3.2 进一步使用执行器改进相应代码(关键代码截图,需出现学号)
使用invokeAll方法。网上搜索得知invokeAll方法有两种格式:(1)invokeAll(tasks) 批量提交不限时任务; (2)invokeAll(tasks, timeout, unit) 批量提交限时任务。这里采用的是第一种格式。使用两个for循环,循环次数要相等,这样可以保证实现加、减的线程运行次数相同且同步,输出结果为0。
//201521123117
public class TestUnSynchronizedThread {
public static void main(String[] args) throws InterruptedException {
// TODO Auto-generated method stub
ArrayList<Callable<Object>> tasks=new ArrayList<>();
ExecutorService executor =(ExecutorService)Executors.newCachedThreadPool();
for(int i=0;i<3;i++){
tasks.add(Executors.callable(new Adder()));
}
for(int i=0;i<3;i++){
tasks.add(Executors.callable(new Subtracter()));
}
executor.invokeAll(tasks);
System.out.println(Counter.getId());
System.out.println("main end");
}
}
4.线程间的合作:生产者消费者问题
4.1 运行MyProducerConsumerTest.java。正常运行结果应该是仓库还剩0个货物。多运行几次,观察结果,并回答:结果正常吗?哪里不正常?为什么?
不正常,当仓库没货的时候,消费者是取不到货物的。主要还是没实现同步互斥。
4.2 使用synchronized, wait, notify解决该问题(关键代码截图,需出现学号)
//201521123117
public synchronized void add(String t){
try{
while(repo.size() >= capacity){
wait();
System.out.println("仓库已满!无法添加货物。");
}
}catch(Exception e){
System.out.println(e);
}
repo.add(t);
notifyAll();
}
public synchronized void remove(){
try{
while(repo.size() <= 0){
wait();
System.out.println("仓库无货!无法从仓库取货");
}
}catch(Exception e){
System.out.println(e);
}
repo.remove(0);
notifyAll();
}
5.查询资料回答:什么是线程安全?(用自己的话与代码总结,写自己看的懂的作业)
对多线程而言,使用synchronized关键字实现一次一个线程享用资源,而不是多个线程同时共享资源
3. 码云上代码提交记录
201521123117 《Java程序设计》第11周学习总结的更多相关文章
- 201521123045 <java程序设计>第11周学习总结
201521123045 <java程序设计>第11周学习总结 1. 本周学习总结 2. 书面作业 2. 书面作业 Q1.1.互斥访问与同步访问完成题集4-4(互斥访问)与4-5(同步访问 ...
- 201521123027 <java程序设计>第11周学习总结
1.本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容. 2.书面作业 1.互斥访问与同步访问 完成题集4-4(互斥访问)与4-5(同步访问) 1.1 除了使用synchro ...
- 2018面向对象程序设计(Java)第11周学习指导及要求
2018面向对象程序设计(Java)第11周学习指导及要求 (2018.11.8-2018.11.11) 学习目标 (1) 掌握Vetor.Stack.Hashtable三个类的用途及常用API: ...
- 面向对象程序设计(JAVA) 第11周学习指导及要求
2019面向对象程序设计(Java)第11周学习指导及要求 (2019.11.8-2018.11.11) 学习目标 理解泛型概念: 掌握泛型类的定义与使用: 掌握泛型方法的声明与使用: 掌握泛型接 ...
- 20145236 《Java程序设计》第九周学习总结
20145236 <Java程序设计>第九周学习总结 教材学习内容总结 第十六章 整合数据库 JDBC简介 1.JDBC是java联机数据库的标准规范.它定义了一组标准类与接口,标准API ...
- 2018-2019 2 20175230《Java程序设计》第九周学习总结
<Java程序设计>第九周学习总结 主要内容 MySQL数据库管理系统 1.下载 2.安装 启动MySQL数据库服务器 1.启动 2.root用户 MySQL客户端管理工具 建立连接 建立 ...
- 20175209 《Java程序设计》第九周学习总结
20175209 <Java程序设计>第九周学习总结 一.教材知识点总结 有关数据库下载中存在可能出现的问题已经在博客<数据库安装和使用过程中出现的一些问题>给出了相应的解决办 ...
- 20175208 《Java程序设计》第九周学习总结
20175208 2018-2019-2 <Java程序设计>第九周学习总结 一.教材学习内容总结: 第11章 JDBC与MySQL数据库 MySQL数据库管理系统 MySQL数据库管理系 ...
- 20175202 《Java程序设计》第九周学习总结
20175202 2018-2019-2 <Java程序设计>第九周学习总结 教材知识点总结 第11章 JDBC与MySQL数据库 MySQL数据库管理系统 MySQL数据库管理系统,简称 ...
- 20175227张雪莹 2018-2019-2 《Java程序设计》第九周学习总结
20175227张雪莹 2018-2019-2 <Java程序设计>第九周学习总结 教材学习内容总结 第十一章 JDBC数据库操作 MySQL数据库管理系统 下载安装MySQL 若下载的是 ...
随机推荐
- Python 第七天
OOP 面向对象编程--Object Oriented Programming,简称OOP,是一种程序设计思想.在Python中,所有数据类型都可以视为对象,当然也可以自定义对象.自定义的对象数据类型 ...
- python3 接口测试 一般方法
第一步: 导入相关包 import http.client 第二步: 将ip和端口号,使用如下命令,赋值给变量. conn = http.client.HTTPConnection("124 ...
- ubuntu14.04 64位安装H3C iNode客户端
环境: OS:ubuntu14.04LTS 64位 iNode: iNode2.40-R0162 for linux(iNode只有32位的,而且是很久以前的版本) 安装方法: 第一种: 检查本机是6 ...
- Go指针
Go 语言指针 Go 语言中指针是很容易学习的,Go 语言中使用指针可以更简单的执行一些任务. 接下来让我们来一步步学习 Go 语言指针. 我们都知道,变量是一种使用方便的占位符,用于引用计算机内存地 ...
- 为什么新生代内存需要有两个Survivor区
转载自:http://blog.csdn.net/antony9118/article/details/51425581 在JVM的新生代内存中,为什么除了Eden区,还要设置两个Survivor区? ...
- 【转】控制台,终端,tty,shell等概念的区别
转自:http://www.2cto.com/os/201403/282583.html http://blog.sina.com.cn/s/blog_bcdac52b0101i2r1.html 控制 ...
- Eclipse自动生成返回值对象的快捷键是什么?
如下代码 List<ShareholderEntity> shareList = fetch.parseShareHolder(data); 如何自动生成 List<Sharehol ...
- 转载:细说Cookie
细说Cookie 转载:http://www.cnblogs.com/fish-li/archive/2011/07/03/2096903.html 阅读目录 开始 Cookie 概述 Cookie的 ...
- 获取MVC中Controller下的Action参数异常
我现在做的一个项目有一个这样的需求, 比如有一个页面需要一个Guid类型的参数: public ActionResult Index(Guid id) { //doing something ... ...
- HashMap 数组应用面试题(Point)
今天看了一题面试题,以为很简单,不过自己写了遍,没有完全写出来: 题目是这样的: 给定一些 points 和一个 origin,从 points 中找到 k 个离 origin 最近的点.按照距离由小 ...