关键词:
一、理论部分
1、多线程并发执行中的问题
i.多个线程相对执行的顺序是不确定的。
ii.线程执行顺序的不确定性会产生执行结果的不确定性。
iii.在多线程对共享数据操作时常常会产生这种不确定性。
2、线程的同步
-多线程并发运行不确定性问题解决方案:引入线程同步机制,使得另一线程要使用该方法,就只能等待。
- 在Java中解决多线程同步问题的方法有两种:
- Java SE 5.0中引入ReentrantLock类
- 在共享内存的类方法前加synchronized修饰符。
……
public synchronized static void sub(int m)
……
(1)解决方案一:锁对象与条件对象
用ReentrantLock保护代码块的基本结构如下:
myLock.lock();
try
critical section
finally
myLock.unlock();
有关锁对象和条件对象的关键要点:
? 锁用来保护代码片段,保证任何时刻只能有一个线程执行被保护的代码。
? 锁管理试图进入被保护代码段的线程。
? 锁可拥有一个或多个相关条件对象。
? 每个条件对象管理那些已经进入被保护的代码段但还不能运行的线程。
(2)解决方案二: synchronized关键字
synchronized关键字作用:
?某个类内方法用synchronized 修饰后,该方法被称为同步方法;
?只要某个线程正在访问同步方法,其他线程欲要访问同步方法就被阻塞,直至线程从同步方法返回前唤醒被阻塞线程,其他线程方可能进入同步方法。
3、在同步方法中使用wait()、notify 和notifyAll()方法
? 一个线程在使用的同步方法中时,可能根据问题的需要,必须使用wait()方法使本线程等待,暂时让出CPU的使用权,并允许其它线程使用这个同步方法。
? 线程如果用完同步方法,应当执行notifyAll()方法通知所有由于使用这个同步方法而处于等待的线程结束等待。
二、实验部分
1、实验目的与要求
(1) 掌握线程同步的概念及实现技术;
(2) 线程综合编程练习
2、实验内容和步骤
实验1:测试程序并进行代码注释。
测试程序1:
l 在Elipse环境下调试教材651页程序14-7,结合程序运行结果理解程序;
l 掌握利用锁对象和条件对象实现的多线程同步技术
package synch; import java.util.*; import java.util.concurrent.locks.*; /** * 一种拥有许多银行帐户的银行,它使用锁来序列化访问。 * @version 1.30 2004-08-01 * @author Cay Horstmann */ public class Bank private final double[] accounts; private Lock bankLock; private Condition sufficientFunds; /** *构建了银行。 * @param 账户数量 * @param 每个账户的初始余额 */ public Bank(int n, double initialBalance) accounts = new double[n]; Arrays.fill(accounts, initialBalance); bankLock = new ReentrantLock(); sufficientFunds = bankLock.newCondition(); /** * 把钱从一个账户转到另一个账户。 * @param 从账户转出 * @param 到账转到 * @param 转帐金额 */ public void transfer(int from, int to, double amount) throws InterruptedException bankLock.lock(); try while (accounts[from] < amount) sufficientFunds.await();//注释掉之后产生死锁现象,都在等待 System.out.print(Thread.currentThread()); accounts[from] -= amount; System.out.printf(" %10.2f from %d to %d", amount, from, to); accounts[to] += amount; System.out.printf(" Total Balance: %10.2f%n", getTotalBalance()); sufficientFunds.signalAll();//注释掉之后产生死锁现象,都在等待 finally bankLock.unlock(); /** *获取所有帐户余额的总和。 * @return 总平衡 */ public double getTotalBalance()//为什么只需要加锁不需要设置对象?没有任何执行不下去的原因,就不需要条件对象。 bankLock.lock(); try double sum = 0; for (double a : accounts) sum += a; return sum; finally bankLock.unlock(); /** * 获取银行中的帐户编号。 * @return 账户数量 */ public int size() return accounts.length;
package synch; /** * 这个程序展示了多线程如何安全地访问数据结构。 * @version 1.31 2015-06-21 * @author Cay Horstmann */ public class SynchBankTest public static final int NACCOUNTS = 100; public static final double INITIAL_BALANCE = 1000; public static final double MAX_AMOUNT = 1000; public static final int DELAY = 10; public static void main(String[] args) Bank bank = new Bank(NACCOUNTS, INITIAL_BALANCE); for (int i = 0; i < NACCOUNTS; i++) int fromAccount = i; Runnable r = () -> try while (true) int toAccount = (int) (bank.size() * Math.random()); double amount = MAX_AMOUNT * Math.random(); bank.transfer(fromAccount, toAccount, amount); Thread.sleep((int) (DELAY * Math.random())); catch (InterruptedException e) ; Thread t = new Thread(r); t.start();
实验结果如下图所示:
测试程序2:
l 在Elipse环境下调试教材655页程序14-8,结合程序运行结果理解程序;
l 掌握synchronized在多线程同步中的应用。
package synch2; import java.util.*; /** * 使用同步原语的具有多个银行帐户的银行 * @version 1.30 2004-08-01 * @author Cay Horstmann */ public class Bank private final double[] accounts; /** * 构建了银行。 * @param 账户数量 * @param 每个账户的初始余额 */ public Bank(int n, double initialBalance) accounts = new double[n]; Arrays.fill(accounts, initialBalance); /** * 把钱从一个账户转到另一个账户。 * @param 从账户转出 * @param 到账转到 * @param 转帐金额 */ public synchronized void transfer(int from, int to, double amount) throws InterruptedException while (accounts[from] < amount) wait();//Object类 System.out.print(Thread.currentThread()); accounts[from] -= amount; System.out.printf(" %10.2f from %d to %d", amount, from, to); accounts[to] += amount; System.out.printf(" Total Balance: %10.2f%n", getTotalBalance()); notifyAll(); /** *获取所有帐户余额的总和。 * @return 总平衡 */ public synchronized double getTotalBalance() double sum = 0; for (double a : accounts) sum += a; return sum; /** * 获取银行中的帐户编号。 * @return 账户数量 */ public int size() return accounts.length;
package synch2; /** * 这个程序展示了多线程如何安全地访问一个数据结构,使用同步方法。 * @version 1.31 2015-06-21 * @author Cay Horstmann */ public class SynchBankTest2 public static final int NACCOUNTS = 100; public static final double INITIAL_BALANCE = 1000; public static final double MAX_AMOUNT = 1000; public static final int DELAY = 10; public static void main(String[] args) Bank bank = new Bank(NACCOUNTS, INITIAL_BALANCE); for (int i = 0; i < NACCOUNTS; i++) int fromAccount = i; Runnable r = () -> try while (true) int toAccount = (int) (bank.size() * Math.random()); double amount = MAX_AMOUNT * Math.random(); bank.transfer(fromAccount, toAccount, amount); Thread.sleep((int) (DELAY * Math.random())); catch (InterruptedException e) ; Thread t = new Thread(r); t.start();
实验结果如下图所示:
测试程序3:
l 在Elipse环境下运行以下程序,结合程序运行结果分析程序存在问题;
l 尝试解决程序中存在问题。
class Cbank private static int s=2000; public static void sub(int m)
int temp=s; temp=temp-m; try Thread.sleep((int)(1000*Math.random()));
catch (InterruptedException e) s=temp; System.out.println("s="+s);
class Customer extends Thread public void run()
for( int i=1; i<=4; i++) Cbank.sub(100);
public class Thread3 public static void main(String args[])
Customer customer1 = new Customer(); Customer customer2 = new Customer(); customer1.start(); customer2.start();
|
存在问题:两个线程各做各的
实验结果如下图所示:
修改后的代码如下:
import javax.sql.rowset.spi.SyncFactory; class Cbank private static int s=2000; public synchronized static void sub(int m) int temp=s; temp=temp-m; try Thread.sleep((int)(1000*Math.random())); catch (InterruptedException e) s=temp; System.out.println("s="+s); class Customer extends Thread public void run() for( int i=1; i<=4; i++) Cbank.sub(100); public class Thread3 public static void main(String args[]) Customer customer1 = new Customer(); Customer customer2 = new Customer(); customer1.start(); customer2.start();
实验结果如下图所示:
实验2 编程练习
利用多线程及同步方法,编写一个程序模拟火车票售票系统,共3个窗口,卖10张票,程序输出结果类似(程序输出不唯一,可以是其他类似结果)。
Thread-0窗口售:第1张票
Thread-0窗口售:第2张票
Thread-1窗口售:第3张票
Thread-2窗口售:第4张票
Thread-2窗口售:第5张票
Thread-1窗口售:第6张票
Thread-0窗口售:第7张票
Thread-2窗口售:第8张票
Thread-1窗口售:第9张票
Thread-0窗口售:第10张票
public class Demo public static void main(String[] args) Mythread mythread = new Mythread(); Thread ticket1 = new Thread(mythread); Thread ticket2 = new Thread(mythread); Thread ticket3 = new Thread(mythread); ticket1.start(); ticket2.start(); ticket3.start(); class Mythread implements Runnable int ticket = 1; boolean flag = true; @Override public void run() while (flag) try Thread.sleep(500); catch (InterruptedException e) // TODO Auto-generated catch block e.printStackTrace(); synchronized (this) if (ticket <= 10) System.out.println(Thread.currentThread().getName() + "窗口售:第" + ticket + "张票"); ticket++; if (ticket > 10) flag = false;
实验结果如下图所示:
三、实验总结
本周学习了同步线程的相关问题,了解了并发多线程的两种解决方法。还有在同步方法中使用wait()、notify 和notifyAll()方法,使用wait()方法使本线程等待,暂时让出CPU的使用权,并允许其它线程使用这个同步方法,这周学习的收获很多。
201771010137赵栋《面向对象程序设计(java)》第十三周学习总结(代码片段)
第一部分:理论知识 第11章事件处理(事件处理基础; 动作; 鼠标事件;AWT事件继承层次)1.事件源(eventsource):能够产生事件的对象都可以成为事件源,如文本... 查看详情
赵栋201771010137《面向对象程序设计(java)》第二周学习总结
第一部分理论知识总结第三章Java的基本程序设计结构1.数据类型:一共有八种基本类型,4中整形,2种浮点类型,1种用于表示Unicode编码的字符单元的字符类型char和1种用于表示真值的boolean类型。2.变量:变量名必须是一个以... 查看详情
赵栋201771010137《面向对象程序设计(java)》第六周学习总结(代码片段)
1、实验目的 在软件开发中,通过继承机制,可以利用已有的数据类型来定义新的数据类型。所定义的新的数据类型不仅拥有新定义的成员,而且还同时拥有旧的成员。因此,类的继承性使所建立的软件具有开放性... 查看详情
赵栋201771010137第三周学习总结(代码片段)
自主学习任务1、复习课程第1-3章学习内容2、回顾实验三中实验题1《2018秋季西北师范大学面向对象程序设计(Java)(ch1-ch3)测试题1》,对照测试题参考答案,反思修改本人答卷的错误内容;3、完成实验2与实验3;4、修改置顶... 查看详情
201771010137赵栋《面向对象程序设计(java)》第十七周学习总结(代码片段)
(1)综合掌握java基本程序结构;(2) 综合掌握java面向对象程序设计特点;(3)综合掌握javaGUI 程序设计结构;(4)综合掌握java多线程编程模型;(5)综合编程练习。2、实验内容和步骤任务1:填写课程课后调查问卷,网址:htt... 查看详情
201771010137赵栋《面向对象程序设计(java)》第十七周学习总结(代码片段)
一、理论部分1、多线程并发执行中的问题i.多个线程相对执行的顺序是不确定的。ii.线程执行顺序的不确定性会产生执行结果的不确定性。iii.在多线程对共享数据操作时常常会产生这种不确定性。2、线程的同步-多线程并发运行... 查看详情
201771010137赵栋《第八周学习总结》(代码片段)
实验六接口的定义与使用实验时间2018-10-181、实验目的与要求(1)接口定义:接口不是类,而是对类的一组需求描述,这些类要遵从接口描述的统一格式进行定义;由常量和一组抽象方法组成;接口中的是所有方法自动地属于public... 查看详情
201771010137赵栋《第九周学习总结》(代码片段)
1、实验目的与要求(1)掌握java异常处理技术;(2)了解断言的用法;(3)了解日志的用途;(4)掌握程序基础调试技巧;1.Throwable类中的常用方法注意:catch关键字后面括号中的Exception类型的参数e。Exception就是try代码块传递给catch代码块... 查看详情
东文财赵栋罗松201771010106《面向对象程序设计(java)》实验14
实验十四 Swing图形界面组件实验时间20178-11-29一、知识部分1.模型-视图-控制器模式模型:储存内容 视图:显示内容 控制器:处理用户输入2.布局管理2-1.流布局管理器(FlowLayout) JPanel对象的默认布局管理器为FlowLayout... 查看详情
08-面向对象----j
一面向对象的程序设计的由来请参考:http://www.cnblogs.com/linhaifeng/articles/6428835.html二什么是面向对象的程序设计及为什么要有它面向过程的程序设计的核心是过程,过程即解决问题的步骤,面向过程的设计就好比精心设计好一条... 查看详情
2020面向对象设计与构造第三单元博客总结(代码片段)
面向对象设计与构造第三单元总结一、JML规格化设计JML,全称TheJavaModelingLanguage,是用于对Java程序进行规格化描述的注释性质语言。笔者在本文总结了常见的JML语法描述。1.注释结构在注释行或注释块中,以@开头的行被认作JML注... 查看详情
面向对象
老王和隔壁的美女猜数字,一共有四次机会,猜到了就有特殊奖励publicclasstest{publicstaticvoidmain(String[]args){inti=(int)(Math.random()*10);Scannerinput=newScanner(System.in);for(intj=0;j<4;j++){System.out.println("老王第"+(j+1)+"次 查看详情
201771010126王燕《面向对象程序设计(java)》第二周学习总结
201771010126王燕《面向对象程序设计(java)》第二周学习总结一.理论知识学习部分3.1j简单的java应用程序标识符由字母、下划线、美元符号和数字组成,且第一个符号不能为数字。标识符可用作:类名、变量名、方法名、数组名... 查看详情
201823072019-2020-1《数据结构与面向对象程序设计》实验1报告
课程:《程序设计与数据结构》班级:1823姓名:王美皓学号:20182322实验教师:王美皓实验日期:2019年9月9日必修/选修:必修1.实验内容基于命令行和IDE(IntelljIDEA简易教程](http://www.cnblogs.com/rocedu/p/4421202.html)进行简单的Java程... 查看详情
杨其菊201771010134《面向对象程序设计(java)》第三周学习总结(代码片段)
《面向对象程序设计(Java)》第三周学习总结第一部分:理论知识 这周课程没有新进度,由于感觉对基础语法的不熟悉,复习了一遍前三章的细碎知识,学到一些之前不知道的原理: 1.计算机高级语言按程序的... 查看详情
c++面向对象的主要体现是啥?
...得对我有用的话,我会再追加分。)C++面向对象是表现在程序设计的过程上,它是突破了C的结构化设计而出现的完全以实际问题为入手点的。C++的面向对象3个特性:1.封装。2.继承。3.多态。尤其重要的是,它采用类的设计,杜... 查看详情
徐思201771010132《面向对象程序设计(java)》第十二周学习总结(代码片段)
一、理论知识部分Java的抽象窗口工具箱(AbstractWindowToolkit,AWT)包含在java.awt包中,它提供了许多用来设计GUI的组件类和容器类。大部分AWT组件都有其Swing的等价组件,Swing组件的名字一般是在AWT组件名前面添加一个字母“J”。通... 查看详情
201771010110孔维滢《面向对象程序设计(java)》第十二周学习总结
理论知识部分1.Java的抽象窗口工具箱(AbstractWindowToolkit,AWT)包含在java.awt包中,它提供了许多用来设计GUI的组件类和容器类。2.Swing用户界面库是非基于对等体的GUI工具箱。Swing类库被放在javax.swing包里。3.大部分AWT组件都有其Swing... 查看详情