1.本周学习总结

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

2.书面作业

1.互斥访问与同步访问

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

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

  1. 可以使用Lock对象和Condition对象来实现互斥同步访问。(注意使用Lock对象时一定要unlock
  2. //201521123027
  3. public void deposit(int money){
  4. Lock.lock();
  5. try{
  6. setBalance(this.balance+money);
  7. }finally{
  8. Lock.unlock();
  9. }
  10. }
  11. public void withdraw(int money){
  12. Lock.lock();
  13. try{
  14. try{
  15. while(this.getBalance()<=0){
  16. condition.await();
  17. condition.signal();
  18. }
  19. }catch(Exception e){
  20. System.out.println(e);
  21. }
  22. setBalance(this.balance-money);
  23. if(balance<0)
  24. throw new IllegalStateException(balance+"");
  25. }finally{
  26. Lock.unlock();
  27. }
  28. }

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

  1. 1)同步方法, 即有synchronized关键字修饰的方法。由于java的每个对象都有一个内置锁,当用此关键字修饰方法时,内置锁会保护整个方法。在调用该方法前,需要获得内置锁,否则就处于阻塞状态。
  2. 2)同步代码块,即有synchronized关键字修饰的语句块。被该关键字修饰的语句块会自动被加上内置锁,从而实现同步。
  3. 实现同步是一种高开销的操作,所以一般使用同步代码块比使用同步方法效率更高一点。

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

  1. 实现互斥访问的原理:每一个对象都有一个监视器锁,当线程获取到此对象的监视器锁的所有权时,此监视器锁就会被锁定,不能在被其他线程所占用,直到此线程执行完毕,监视器锁将释放,其他线程才能再次争夺监视器锁的所有权。例如:
  2. class Counter {
  3. private static int id = 0;
  4. public synchronized static void addId() {
  5. id++;
  6. }
  7. public synchronized static void subtractId() {
  8. id--;
  9. }
  10. public static int getId() {
  11. return id;
  12. }
  13. }
  14. 在这段代码中,线程1获得对象的监视器锁的所有权,获得id的值为0,然后执行addId(),id自增,此时id的值变为1,线程1执行完毕,id的监视器锁释放;线程2会获得id的监视器锁的所有权,获得id的值为1,然后执行subtractId(),id自减,此时id的值变为0,线程2执行完毕,id的监视器锁释放。所有线程执行完毕后输出id的值为0
  15. 线程的状态变化:多个线程运行—>多个线程竞争对象的监视器锁—>一个线程获得监视器锁的所有权,运行,其余线程等待—>一个线程结束,其余线程可竞争对象监视器锁—>直到所有线程运行结束输出结果

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

  1. wait()、notify()/notifyAll()实现了线程之间的协同工作;
  2. 当多个线程同时工作时,若没有将同步访问一般都要放到synchronized方法或者代码块中,线程之间会因为争夺同一个资源导致发生冲突,线程运行会出现混乱,最后结果会出现错误。

2.交替执行

2.1实验总结

  1. 1)使用.split" ")方法存储字符串;
  2. 2)每个方法都需要用synchronized关键字修饰;
  3. 3)线程之间的合作需要使用wait()、notify()函数;.
  4. 4)线程的交替运行需要通过对数组下标进行奇偶判断,若为偶数,则唤醒worker1worker2进入wait队列,若为奇数,则唤醒worker2worker1进入wait队列;
  5. 5)输出格式:System.out.println(Thread.currentThread().getName()+"finish"+a[i]);

3.互斥访问

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

  1. 此题只需加上synchronized关键字即可
  2. //201521123027
  3. class Counter {
  4. private static int id = 0;
  5. public synchronized static void addId() {
  6. id++;
  7. }
  8. public synchronized static void subtractId() {
  9. id--;
  10. }
  11. public static int getId() {
  12. return id;
  13. }
  14. }

