201521123122 《java程序设计》第十一周实验总结


1. 本周学习总结

1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容。

其实这周也没讲多少内容,所以思维导图画的也比较简单。我在这里在总结一下:

1.如果有多个线程需要对同一个数据进行存取时,一定要考虑互斥的情况。

2.关于synchronized的用法:

synchronized有两种用法,第一种是代表原子操作的程序代码前加上synchronized标记,第二种是使用synchronized代码块。如

  1. public static synchronized void addId() {
  2. id++;
  3. }//第一种方法
  4. public static void addId() {
  5. synchronized (Counter.class) {//代表Counter类型的对象
  6. id++;
  7. }
  8. }//第二种方法

3.synchronized的作用:在java中,每个对象都有一把对象锁,只有获得对象的对象锁,才能执行此功能,而synchronized的作用在于把对象的对象锁“夺”过来,让自己执行此功能,而其他线程无法执行。(不知道这样理解对不对- -)。

4.关于多线程的同步问题:我的理解就是对于同一个数据,某个线程向其存储的速度和另外一个线程向其读取数据的速度不同,所以要通过wait () 和notify()/notifyAll()来控制。

5.关于wait () 和notify()/notifyAll()的用法:

wait()方法的作用就是让其线程释放被其“夺走”的对象锁,然后进入wait队列。notify()的作用就是再让进入wait队列的线程重新移入等待同一个“对象互斥锁”的队列。因为这两个方法都和“对象锁”有关,所以他们只能在声明为synchronized的代码中调用。

本次PTA作业题集多线程

互斥访问与同步访问

完成题集4-4(互斥访问)与4-5(同步访问)

1.1 除了使用synchronized修饰方法实现互斥同步访问,还有什么办法实现互斥同步访问(请出现相关代码)?

利用lock对象,代码如下:

  1. import java.util.concurrent.locks.Lock;
  2. import java.util.concurrent.locks.ReentrantLock;
  3. public class ProducerConsumerTestWithLock {
  4. public static void main(String[] args) {
  5. ProducerConsumerPool pool = new ProducerConsumerPool();
  6. ProducerConsumerPool.Producer p1 = pool.new Producer();
  7. ProducerConsumerPool.Consumer c1 = pool.new Consumer();
  8. Thread t1 = new Thread(p1);
  9. Thread t2 = new Thread(c1);
  10. t1.start();
  11. t2.start();
  12. System.out.println("Starting!");
  13. }
  14. }
  15. class ProducerConsumerPool{
  16. private Lock poolLock = new ReentrantLock();//建一个lock对象
  17. private static final int ITEMS = 1000;
  18. private int count=0;
  19. public int getCount(){
  20. return count;
  21. }
  22. public void setCount(int i){
  23. count = i;
  24. }
  25. public void produce(){
  26. poolLock.lock();//实现互斥访问
  27. setCount(getCount()+1);
  28. poolLock.unlock();//
  29. }
  30. public void consume(){
  31. poolLock.lock();//实现互斥访问
  32. setCount(getCount()-1);
  33. poolLock.unlock();//
  34. }
  35. public class Producer implements Runnable {
  36. public void run() {
  37. for (int i = 0; i < ITEMS; i++)
  38. produce();
  39. System.out.println(Thread.currentThread().getName()+" produed "+ITEMS+ " now the count is "+count);
  40. }
  41. }
  42. public class Consumer implements Runnable {
  43. public void run() {
  44. for (int i = 0; i < ITEMS; i++)
  45. consume();
  46. try {
  47. Thread.sleep(500);
  48. } catch (InterruptedException e) {
  49. e.printStackTrace();
  50. }
  51. System.out.println(Thread.currentThread().getName()+" consumed "+ITEMS+ " now the count is "+count);
  52. }
  53. }
  54. }

Lock和synchronized的区别

1.2 同步代码块与同步方法有何区别?

以下转自链接:https://www.nowcoder.com/questionTerminal/08ee355c9e7948ed9afbcf9a0d19998d

为何要使用同步?java允许多线程并发控制,当多个线程同时操作一个可共享的资源变量时(如数据的增删改查),将会导致数据不准确,相互之间产生冲突,因此加入同步锁以避免在该线程没有完成操作之前,被其他线程的调用,从而保证了该变量的唯一性和准确性.1.同步方法,即有synchronized关键字修饰的方法.由于java的每个对象都有一个内置锁,当用此关键字修饰方法时,内置锁会保护整个方法。在调用该方法前,需要获得内置锁,否则就处于阻塞状态.代码如:public synchronized void save(){}注:synchronized关键字也可以修饰静态方法,此时如果调用该静态方法,将会锁住整个类2.同步代码块。即有synchronized关键字修饰的语句块。被该关键字修饰的语句块会自动被加上内置锁,从而实现同步

代码如:synchronized(object){}注:同步是一种高开销的操作,因此应该尽量减少同步的内容。 通常没有必要同步整个方法,使用synchronized代码块同步关键代码即可。

