第十章多线程教案.doc
《第十章多线程教案.doc》由会员分享,可在线阅读,更多相关《第十章多线程教案.doc(20页珍藏版)》请在文库网上搜索。
1、第十章 多线程教案名称:教案大小:教案类型:WORD文档星级评定:教案简介:本讲介绍了Java线程的一些基本知识和简单应用,通过对线程简介,阐明了线程与进程的区别,通过描述线程的概念模型的基本原理以及线程体的构造方法和应用实例,讲解了线程的基本特性和线程的不同状态的转换关系和调用方法,明确了线程的使用方法,然后,我们又进一步讲述了线程的几种调度策略,在不同的调度策略下优先级的作用。以及如何进行基本的线程的控制,线程的重点和难点在于多线程的互斥与同步,首先我们必须明白互斥锁的概念和作用,如何使用互斥锁来控制和处理多线程的同步问题。下载一【课前思考】1. 什么是线程?它和进程有什么区别?适用方向是
2、什么?2. Java的线程是如何实现的?3. Java的线程是如何调度的?4. Java中的多线程有什么特点?同步和互斥的原理是如何实现的?【学习目标】学习java中线程的使用,掌握线程的调度和控制方法,清楚地理解多线程的互斥和同步的实现原理,以及多线程的应用。【学习指南】掌握线程之间的相互调度关系,尤其是通过线程睡眠来使其它线程获得执行机会的机制,以及互斥和同步的实现机制。【难 重 点】 1. 多线程的调度和控制。2. 多线程的互斥和同步。【知 识 点】10.1 线程简介10.1.1 线程的概念模型10.1.2 线程体10.1.3线程的调度10.1.4基本的线程控制10.2多线程的互斥与同步
3、10.2.1互斥锁10.2.2多线程的同步第十章 多线程101 线程简介随着计算机的飞速发展,个人计算机上的操作系统也纷纷采用多任务和分时设计,将早期只有大型计算机才具有的系统特性带到了个人计算机系统中。一般可以在同一时间内执行多个程序的操作系统都有进程的概念。一个进程就是一个执行中的程序,而每一个进程都有自己独立的一块内存空间、一组系统资源。在进程概念中,每一个进程的内部数据和状态都是完全独立的。Java程序通过流控制来执行程序流,程序中单个顺序的流控制称为线程,多线程则指的是在单个程序中可以同时运行多个不同的线程,执行不同的任务。多线程意味着一个程序的多行语句可以看上去几乎在同一时间内同时
4、运行。线程与进程相似,是一段完成某个特定功能的代码,是程序中单个顺序的流控制;但与进程不同的是,同类的多个线程是共享一块内存空间和一组系统资源,而线程本身的数据通常只有微处理器的寄存器数据,以及一个供程序执行时使用的堆栈。所以系统在产生一个线程,或者在各个线程之间切换时,负担要比进程小的多,正因如此,线程被称为轻负荷进程(light-weight process)。一个进程中可以包含多个线程。一个线程是一个程序内部的顺序控制流。1. 进程:每个进程都有独立的代码和数据空间(进程上下文) ,进程切换的开销大。 2. 线程:轻量的进程,同一类线程共享代码和数据空间,每个线程有独立的运行栈和程序计数
5、器(PC),线程切换的开销小。 3. 多进程:在操作系统中,能同时运行多个任务程序。 4. 多线程:在同一应用程序中,有多个顺序流同时执行。1011 线程的概念模型Java内在支持多线程,它的所有类都是在多线程下定义的,Java利用多线程使整个系统成为异步系统。Java中的线程由三部分组成,如图6.1所示。1. 虚拟的CPU,封装在java.lang.Thread类中。2. CPU所执行的代码,传递给Thread类。3. CPU所处理的数据,传递给Thread类。图10.1线程1012 线程体Java的线程是通过java.lang.Thread类来实现的。当我们生成一个Thread类的对象之后
6、,一个新的线程就产生了。此线程实例表示Java解释器中的真正的线程,通过它可以启动线程、终止线程、线程挂起等,每个线程都是通过类Thread在Java的软件包Java.lang中定义,它的构造方法为:public Thread (ThreadGroup group,Runnable target,String name);其中,group 指明该线程所属的线程组;target实际执行线程体的目标对象,它必须实现接口Runnable; name为线程名。Java中的每个线程都有自己的名称,Java提供了不同Thread类构造器,允许给线程指定名称。如果name为null时,则Java自动提供唯一
7、的名称。当上述构造方法的某个参数为null时,我们可得到下面的几个构造方法:public Thread ();public Thread (Runnable target);public Thread (Runnable target,String name);public Thread (String name);public Thread (ThreadGroup group,Runnable target);public Thread (ThreadGroup group,String name);一个类声明实现Runnable接口就可以充当线程体,在接口Runnable中只定义了一个方
8、法 run(): public void run();任何实现接口Runnable的对象都可以作为一个线程的目标对象,类Thread本身也实现了接口Runnable,因此我们可以通过两种方法实现线程体。(一)定义一个线程类,它继承线程类Thread并重写其中的方法 run(),这时在初始化这个类的实例时,目标target可为null,表示由这个实例对来执行线程体。由于Java只支持单重继承,用这种方法定义的类不能再继承其它父类。(二)提供一个实现接口Runnable的类作为一个线程的目标对象,在初始化一个Thread类或者Thread子类的线程对象时,把目标对象传递给这个线程实例,由该目标对象
9、提供线程体 run()。这时,实现接口Runnable的类仍然可以继承其它父类。每个线程都是通过某个特定Thread对象的方法run( )来完成其操作的,方法run( )称为线程体。图6.2表示了java线程的不同状态以及状态之间转换所调用的方法。图10.2 线程的状态 1. 创建状态(new Thread)执行下列语句时,线程就处于创建状态:Thread myThread = new MyThreadClass( );当一个线程处于创建状态时,它仅仅是一个空的线程对象,系统不为它分配资源。2. 可运行状态( Runnable )Thread myThread = new MyThreadCl
10、ass( );myThread.start( );当一个线程处于可运行状态时,系统为这个线程分配了它需的系统资源,安排其运行并调用线程运行方法,这样就使得该线程处于可运行( Runnable )状态。需要注意的是这一状态并不是运行中状态(Running ),因为线程也许实际上并未真正运行。由于很多计算机都是单处理器的,所以要在同一时刻运行所有的处于可运行状态的线程是不可能的,Java的运行系统必须实现调度来保证这些线程共享处理器。3. 不可运行状态(Not Runnable)进入不可运行状态的原因有如下几条:1) 调用了sleep()方法;2) 调用了suspend()方法;3) 为等候一个条
11、件变量,线程调用wait()方法;4) 输入输出流中发生线程阻塞;不可运行状态也称为阻塞状态(Blocked)。因为某种原因(输入/输出、等待消息或其它阻塞情况),系统不能执行线程的状态。这时即使处理器空闲,也不能执行该线程。4. 死亡状态(Dead)线程的终止一般可通过两种方法实现:自然撤消(线程执行完)或是被停止(调用stop()方法)。目前不推荐通过调用stop()来终止线程的执行,而是让线程执行完。线程体的构造任何实现接口Runnable的对象都可以作为一个线程的目标对象,上面已讲过构造线程体有两种方法,下面通过实例来说明如何构造线程体的。例101 通过继承类Thread构造线程体cl
12、ass SimpleThread extends Thread public SimpleThread(String str) super(str); /调用其父类的构造方法public void run() /重写run方法for (int i = 0; i 10; i+) System.out.println(i + + getName();/打印次数和线程的名字try sleep(int)(Math.random() * 1000);/线程睡眠,把控制权交出去 catch (InterruptedException e) System.out.println(DONE! + getNam
13、e();/线程执行结束public class TwoThreadsTest public static void main (String args) new SimpleThread(First).start();/第一个线程的名字为Firstnew SimpleThread(Second).start();/第二个线程的名字为Second 运行结果:0 First0 Second1 Second1 First2 First2 Second3 Second3 First4 First4 Second5 First5 Second6 Second6 First7 First7 Second8
14、 Second9 Second8 FirstDONE! Second9 FirstDONE! First仔细分析一下运行结果,会发现两个线程是交错运行的,感觉就象是两个线程在同时运行。但是实际上一台计算机通常就只有一个CPU,在某个时刻只能是只有一个线程在运行,而java语言在设计时就充分考虑到线程的并发调度执行。对于程序员来说,在编程时要注意给每个线程执行的时间和机会,主要是通过让线程睡眠的办法(调用sleep()方法)来让当前线程暂停执行,然后由其它线程来争夺执行的机会。如果上面的程序中没有用到sleep()方法,则就是第一个线程先执行完毕,然后第二个线程再执行完毕。所以用活sleep()
15、方法是学习线程的一个关键。例102 通过接口构造线程体public class Clock extends java.applet.Applet implements Runnable /实现接口Thread clockThread; public void start() /该方法是Applet的方法,不是线程的方法if (clockThread = null) clockThread = new Thread(this, Clock);/*线程体是Clock对象本身,线程名字为Clock*/clockThread.start(); /启动线程 public void run() /run(
16、)方法中是线程执行的内容while (clockThread != null) repaint(); /刷新显示画面try clockThread.sleep(1000); /睡眠1秒,即每隔1秒执行一次 catch (InterruptedException e) public void paint(Graphics g) Date now = new Date(); /获得当前的时间对象g.drawString(now.getHours() + : + now.getMinutes()+ : +now.getSeconds(), 5, 10);/显示当前时间public void stop
17、() /该方法是Applet的方法,不是线程的方法clockThread.stop(); clockThread = null; 上面这个例子是通过每隔1秒种就执行线程的刷新画面功能,显示当前的时间;看起来的效果就是一个时钟,每隔1秒就变化一次。由于采用的是实现接口Runnable的方式,所以该类Clock还继承了Applet, Clock就可以Applet的方式运行。构造线程体的两种方法的比较:1. 使用Runnable接口1) 可以将CPU,代码和数据分开,形成清晰的模型;2) 还可以从其他类继承;3) 保持程序风格的一致性。2. 直接继承Thread类1) 不能再从其他类继承;2) 编写
18、简单,可以直接操纵线程,无需使用Thread.currentThread()。1013 线程的调度Java提供一个线程调度器来监控程序中启动后进入就绪状态的所有线程。线程调度器按照线程的优先级决定应调度哪些线程来执行。线程调度器按线程的优先级高低选择高优先级线程(进入运行中状态)执行,同时线程调度是抢先式调度,即如果在当前线程执行过程中,一个更高优先级的线程进入可运行状态,则这个线程立即被调度执行。线程的优先级线程的优先级用数字来表示,范围从1到10,即Thread.MIN_PRIORITY到Thread.MAX_PRIORITY。一个线程的缺省优先级是5,即Thread.NORM_PRIOR
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 第十 多线程 教案