论坛首页 入门技术论坛

JAVA 线程编程 ----两个线程程序

浏览 1151 次
精华帖 (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类
  • 在继承Thread的子类中重写run( )方法;
  • 创建这个新类的实例
  • 调用该实例的start( )方法


继承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: }


























论坛首页 入门技术版

跳转论坛:
Global site tag (gtag.js) - Google Analytics