总结来说就是,同步方法是对整个方法用内部锁修饰,而同步代码块只是对某段代码进行修饰,范围上看,是同步方法的范围较大些,但是就如上文所说,如果没有必要同步整个方法时,同步关键代码块是一个较好的选择。

1.3 实现互斥访问的原理是什么?请使用对象锁概念并结合相应的代码块进行说明。当程序执行synchronized同步代码块或者同步方法时,线程的状态是怎么变化的?

先谈什么是对象锁,说是对象锁,但我个人更喜欢把它比喻成钥匙,只有拥有对象锁的线程,才能获得这个对象的执行权利,首先一个对象只拥有一把对象锁,当多个线程同时访问这个对象时,就意味着多个线程同时争这个对象锁,这时被synchronized修饰的首先拥有了对象锁,这时只有他能访问其对象,其他人只能等他访问结束后在继续争其对象锁。

线程的状态变化:多个线程运行->多个线程竞争对象锁->一个线程获得对象锁运行,其余等待->一个线程结束,其余线程竞争对象锁->直到所有线程运行结束。

1.4 Java多线程中使用什么关键字实现线程之间的通信,进而实现线程的协同工作?为什么同步访问一般都要放到synchronized方法或者代码块中?

1.使用wait()和nitify()关键字实现。

2.wait()方法的作用就是让其线程释放被其“夺走”的对象锁,然后进入wait队列。nitify()的作用就是再让进入wait队列的线程重新移入等待同一个“对象互斥锁”的队列。因为这两个方法都和“对象锁”有关,所以他们只能在声明为synchronized的代码中调用。

交替执行

实验总结(不管有没有做出来)

实现交替访问关键是利用好Boolean变量,因为Boolean变量只有True和FALSE两个值,当每次执行完一个线程是,改变Boolean的值就可实现交替执行。关键代码如下:

  1. public synchronized void run1() {
  2. while (!flag) {
  3. try {
  4. wait();
  5. } catch (Exception e) {
  6. e.printStackTrace();
  7. }
  8. }
  9. if (getSize() > 0) {
  10. flag = false;
  11. System.out.println(Thread.currentThread().getName() + " finish " + task[i++]);
  12. notify();
  13. }
  14. }
  15. public synchronized void run2() {
  16. while (flag) {
  17. try {
  18. wait();
  19. } catch (Exception e) {
  20. e.printStackTrace();
  21. }
  22. }
  23. if (getSize() > 0) {
  24. flag = true;
  25. System.out.println(Thread.currentThread().getName() + " finish " + task[i++]);
  26. notify();
  27. }
  28. }
  29. }

还要注意的是把传递进来的字符串以空格区分分解为多个不同的任务要用split方法,每次执行完线程后,要删除这个任务,所以public int getSize() 时要返回task.length - i。

互斥访问

3.1 修改TestUnSynchronizedThread.java源代码使其可以同步访问。(关键代码截图,需出现学号)

加入synchronized的作用在于当这一线程执行此代码时,其他线程无法进行执行。

3.2 进一步使用执行器改进相应代码(关键代码截图,需出现学号)



线程间的合作:生产者消费者问题

4.1 运行MyProducerConsumerTest.java。正常运行结果应该是仓库还剩0个货物。多运行几次,观察结果,并回答:结果正常吗?哪里不正常?为什么?

每次运行的结果都不同,结果不正常,如图所示:

其原因就是线程没有同步访问,

4.2 使用synchronized, wait, notify解决该问题(关键代码截图,需出现学号)

查询资料回答:什么是线程安全?(用自己的话与代码总结,写自己看的懂的作业)

线程安全指的是 当多个线程在访问同一个对象时,不会发生冲突,多线程访问的结果和单线程访问的结果都是一样的。

链接

3. 码云上代码提交记录

题目集:多线程(4-4到4-10)

3.1. 码云代码提交记录

在码云的项目中,依次选择“统计-Commits历史-设置时间段”, 然后搜索并截图

3.2 截图多线程PTA提交列表

