JAVA2015086第十一周作业
- 本周学习总结
1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容。
- 书面作业
本次PTA作业题集多线程
1.互斥访问与同步访问
完成题集4-4(互斥访问)与4-5(同步访问)
1.1 除了使用synchronized修饰方法实现互斥同步访问,还有什么办法实现互斥同步访问(请出现相关代码)?
还可以使用Lock对象和Condition对象实现互斥同步访问。
public void deposit(int money) {
lock.lock();
try {
this.balance = getBalance() + money;
} finally {
lock.unlock();
}
}
public synchronized void withdraw(int money) {
lock.lock();
this.balance = getBalance() - money;
try {
try {
if (balance < 0) {
condition.await();
}
condition.signalAll();
} catch (Exception e) {
e.printStackTrace();
}
if (balance < 0) {
throw new IllegalStateException(balance + "");
}
} finally {
lock.unlock();
}
}
1.2 同步代码块与同步方法有何区别?
主要是格式上的不同,同步代码块主要是在方法中使用synchronized关键字实现互斥public void 方法名{ synchronized(类对象名T) {//同步代码块} };同步方法是在定义方法时,加在void前,public synchronized void 方法名{//同步代码块}
1.3 实现互斥访问的原理是什么?请使用对象锁概念并结合相应的代码块进行说明。当程序执行synchronized同步代码块或者同步方法时,线程的状态是怎么变化的?
①需要实现互斥访问的原因是有时两个或两个以上的线程需要同时对同一资源存取,而线程之间如果不加以控制,会产生一种情况-竞争;线程对该资源进行操作时即上锁,这样其他同步线程就无法获取资源,等执行中的线程结束之后,释放资源,让其他线程继续争取对资源的使用权。
②关于对象锁:每个对象都有一把锁,如果未获得对象锁,下面的代码就无法执行,必须等待。从而通过对象锁实现了互斥访问。
class MyCounter{
private int i = 0;
public void increment(){
synchronized (this) {//获得this这个对象上的内部锁
i++;
}
}
}
③假设线程A与线程B同时争取同一个资源,当A和B同时获取CPU进入Running之后,只能有一个线程(假设为A)获得资源的使用权,那么A线程就能进入LockPool获取同步锁继续执行,结束后将同步锁释放出来,而B就要进入WaitPool(通过wait();)等待同步锁被释放后进入LockPool(通过notify();/notifyAll();)重新争取同步锁。
如果有多个线程,则每次只有其中的一个线程获得对象锁,进入addId()、substractId()方法,执行id++、id--。直到该方法结束,同步锁被释放,下一个线程才能获得对象锁。
1.4 Java多线程中使用什么关键字实现线程之间的通信,进而实现线程的协同工作?为什么同步访问一般都要放到synchronized方法或者代码块中?
wait()和notify()/nofityAll()来实现线程之间的通信;存在多个线程同时运行时,没有互斥,将会使线程运行混乱,例如存钱与取钱,可能有多个人要进行,若是synchronized关键字,存钱时与取钱时,可能会造成A存钱时,B存钱也同时进行,而C取钱,余额在A,B,C的操作下同时进行,数值将会错误
2.交替执行
实验总结(不管有没有做出来)
实验总结:先是建立Repo类,存储字符串items.split(" ");;Worker1和Worker2类继承Runnable接口,若是要两个线程交替运行,需要使用wait()和notify()函数,需要在run函数加入synchronized关键字,run函数中循环while(),循环条件是this.repo.getSize()!=0,输出字符串数组,并将null该字符串数组,达到将任务删除,通过判断数组下标的奇偶数来决定接下来需要运行的线程,若是偶数,则运行Worker1,Worker2则wait();若是奇数则运行Worker2,Worker1则wait();以此来交替线程运行;题目的难点在于怎么样设计这个两个线程的交替实现,由于线程争用cpu的随机性,就需要Worker1和Worker2线程执行,在Worker1执行完一次进入等待之前唤醒Worker2,如此往复,主要要用上wait()和notify()了。
3.1
修改TestUnSynchronizedThread.java源代码使其可以同步访问。(关键代码截图,需出现学号)**
//201521123086
public static synchronized void addId() {
id++;
}
public static synchronized void subtractId() {
id--;
}
3.2
进一步使用执行器改进相应代码(关键代码截图,需出现学号)
//201521123086
List<Callable> task=new ArrayList<>();
ExecutorService exec=Executors.newCachedThreadPool();
for (int i = 0; i < 3; i++) {
task.add(Executors.callable(new Adder()));
}
for (int i = 0; i < 3; i++) {
task.add(Executors.callable(new Subtracter()));
}
exec.invokeAll(task);
System.out.println(Counter.getId());
System.out.println("main end")
4.线程间的合作:生产者消费者问题
4.1 运行MyProducerConsumerTest.java。正常运行结果应该是仓库还剩0个货物。多运行几次,观察结果,并回答:结果正常吗?哪里不正常?为什么?
并不是每一次的结果都是0,结果不正常,因为生产者的生产速度和消费者的取货速度不一致,二者没有合作,则可能可出现供给需求不一致的情况。
4.2 使用synchronized, wait, notify解决该问题(关键代码截图,需出现学号)
//201521123086
public synchronized void add(String t) {
while(repo.size() == capacity) {
System.out.println("仓库已满!无法添加货物。");
try {
wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
repo.add(t);
notifyAll();
}
public synchronized void remove() {
while (repo.size() == 0) {
System.out.println("仓库无货!无法从仓库取货");
try {
wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
5.查询资料回答:什么是线程安全?(用自己的话与代码总结,写自己看的懂的作业)
线程安全:如果代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,那么就是线程安全的。更专业一点的说法是:一个类或者程序所提供的接口对于线程来说是原子操作或者多个线程之间的切换不会导致该接口的执行结果存在二义性,也就是说我们不用考虑同步的问题。
举个反例:
比如i++执行的时候可能会有两步来完成:1.x=i+1;2.i=x。在单线程运行的情况下,如果 i= 0,i++后,i=1;而如果是在多线程情况下,比如有两个线程,线程 A 取i得值为0,但是此时 CPU 调度线程A暂停,线程 B 得到运行的机会。线程B也取得i=0的值,因为此时i还是原来的值,并未被线程A改变,所以线程B也是i=0。然后线程A和线程B都继续运行,都执行i++,结果i都等于1。这就很明显,i通过线程AB进行两次自加,结果应该为2的,但是却等于1,这就是"线程不安全"了。
- 码云上代码提交记录
题目集:多线程(4-4到4-10)
3.1. 码云代码提交记录
在码云的项目中,依次选择“统计-Commits历史-设置时间段”, 然后搜索并截图
3.2 截图多线程PTA提交列表
JAVA2015086第十一周作业的更多相关文章
- 2017-2018-2 20179205《网络攻防技术与实践》第十一周作业 SQL注入攻击与实践
<网络攻防技术与实践>第十一周作业 SQL注入攻击与实践 1.研究缓冲区溢出的原理,至少针对两种数据库进行差异化研究 缓冲区溢出原理 在计算机内部,输入数据通常被存放在一个临时空间内, ...
- 《Linux内核原理与设计》第十一周作业 ShellShock攻击实验
<Linux内核原理与设计>第十一周作业 ShellShock攻击实验 分组: 和20179215袁琳完成实验及博客攥写 实验内容: Bash中发现了一个严重漏洞shellshock, ...
- 2019-2020-1 20199329《Linux内核原理与分析》第十一周作业
<Linux内核原理与分析>第十一周作业 一.本周内容概述: 学习linux安全防护方面的知识 完成实验楼上的<ShellShock 攻击实验> 二.本周学习内容: 1.学习& ...
- 2020-2021-1 20209307 《Linux内核原理与分析》第十一周作业
这个作业属于哪个课程 <2020-2021-1Linux内核原理与分析)> 这个作业要求在哪里 <2020-2021-1Linux内核原理与分析第十一周作业> 这个作业的目标 ...
- 1903021121-刘明伟-java十一周作业-java面向对象编程
项目 内容 课程班级博客链接 19级信计班(本) 作业要求链接 第十一周作业 博客名称 1903021121-刘明伟-java十一周作业-java面向对象 要求 每道题要有题目,代码(使用插入代码,不 ...
- 2017-2018-2 1723《程序设计与数据结构》第十一周作业 & 实验三 & (总体)第三周结对编程 总结
作业地址 第十一次作业:https://edu.cnblogs.com/campus/besti/CS-IMIS-1723/homework/1933 (作业界面已评分,可随时查看,如果对自己的评分有 ...
- C语言第十一周作业
这个作业属于哪个课程 C语言程序设计II 这个作业要求在哪里 https://edu.cnblogs.com/campus/zswxy/computer-scienceclass3-2018/ ...
- 2019春第十一周作业Compile Summarize
这个作业属于那个课程 C语言程序设计II 这个作业要求在哪里 这里 我在这个课程的目标是 能按自己的想法解出题目 这个作业在那个具体方面帮助我实现目标 能朝着软件工程师方向发展 参考文献与网址 C语言 ...
- 201521123035《Java程序设计》第十一周作业
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容. 本周对多线程的冲突是从多线程的冲突开始讲起,从而提出互斥共享与互斥访问.其中,互斥访问提到了synchronize ...
随机推荐
- AIX安装恢复oracle问题-内存不足
AIX安装恢复oracle问题-----------------------2013/10/19 oracle 安装后后,进行rman恢复数据库时,启动不了dummy实例,报内存不足 RMAN&g ...
- python---scrapy之MySQL同步存储
假设我们已经能获取到item里定义的字段的数据,接下来就需要保存item的数据到mysql数据库. pipeline用来存储item中的数据,将爬取到的数据进行二次处理 首先,要做的准备的工作,安装M ...
- exit()与_exit()的区别(转)
http://blog.csdn.net/lwj103862095/article/details/8640037 从图中可以看出,_exit 函数的作用是:直接使进程停止运行,清除其使用的内存空间, ...
- html4与html5的区别及html5的一些新特性
区别 1.html5语法的改变 HTML5简化了很多细微的语法,例如: 1.1doctype的声明; html4: <!DOCTYPE HTML PUBLIC "-//W3C//DTD ...
- IT科技企业逻辑思维面试题
逻辑思维面试题 一.假设有一个池塘,里面有无穷多的水.现有2个空水壶,容积分别为5升和6升.问题是如何只用这2个水壶从池塘里取得3升的水.[请描述操作过程] 答:(1)先用容积为6升的水壶装满水: ( ...
- BotVS趋势交易策略-MA均线
1. 均线策略1号 思路:使用MA小时线,入市线金叉买入,出市线死叉时卖出.代码如下 import types def main(): STATE_IDLE = -1 state = STATE_ID ...
- 记一次Spring aop的所遇到的问题
由来 项目中需要实现某个订单的状态改变后然后推送给第三方的功能,由于更改状态的项目和推送的项目不是同一个项目,所以为了不改变原项目的代码,我们考虑用spring的aop来实现. 项目用的是spring ...
- 迈向angularjs2系列(1):typescript指南
typescript指南 前言 typescript是angularjs2推荐使用的脚本语言.它由微软2012年首次发布. 一. typescript和javascript的区别 1.从遵循的规 ...
- MyEclipse简介
- [C]成员运算符"."和间接成员运算符"->"浅析
成员运算符: . 成员运算符一般和结构或者联合名一起使用,指定结构或者联合中的某个成员. 举个栗子: 如果Ronz是一个结构的名称,linux是这个结构模板指定的一个成员名. struct{ //匿名 ...