结果截图:

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

  1. 使用invokeAll方法。网上搜索得知invokeAll方法有两种格式:(1invokeAll(tasks) 批量提交不限时任务; 2invokeAll(tasks, timeout, unit) 批量提交限时任务。这里采用的是第一种格式。使用两个for循环,循环次数要相等,这样可以保证实现加、减的线程运行次数相同且同步,输出结果为0
  2. //201521123027
  3. public class TestUnSynchronizedThread {
  4. public static void main(String[] args) throws InterruptedException {
  5. // TODO Auto-generated method stub
  6. ArrayList<Callable<Object>> tasks=new ArrayList<>();
  7. ExecutorService executor =(ExecutorService)Executors.newCachedThreadPool();
  8. for(int i=0;i<3;i++){
  9. tasks.add(Executors.callable(new Adder()));
  10. }
  11. for(int i=0;i<3;i++){
  12. tasks.add(Executors.callable(new Subtracter()));
  13. }
  14. executor.invokeAll(tasks);
  15. System.out.println(Counter.getId());
  16. System.out.println("main end");
  17. }
  18. }

参考资料:Java多线程之Executor、ExecutorService、Executors、Callable、Future与FutureTask

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

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

  1. 结果不正常,运行多次代码发现每次代码运行结果并不一定相同,有时候结果还会为仓库还剩10个货物。因为生产者和消费者的存取速度不同,没有做到线程间的同步,所以会出现以下问题:(1)生产者比消费者快时,消费者来不及取数据; 2)消费者比生产者快时,消费者可能取不到数据。

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

  1. //201521123027
  2. public synchronized void add(String t){
  3. try{
  4. while(repo.size() >= capacity){
  5. wait();
  6. System.out.println("仓库已满!无法添加货物。");
  7. }
  8. }catch(Exception e){
  9. System.out.println(e);
  10. }
  11. repo.add(t);
  12. notifyAll();
  13. }
  14. public synchronized void remove(){
  15. try{
  16. while(repo.size() <= 0){
  17. wait();
  18. System.out.println("仓库无货!无法从仓库取货");
  19. }
  20. }catch(Exception e){
  21. System.out.println(e);
  22. }
  23. repo.remove(0);
  24. notifyAll();
  25. }

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

  1. 线程安全就是说多个线程访问同一段代码时不会出现不确定的结果。要想实现线程安全就要实现线程同步,此时就要使用到synchronized关键字,synchronized关键字保证了同一段代码同时只有一个线程在操作。如果一段代码每次多线程运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。例如:
  2. class Counter {
  3. private static int id = 0;
  4. public synchronized static void addId() {
  5. id++;
  6. }
  7. public synchronized static void subtractId() {
  8. id--;
  9. }
  10. public static int getId() {
  11. return id;
  12. }
  13. }
  14. 因为这段代码每一次的运行结果都是0,所以就可以说这段代码时线程安全的。

3.码云上代码提交记录

3.1码云代码提交记录

3.2截图PTA提交记录