201521123122 《java程序设计》第十一周学习总结的更多相关文章

  1. “2017面向对象程序设计(Java)第十一周学习总结”存在问题的反馈及教学安排

    “2017面向对象程序设计(Java)第十一周学习总结”存在问题的反馈及教学安排1.“提出表扬的同学:姜依萍,王雪玲,徐楠,相文君,赵晓未提交作业的同学:任红强,王瑞强,宗鹏新,扎西才让,布旦刀杰,范 ...

  2. 201871010106-丁宣元 《面向对象程序设计(java)》第十一周学习总结

    201871010106-丁宣元 <面向对象程序设计(java)>第十一周学习总结 正文开头: 项目 内容 这个作业属于哪个课程 https://home.cnblogs.com/u/nw ...

  3. 20155304田宜楠2006-2007-2 《Java程序设计》第一周学习总结

    20155304田宜楠2006-2007-2 <Java程序设计>第一周学习总结 教材学习内容总结 - 浏览教材,根据自己的理解每章提出一个问题 第一章 除了书上提到的开发工具还有什么适合 ...

  4. 20175204 张湲祯 2018-2019-2《Java程序设计》第九周学习总结

    20175204 张湲祯 2018-2019-2<Java程序设计>第九周学习总结 教材学习内容总结 -第十一章JDBC和MySQL数据库要点: 1.下载MySQL和客户端管理工具navi ...

  5. 20175313 张黎仙《Java程序设计》第九周学习总结

    目录 学号 20175313 <Java程序设计>第九周学习总结 一.教材学习内容总结 二.教材学习中的问题和解决过程 三.代码托管 四.心得体会 五.学习进度条 六.参考资料 学号 20 ...

  6. 20175317 《Java程序设计》第九周学习总结

    20175317 <Java程序设计>第九周学习总结 学前准备 首先下载XAMPP,下载完成后打开: 将前两个选项打开 在IDEA中配置驱动mysql-connector-java-5.1 ...

  7. 20175202 《Java程序设计》第九周学习总结

    20175202 2018-2019-2 <Java程序设计>第九周学习总结 教材知识点总结 第11章 JDBC与MySQL数据库 MySQL数据库管理系统 MySQL数据库管理系统,简称 ...

  8. 20175227张雪莹 2018-2019-2 《Java程序设计》第九周学习总结

    20175227张雪莹 2018-2019-2 <Java程序设计>第九周学习总结 教材学习内容总结 第十一章 JDBC数据库操作 MySQL数据库管理系统 下载安装MySQL 若下载的是 ...

  9. 20175126《Java程序设计》第九周学习总结

    # 20175126 2016-2017-2 <Java程序设计>第九周学习总结 ## 教材学习内容总结 - 本周学习方式主要为手动敲代码并理解内容学习. - 学习内容为教材第十一章,本章 ...

  10. 20172325 2017-2018-2 《Java程序设计》第九周学习总结

    20172325 2017-2018-2 <Java程序设计>第九周学习总结 教材学习内容总结 异常 1.学习了异常的基本概念: 2.区分异常与错误: 一个异常是指一个定义非正常情况或错误 ...

随机推荐

  1. oracle启动 init.ora spfile pfile[转]

    昨天晚上快下班的时候,公司数据库突然堵住了,一个buf表中累计了20多W的数据提取不出来,改了程序,效果不明显.因为之前有一次也重启过oracle,所以这次还是想把oracle重启一下,因为那些数据都 ...

  2. Android 5.1.1在外置SD卡中创建文件夹

    Android 4.4之后WRITE_MEDIA_STORAGE 权限仅提供给系统应用,不再授予第三方App,WRITE_EXTERNAL_STORAGE 权限,仅仅用于授权用户写 primary e ...

  3. 百度将与W3C中国召开MIP技术研讨会

    百度计划与W3C中国共同组织国内W3C会员,于8月30日召开MIP 技术研讨会,讨论 MIP 等技术相关的应用标准,以期推进 MIP/AMP 在W3C中国的标准化进程. MIP (Mobile Ins ...

  4. MySql分库分表总结(转)

    为什么要分库分表 可以用说用到MySQL的地方,只要数据量一大, 马上就会遇到一个问题,要分库分表. 这里引用一个问题为什么要分库分表呢?MySQL处理不了大的表吗? 其实是可以处理的大表的.我所经历 ...

  5. 【Maven】项目中没有resources目录

    在eclipse中创建maven项目,src/main/只有java没有resources目录 解决办法: 1.项目右键properties 2.Java Build Path 中移除JRE Syst ...

  6. 2017-04-21周C语言学习笔记

    C语言学习笔记:... --------------------------------- C语言学习笔记:学习程度的高低取决于.自学能力的高低.有的时候生活就是这样的.聪明的人有时候需要.用笨的方法 ...

  7. extjs6中grid里放置图片

    1.加黑体的是实现代码,在view中操作 /** * Created by Wwei on 2017/7/1. */ Ext.define('Admin.view.userpanoram.UserPa ...

  8. SpringMVC(三)-- 视图和视图解析器、数据格式化标签、数据类型转换、SpringMVC处理JSON数据、文件上传

    1.视图和视图解析器 请求处理方法执行完成后,最终返回一个 ModelAndView 对象 对于那些返回 String,View 或 ModeMap 等类型的处理方法,SpringMVC 也会在内部将 ...

  9. JEasyPoi 2.1.4 (Jeecg订制) 版本发布,Excel 和 Word 简易工具类

    JEasyPoi 2.1.4 (jeecg订制)版本发布,EasyPoi Excel 和 Word 简易工具类 easypoi 功能如同名字easy,主打的功能就是容易,让一个没见接触过poi的人员 ...

  10. 通俗语言解释内外网IP与端口映射

    IP:分为外网IP和内网IP 也就是我们说的外网IP属于实体IP 实体IP,它是独一无二的,在网络的世界里,每一部计算机的都有他的位置,一个 IP 就好似一个门牌!例如,你要去百度的网站的话,就要去『 ...