浏览 1160 次
锁定老帖子 主题:JAVA 线程编程 ----两个线程程序
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
|
|
---|---|
作者 | 正文 |
发表时间:2009-11-20
最后修改:2009-11-20
Java的内置线程支持 关于Java好的方面是它有内置的写多线程的支持。Java的设计者知道多线程编程的价值,所以聪明决定在Java的核心部分就决定直接支持线程。在第7章“并发存取对象和变量”就阐述了在Java语言中,synchronized关键字如何被用来锁住对象和类来控制并发存取数据。Thread和ThreadGroup类就在java.lang包的核心API中。通过wait( ) notify( ) 方法的支持,Java中所有类的父类Object具有线程间通信的能力。即使当前操作系统不支持线程概念,一个写好的JVM也可以模拟多线程环境。在Java中,线程的支持不是事后的,而是从一开始设计时就包括进来的。 第2章:一个简单的两个线程例子 概述 这章展示了创建并在一个小的Java应用中运行线程是如何简单。第一个线程是总是有JVM创建的“main”线程,该线程开始运行程序。这个主线程然后创建第二个线程。每个线程将在控制台打印各自信息,以此来证明他们好像以同步方式运行。 这章中例子创建一个线程的步骤如下:
继承java.lang.thread类 在JavaVM中每个线程与java.lang.Thread类的一个实例想关联。这些Thread对象作为与底层操作系统线程交互的接口。通过类中的方法,可以对线程进行开始,停止,中断,命名、设置优先级和查询有关当前状态。 注意:有两种方式可创建一个允许线程运行与其中的类。一种方式是继承Thread类。另一种是继承任意类并且实现Runnable接口。为了说明的原因,继承Thread类是最简单的方法,本书开始就是用该方法。在实际中,实现Runnable接口工作地更好些。 在这个例子中,创建一个新线程的第一步是继承java.lang.Thread类: public class TwoThread extends Thread { // ... } TwoThread子类是一个(IS-A) Thread,并且继承了父类的protected 和public成员。TwoThread除了从父类中继承的其他行为之外,它还能被开始、停止、中断、命名、设置优先级以及查询线程当前状态。 重写run()方法 继承Thread类之后,下一步就是重写run()方法,因为Thread类中方法什么也没做: Public void run() { } 当一个新线程开始运行,进入该线程的入口就是run()方法。run()中的第一条语句就是新线程执行的第一条语句。线程执行的每条语句都包含在run()方法中或包含在被run()方法直接或间接调用的其他方法中。从run()被调用之前到run()返回,新线程被认为是活着的。run()返回之后,线程就死掉了。当一个线程死了之后不能重新开始。 在本章的例子中,run()方法被重写为迭代10次并且每次打印New thread信息: public void run() { for ( int i = 0; i < 10; i++ ) { System.out.println(“New thread”); } } 循环完成之后,线程从run()方法返回,然后安静死掉。 创建一个新线程 Spawning a New Thread 新线程从已经运行的线程来创建。首先,构造一个新Thread实例。在此例中,一个新TwoThread对象将正好,因为TwoThread IS-A Thread: TwoThread tt = new TwoThread(); 下一步是调用start()方法准备开始执行线程(start()从Thread继承而来): tt.start(); start()调用后立即返回,不需要等待其他线程开始执行。在start()中,父线程异步地从JavaVM中给线程调度程序发出信号,告诉它只要它方便,其他线程可以开始执行了。在未来某个不可预期的时间里,其他线程将被激活,并调用Thread对象的run()方法(在该例中是指TwoThread中实现的重写方法run())。与此同时,原来的线程继续执行start()方法后面的语句。 两个线程并发、独立地运行。在一个多处理器的机器上,这两个线程可能在同一时刻真正都在运行,各自运行在自己的处理器上。 一个更常见的情况是只有一个处理器,JavaVM和操作系统共同工作来短时间调度每个线程。当其他线程被冻结,等待处理器的下一次机会时,每个线程就得到一次运行的机会。这种在线程之间的上下文切换是非常快的,给人一种同时执行的假象。 提示:一个新创建的线程可能在start()被调用之后的任何时间开始执行(进入run()方法)。这意味着start()方法之后的任何语句被执行之前,原来线程都可能被换出(swapped out)。 假如原来的线程正在执行下面的代码: stmt1(); tt.start(); 新线程有一个run()方法如下: public void run() { stmtA(); stmtB(); } stmt2(); 处理器中实际执行的语句顺序可能是stmt1(), tt.start(), stmt2(), stmtA(), and stmtB()。也可能是:stmt1(), tt.start(), stmtA(), stmtB(), and stmt2().也许还可能是另外一种顺序。 重要的是要记住:尽管每个线程将执行各自语句的顺序是已知和简单的,但是实际运行在处理器上的语句却是不确定的。对于程序的正确性而言,没有一种特殊的顺序可以依赖。 把所有放在一起 TwoThread.java, shown in Listing 2.1. Listing 2.1 TwoThread.java—The Complete Code for the TwoThread Example 1: public class TwoThread extends Thread { 2: public void run() { 3: for ( int i = 0; i < 10; i++ ) { 4: System.out.println(“New thread”); 5: } 6: } 7: 8: public static void main(String[] args) { 9: TwoThread tt = new TwoThread(); 10: tt.start(); 11: 12: for ( int i = 0; i < 10; i++ ) { 13: System.out.println(“Main thread”); 14: } 15: } 16: } 声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |