欢迎来到三一办公! | 帮助中心 三一办公31ppt.com(应用文档模板下载平台)
三一办公
全部分类
  • 办公文档>
  • PPT模板>
  • 建筑/施工/环境>
  • 毕业设计>
  • 工程图纸>
  • 教育教学>
  • 素材源码>
  • 生活休闲>
  • 临时分类>
  • ImageVerifierCode 换一换
    首页 三一办公 > 资源分类 > PPT文档下载  

    【教学课件】第10章多线程.ppt

    • 资源ID:4879134       资源大小:398.47KB        全文页数:76页
    • 资源格式: PPT        下载积分:15金币
    快捷下载 游客一键下载
    会员登录下载
    三方登录下载: 微信开放平台登录 QQ登录  
    下载资源需要15金币
    邮箱/手机:
    温馨提示:
    用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)
    支付方式: 支付宝    微信支付   
    验证码:   换一换

    加入VIP免费专享
     
    账号:
    密码:
    验证码:   换一换
      忘记密码?
        
    友情提示
    2、PDF文件下载后,可能会被浏览器默认打开,此种情况可以点击浏览器菜单,保存网页到桌面,就可以正常下载了。
    3、本站不支持迅雷下载,请使用电脑自带的IE浏览器,或者360浏览器、谷歌浏览器下载即可。
    4、本站资源下载后的文档和图纸-无水印,预览文档经过压缩,下载后原文更清晰。
    5、试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。

    【教学课件】第10章多线程.ppt

    第10章 多线程,学习目标 本章要点上机练习 习 题,学习目标,在计算机处理任务时,往往需要操作系统同时处理多个任务,这些任务就由操作系统本身和CPU来控制。有时也需要自己创建多个任务来让系统同时完成,这些用户自己定制的任务都是非常小的程序段,称为线程。本章介绍线程的基本概念和多线程技术,详细讲解线程的创建,如何实现多线程,还介绍线程同步、线程死锁和线程调度等控制技术。,本章要点,线程的生命周期多线程技术线程的创建和实现多线程线程的同步和死锁,多线程技术的基本概念,多线程是Java程序的一个重要特征,线程本来是操作系统中的概念,Java将这一概念引入到程序设计语言中,让开发人员利用线程机制编写多线程程序,使系统能够同时运行多个执行体,从而加快程序的响应速度,提高电脑资源的利用率。本节主要介绍多线程机制的一些基本概念。,多线程技术的基本概念,什么是线程线程的生命周期 多线程技术,什么是线程,线程和进程类似,是一段完成特定功能的代码。线程是程序中单个程序的控制流,也是一个进程内的基本调度单位。线程和进程一样拥有独立的执行控制,并由操作系统负责调度。,什么是线程,同一个进程可以包含多个线程,这些线程共享属于该进程的一块内存空间和一组系统资源,而线程自身的数据通常只有CPU的寄存器数据,以及一个供程序执行时使用的堆栈,系统在产生一个线程,或者在各个线程之间切换时,负荷比进程小得多。,什么是线程,此外,由于线程知识在单个进程的作用域内活动,所以线程之间的通信也比进程简单。线程的实现要依赖操作系统,现在一般操作系统都支持线程技术。,什么是线程,一个或更多的线程构成了一个进程。一个线程或执行上下文由一个虚拟处理机、CPU执行的代码和代码操作的数据3个主要部分组成。,线程执行的主要组成部分,什么是线程,代码可以由多个线程共享,也可以不被共享,这些都和数据是独立的。两个线程如果执行同一个类的实例代码,则它们可以共享相同的代码。,什么是线程,类似地,数据可以由多个线程共享,也可以不被共享,这些都和代码是独立的。两个线程如果共享对一个公共对象的存取,则它们可以共享相同的数据。,什么是线程,在Java编程中,虚拟处理机封装在Thread类的一个实例。构造线程时,代码和数据是由传递给它的构造方法的对象指定的。,线程的生命周期,每个Java程序都有一个默认的主线程,对于Java Applicatin应用程序,主线程就是main方法执行的指令序列,对于Java Applet,主线程指挥浏览器装载并执行Java Applet。线程在它完整的生命周期中包括新建、就绪、运行、阻塞和死亡5种状态。,线程的生命周期,新建(New):代表线程的对象已经被初始化,但尚未运行run方法。就绪(Runnable):线程正在运行run方法,但这只说明线程目前处于的状态。如果系统没有能力抽出CPU执行时间给线程,线程就“不执行”,这里的“不执行”不代表“阻塞”或“死亡”。,线程的生命周期,运行(Running):线程获得了CPU资源正在执行任务,将一直运行到结束,除非此时它自动放弃资源或有更加高优先级的线程进入。,线程的生命周期,阻塞(Blcked):线程是可以执行的,但由于某些因素的阻碍处于停滞状态,系统排程器略过了应给线程的CPU执行时间。死亡(Dead):线程的正式结束方式,run方法执行完毕并返回。,多线程技术,多线程是这样一种机制,它允许在程序中并发执行多个指令流,每个指令流都称为一个线程,线程彼此间互相独立。,多线程技术,线程和进程一样拥有独立的执行控制,由操作系统负责调度,区别在于线程没有独立的存储空间,而是和所属进程中的其它线程共享一个存储空间,这使得线程间的通信比进程简单。,多线程技术,多个线程的执行是并发的,也就是在逻辑上“同时”,而不管是否是物理上的“同时”。如果系统只有一个CPU,那么真正的“同时”是不可能的,但是由于CPU的速度非常快,用户感觉不到其中的区别,因此也不用关心它,只需要设想各个线程是同时执行即可。,多线程技术,多线程和传统的单线程在程序设计上最大的区别在于,由于各个线程的控制流彼此独立,使得各个线程之间的代码是无序执行的。,多线程技术的实现,在了解了线程的基本概念之后,本节将继续深入,介绍如何创建线程,如何实现多线程,线程同步、死锁和调度等等多线程技术。线程的创建 如何实现多线程,多线程技术的实现,线程同步 线程死锁 线程调度 应用举例抢硬币,线程的创建,为了创建一个新的线程,必须指明这个线程所要执行的代码,Java是通过类做到这一点的。作为一个面向对象的程序设计语言,Java提供了类来进行多线程编程,这个类提供了大量的方法给开发人员控制自己的各个线程,以后的讨论都将围绕这个类进行。,线程的创建,那么如何提供给Java开发人员线程执行的代码呢?通过Thread类来实现。主要有两种方法。继承Thread类,覆盖方法run()实现Runnable接口,继承Thread类,覆盖方法run(),Thread类最重要的方法是run()方法,它为Thread类的start()方法所调用,并提供线程所要执行的代码。为了指定自己新的代码,只需要覆盖它即可,即在创建的Thread类的子类中重写run(),加入线程所要执行的代码。,继承Thread类,覆盖方法run(),该方法的步骤是:从Thread类派生一个类,覆盖Thread类中的run 方法,然后创建该子类的对象,再调用start方法启动本线程。【例10-1】线程的创建(光盘:源文件第10章例10-1.txt)。,类“DrawLine”运行结果,继承Thread类,覆盖方法run(),这种方法简单明了,符合大家的习惯。但是,它也有一个很大的缺点,那就是如果类已经从一个类继承,则无法再继承Thread类。如果又不想建立一个新的类,应该怎么办呢?这时可以采用另外一种方法来实现:实现Runnable接口。,实现Runnable接口,Runnable接口只有一个方法run(),可以通过声明自己的类实现Runnable接口并提供这一方法,并将线程代码写入其中,就完成了这一部分的任务。但是Runnable接口并没有任何对线程的支持,还必须创建Thread类的实例,这一点通过Thread类的构造方法publicThread(Runnable target)来实现。,实现Runnable接口,【例10-2】通过实现Runnable接口来创建一个线程(光盘:源文件第10章例10-2.txt)。,实现Runnable接口,严格地说,创建Thread子类的实例也是可行的,但是必须注意的是,该子类必须没有覆盖Thread类的run方法,否则该线程执行的将是子类的run方法,而不是用以实现Runnable 接口的类run方法,对此可以试验一下。,实现Runnable接口,使用Runnable接口来实现多线程使得开发人员能够在一个类中包容所有的代码,有利于封装,但是其缺点在于只能使用一套代码。若想创建多个线程并使各个线程执行不同的代码,则仍必须额外创建类,在大多数情况下这也许还不如直接用多个类分别继承Thread更为恰当。,如何实现多线程,Java语言提供的实现多线程应用程序的方法主要有如下两种。继承Thread类的多线程程序设计方法是使应用程序类继承Thread类并且在该类的run方法中实现并发性处理过程。,如何实现多线程,多线程对象实现Runnable接口并且在该类中定义用于启动线程的run方法。后一种方法的好处在于多线程应用对象可以继承其它对象而不是必须继承Thread类,从而能够增加类定义的逻辑性。,如何实现多线程,这两种方法解决了Java实现多线程的技术障碍,是多线程技术实现的核心部分。下面通过例子演示如何使用这两种方法来实现多线程技术。【例10-3】多线程的实现(光盘:源文件第10章例10-3.txt)。,类“Consumer”运行结果,如何实现多线程,从上面的程序代码可以看出:多线程类Consumer继承Java语言中的线程类Thread,并且在main方法中创建3个Consumer对象的实例。当调用对象实例的start方法时,自动调用Consumer类中定义的run方法启动对象线程运行。,如何实现多线程,线程运行的结果是每间隔nTime时间打印出对象实例中的字符串成员变量strConsumer的内容。如果采用实现Runnable接口来实现多线程应用程序,则代码如下:,如何实现多线程,从上述代码可以看出:该类实现了Runnable接口并且在该类中定义了run方法。这种多线程应用程序的实现方式与继承Thread类的多线程应用程序的重要区别在于启动多线程对象的方法设计方法不同。在上述代码中,通过创建Thread对象实例并且将应用对象作为创建Thread类实例的参数。,线程同步,Java应用程序的多个线程共享同一进程的数据资源,多个用户线程在并发运行过程中可能同时访问具有共享的内容。在Java中定义了线程同步的概念,实现对共享资源的一致性维护。,线程同步,下面以移动通信计费系统中线程间同步控制方法,说明Java语言中多线程同步方式的实现过程。在没有多线程同步控制策略条件下,客户账户类定义代码如下所示:,线程同步,public class RegisterAccountfloat fBalance;/客户缴费方法public void deposit(float fFees)fBalance+=fFees;,线程同步,/通话计费方法public void withdraw(float fFees)fBalance-=fFees;,线程同步,这段代码也许会被认为:完全能够满足计费系统实际的需要。确实,在单线程环境下该程序确实是可靠的。但是,多进程并发运行的情况是怎样的呢?,线程同步,假设发生这种情况:客户在客户服务中心进行缴费的同时也正在利用移动电话进行通话,客户通话结束时计费系统启动计费进程,而同时客户服务中心的工作人员也提交缴费进程运行。可以看到,如果发生这种情况,对客户账户的处理是有冲突的。,线程同步,如何解决这种问题呢?很简单,在RegisterAccount类方法的定义中加上用于标识同步方法的关键字synchronized。这样,在同步方法执行过程中该方法涉及的共享资源将被加上共享锁,以确保在方法运行期间只有该方法能够对共享资源进行访问,直到该方法的线程运行结束打开共享锁,其它线程才能够访问这些共享资源。,线程同步,在共享锁没有打开的时候,其它访问共享资源的线程处于阻塞状态。进行线程同步控制后的RegisterAccount类定义如下面代码所示:,线程同步,public class RegisterAccount float fBalance;public synchronized void deposit(float fFees)fBalance+=fFees;public synchronized void withdraw(float fFees)fBalance-=fFees;,线程同步,从以上代码可以看出:在对共享资源进行访问的方法访问属性关键字(public)后附加同步定义关键字synchronized,使得同步方法在对共享资源访问的时候,为这些资源附加共享锁来控制方法执行期间的资源独占性,实现了应用系统数据资源的一致性管理和维护。,线程死锁,死锁是这样一种情形:多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。死锁的根源 隐性死锁,死锁的根源,导致死锁的根源在于不适当地运用synchronized关键字来管理线程对特定对象的访问。前面介绍过,synchronized关键字的作用是,确保在某个时刻只有一个线程被允许执行特定的代码块。因此,被允许执行的线程首先必须拥有对变量或对象的访问权。,死锁的根源,当线程访问对象时,线程会给对象加锁,而这个锁导致其它也想访问同一对象的线程被阻塞,直至第一个线程释放它加在对象上的锁。由于这个原因,在使用synchronized关键词时,很容易出现两个线程互相等待对方做出某个动作的情形。,隐性死锁,隐性死锁由于不规范的编程方式引起,但不一定每次测试运行时都会出现程序死锁的情形。由于这个原因,一些隐性死锁可能要到应用正式发布之后才会被发现,因此它的危害性比普通死锁更大。下面介绍两种导致隐性死锁的情况:加锁次序和占有并等待。,隐性死锁,1)加锁次序当多个并发的线程分别试图同时占有两个锁时,会出现加锁次序冲突的情形。如果一个线程占有了另一个线程必需的锁,就有可能出现死锁。考虑下面的情形,ThreadA和ThreadB两个线程分别需要同时拥有lock_1、lock_2两个锁,加锁过程可能如下:,隐性死锁,ThreadA获得lock_1。ThreadA被抢占,调度程序转到ThreadB。ThreadB获得lock_2。ThreadB被抢占,调度程序转到ThreadA。ThreadA试图获得lock_2,但lock_2被ThreadB占有,所以ThreadA阻塞。,隐性死锁,调度程序转到ThreadB。ThreadB试图获得lock_1,但lock_1被ThreadA占有,所以ThreadB阻塞。ThreadA和ThreadB死锁。,隐性死锁,必须指出的是,在代码丝毫不做变动的情况下,有些时候上述死锁过程不会出现,调度程序可能让其中一个线程同时获得lock_1和lock_2两个锁,即线程获取两个锁的过程没有被中断。在这种情形下,常规的死锁检测很难确定错误所在。,隐性死锁,2)占有并等待如果一个线程获得了一个锁之后还要等待来自另一个线程的通知,可能出现另一种隐性死锁。以上代码中,Producer向队列加入一项新的内容后通知Consumer,以便它处理新的内容。,隐性死锁,问题在于,Consumer可能保持加在队列上的锁,阻止Producer访问队列,甚至在Consumer等待Producer的通知时也会继续保持锁。这样,由于Producer不能向队列添加新的内容,而Consumer却在等待Producer加入新内容的通知,结果就导致死锁。,隐性死锁,在等待时占有的锁是一种隐性的死锁,这是因为事情可能按照比较理想的情况发展Producer线程不需要被Consumer占据的锁。尽管如此,除非有绝对可靠的理由肯定Producer线程永远不需要该锁,否则这种编程方式仍是不安全的。,隐性死锁,有时“占有并等待”还可能引发一连串的线程等待。要改正这个错误,只需修改Consumer类,把wait()移出synchronized()即可。,线程调度,对于多线程程序,每个线程的重要程度不尽相同,如多个线程在等待获得CPU时间时,往往需要优先级高的线程优先抢占到CPU时间得以执行;又如多个线程交替执行时,优先级决定了级别高的线程得到CPU的次数多一些且时间长一些。这样,高优先级的线程处理的任务效率就高一些。,线程调度,线程通常是抢占式的而不需要时间片分配进程。抢占式调度模型就是许多线程属于可以运行状态,但实际上只有一个线程在运行。该线程一直运行到它终止进入可运行状态或是另一个具有更高优先级的线程变成可运行状态。,线程调度,后一种情况下,低优先级的线程被高优先级的线程抢占,高优先级的线程获得运行的机会。线程可以因为各种各样的原因进入阻塞状态。,线程调度,所有可运行的线程根据优先级保持在不同的池中。一旦被阻塞的线程返回可运行状态,它将会被放回适当的可运行池中。非空最高优先级池中的线程将获得CPU时间。Java中线程的优先级从低到高以整数110表示,共分为10级,设置优先级是通过调用线程对象的setPriority()方法。,线程调度,Java的Object类提供了3个方法来实现线程的调度和通信。wait()等待:当前线程放弃监视器并进入睡眠状态,直到其他线程进入同一监视器并调用notify()为止。,线程调度,notify()唤醒:唤醒同一对象监视器中调用wait()的第一个线程。notifyall()唤醒所有的线程:唤醒同一对象监视器中调用wait()的所有线程。定义方法:,线程调度,public final void wait():使当前正在执行的线程暂时挂起,从执行状态退回阻塞状态,放弃所占用的资源。public final void notify():从管程队列中选择优先级别最高的一个被挂起的线程并唤醒它,使其占用该管程及其相应资源。,线程调度,这些方法都是不可重载的方法,并且只能在同步方法中调用。,应用举例抢硬币,本节的应用举例将通过线程同步的方法实现一个抢硬币的游戏(光盘:源文件第10章应用举例ThreadSyncDemo.txt)。,类“ThreadSyncDemo”运行结果,上 机 练 习,本次练习将创建两个线程,每个线程均输出“你好”,接着输出线程名及消息数字,每个线程输出5次“你好”,显示出这些信息(光盘:源文件第10章上机练习.txt)。,类“Test_Runnable”运行结果,习 题,编写两个线程,其中一个线程可以用来计算2-10000000之间素数的个数。另一个线程用来计算10000000-20000000之间的素数,问:哪个线程先执行完?2-10000000之间的素数多还是10000000-20000000之间的素数多?,习 题,(4)设置m=“HelloJava”,请写出下列方法的值。m.length()m.contact(“mickle”)m.substring(2,5);m.replace(J,a);(5)设定10个字符串,打印出以字母“b”开头的字符串。,

    注意事项

    本文(【教学课件】第10章多线程.ppt)为本站会员(小飞机)主动上传,三一办公仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知三一办公(点击联系客服),我们立即给予删除!

    温馨提示:如果因为网速或其他原因下载失败请重新下载,重复下载不扣分。




    备案号:宁ICP备20000045号-2

    经营许可证:宁B2-20210002

    宁公网安备 64010402000987号

    三一办公
    收起
    展开