201521123027 <java程序设计>第11周学习总结的更多相关文章

  1. 201521123045 <java程序设计>第11周学习总结

    201521123045 <java程序设计>第11周学习总结 1. 本周学习总结 2. 书面作业 2. 书面作业 Q1.1.互斥访问与同步访问完成题集4-4(互斥访问)与4-5(同步访问 ...

  2. 201521123027 <java程序设计>第13周学习总结

    1.本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 2.书面作业 1. 网络基础 1.1 比较ping www.baidu.com与ping cec.jmu.e ...

  3. 201521123027 <java程序设计>第七周学习总结

    1.本周学习总结 2.书面作业 Q1.ArrayList代码分析 1.1 解释ArrayList的contains源代码 答: 源代码: //contains()方法 public boolean c ...

  4. 201521123027 <java程序设计>第十周学习总结

    1.本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结异常与多线程相关内容. 异常: 多线程: 2.书面作业 Q1.finally 题目4-2 1.1 截图你的提交结果(出现学号) 1.2 ...

  5. 201521123027 <java程序设计>第八周学习总结

    1.本周学习总结 1.1思维导图 2.书面作业 Q1.List中指定元素的删除(题目4-1) 1.1 实验总结 总结:判断List中是否存在指定元素,需要用到equals方法,若存在就用remove进 ...

  6. 2018面向对象程序设计(Java)第11周学习指导及要求

    2018面向对象程序设计(Java)第11周学习指导及要求 (2018.11.8-2018.11.11)   学习目标 (1) 掌握Vetor.Stack.Hashtable三个类的用途及常用API: ...

  7. 面向对象程序设计(JAVA) 第11周学习指导及要求

    2019面向对象程序设计(Java)第11周学习指导及要求 (2019.11.8-2018.11.11)   学习目标 理解泛型概念: 掌握泛型类的定义与使用: 掌握泛型方法的声明与使用: 掌握泛型接 ...

  8. 20145236 《Java程序设计》第九周学习总结

    20145236 <Java程序设计>第九周学习总结 教材学习内容总结 第十六章 整合数据库 JDBC简介 1.JDBC是java联机数据库的标准规范.它定义了一组标准类与接口,标准API ...

  9. 2018-2019 2 20175230《Java程序设计》第九周学习总结

    <Java程序设计>第九周学习总结 主要内容 MySQL数据库管理系统 1.下载 2.安装 启动MySQL数据库服务器 1.启动 2.root用户 MySQL客户端管理工具 建立连接 建立 ...

随机推荐

  1. 【LeetCode】数组-3(605)-种花问题( 1 的两侧不能有 1 )

    开始的思路:首先统计需要种几只花,用花的数目统计连续 0 的个数.... ...[囧]突然觉得情况有点复杂啊,有连续的又有分散的怎么能统计全呢? 好吧这里喔偷偷的瞄了一眼参看答案... ...(就一眼 ...

  2. leetcode【sql】 Delete Duplicate Emails

    Write a SQL query to delete all duplicate email entries in a table named Person, keeping only unique ...

  3. MFC对话框中显示背景图片

    在MFC对话框中显示图片,四个步骤. 1.首先得在VC6.0或者VS2008(其他版本也是一样)中导入GDI文件.(网上下载:) GDI含义是图形设备接口,主要任务是负责系统与绘图程序之间的信息交换, ...

  4. 奇货商城重构——webpack自动化工程

    近几年,前端各种框架工具层出不穷,从两三年前还是一个jQuery搞定全站,到之后requirejs/seajs,node,gulp/webpack,Angular/React/Vue,RN/weex的 ...

  5. Linux巩固记录(1) J2EE开发环境搭建及网络配置

    由于要近期使用hadoop等进行相关任务执行,操作linux时候就多了 以前只在linux上配置J2EE项目执行环境,无非配置下jdk,部署tomcat,再通过docker或者jenkins自动部署上 ...

  6. java基础部分细节

    访问控制修饰符 Java中,可以使用访问控制符来保护对类.变量.方法和构造方法的访问.Java支持4种不同的访问权限. 默认的,也称为 default,在同一包内可见,不使用任何修饰符. 私有的,以  ...

  7. python基础教程(三)

    序列概览 Python 包含6 种内建的序列,这里重点讨论最常用的两种类型:列表和元组. 列表与元组的主要区别在于,列表可以修改,元组则不能.也就是说如果要根据要求来添加元素,那么列表可以会更好用:而 ...

  8. 利用wireshark任意获取qq好友IP实施精准定位

    没事玩一把,感觉还挺有趣,首先打开wireshark: 不管你连接的什么网,如图我连接的是WLAN,双击进入如图界面: ctrl-f进行搜索:如图 选择分组详情,字符串,并输入020048.这时候你就 ...

  9. chrome开发工具指南(八)

    编辑 DOM Chrome DevTools 的 Elements 面板中的 DOM 树视图可以显示当前网页的 DOM 结构.通过 DOM 更新实时修改页面的内容和结构. DOM 定义您的页面结构.每 ...

  10. web服务的三大主流架构

    1.远程过程调用 2.服务导向架构 3.表述性状态转移 今天在自学Spring架构时貌似发现 rest的表现.与RPC方式的最大差别,SOA方式更加关注如何去连接服务而不是去特定某个实现的细节.而re ...