友情提示:如果本网页打开太慢或显示不完整,请尝试鼠标右键“刷新”本网页!
第三电子书 返回本书目录 加入书签 我的书架 我的书签 TXT全本下载 『收藏到我的浏览器』

Java编程思想第4版[中文版](PDF格式)-第117部分

快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部! 如果本书没有阅读完,想下次继续接着阅读,可使用上方 "收藏到我的浏览器" 功能 和 "加入书签" 功能!


不需要涉及一个独立的对象。但为这种便利也是要付出代价的——只可为那个特定的对象运行单独一个线程 

 (尽管可创建那种类型的多个对象,或者在不同的类里创建其他对象)。  

注意Runnable 接口本身并不是造成这一限制的罪魁祸首。它是由于 Runnable 与我们的主类合并造成的,因 

为每个应用只能主类的一个对象。  



14。1。4 制作多个线程  



现在考虑一下创建多个不同的线程的问题。我们不可用前面的例子来做到这一点,所以必须倒退回去,利用 

从Thread 继承的多个独立类来封装run()。但这是一种更常规的方案,而且更易理解,所以尽管前例揭示了 

我们经常都能看到的编码样式,但并不推荐在大多数情况下都那样做,因为它只是稍微复杂一些,而且灵活 

性稍低一些。  

下面这个例子用计数器和切换按钮再现了前面的编码样式。但这一次,一个特定计数器的所有信息(按钮和 

文本字段)都位于它自己的、从Thread 继承的对象内。Ticker 中的所有字段都具有private (私有)属性, 

这意味着Ticker 的具体实现方案可根据实际情况任意修改,其中包括修改用于获取和显示信息的数据组件的 

数量及类型。创建好一个Ticker 对象以后,构建器便请求一个 AWT 容器(Container)的句柄——Ticker 用 

自己的可视组件填充那个容器。采用这种方式,以后一旦改变了可视组件,使用Ticker 的代码便不需要另行 

修改一道。  

  

//: Counter4。java  

// If you separate your thread from the main  

// class; you can have as many threads as you  

// want。  

import java。awt。*;  

import java。awt。event。*;  

import java。applet。*;  

  



                                                                                    495 


…………………………………………………………Page 497……………………………………………………………

class Ticker extends Thread {  

  private Button b = new Button(〃Toggle〃);  

  private TextField t = new TextField(10);  

  private int count = 0;  

  private boolean runFlag = true;  

  public Ticker(Container c) {  

    b。addActionListener(new ToggleL());  

    Panel p = new Panel();  

    p。add(t);  

    p。add(b);  

    c。add(p);  

  }  

  class ToggleL implements ActionListener {  

    public void actionPerformed(ActionEvent e) {  

      runFlag = !runFlag;  

    }  

  }  

  public void run() {  

    while (true) {  

      if(runFlag)  

        t。setText(Integer。toString(count++));  

       try {  

        sleep(100);  

      } catch (InterruptedException e){}  

    }  

  }  

}  

  

public class Counter4 extends Applet {  

  private Button start = new Button(〃Start〃);  

  private boolean started = false;  

  private Ticker'' s;  

  private boolean isApplet = true;  

  private int size;  

  public void init() {  

    // Get parameter 〃size〃 from Web page:  

    if(isApplet)  

      size =   

        Integer。parseInt(getParameter(〃size〃));  

    s = new Ticker'size';  

    for(int i = 0; i 《 s。length; i++)  

      s'i' = new Ticker(this);  

    start。addActionListener(new StartL());  

    add(start);  

  }  

  class StartL implements ActionListener {  

    public void actionPerformed(ActionEvent e) {  

      if(!started) {  

        started = true;  

        for(int i = 0; i 《 s。length; i++)  

          s'i'。start();  

      }  



                                                                                             496 


…………………………………………………………Page 498……………………………………………………………

    }  

  }  

  public static void main(String'' args) {  

    Counter4 applet = new Counter4();  

    // This isn't an applet; so set the flag and  

    // produce the parameter values from args:  

    applet。isApplet = false;  

    applet。size =   

      (args。length == 0 ? 5 :  

        Integer。parseInt(args'0'));  

    Frame aFrame = new Frame(〃Counter4〃);  

    aFrame。addWindowListener(  

      new WindowAdapter() {  

        public void windowClosing(WindowEvent e) {  

          System。exit(0);  

        }  

      });  

    aFrame。add(applet; BorderLayout。CENTER);  

    aFrame。setSize(200; applet。size * 50);  

    applet。init();  

    applet。start();  

    aFrame。setVisible(true);  

  }  

} ///:~  

  

Ticker 不仅包括了自己的线程处理机制,也提供了控制与显示线程的工具。可按自己的意愿创建任意数量的 

线程,毋需明确地创建窗口化组件。  

在Counter4 中,有一个名为s 的Ticker 对象的数组。为获得最大的灵活性,这个数组的长度是用程序片参 

数接触Web 页而初始化的。下面是网页中长度参数大致的样子,它们嵌于对程序片(applet)的描述内容 

中:  

  

  

  

其中,param,name 和 value 是所有 Web 页都适用的关键字。name 是指程序中对参数的一种引用称谓,value 

可以是任何字串(并不仅仅是解析成一个数字的东西)。  

我们注意到对数组 s 长度的判断是在 init()内部完成的,它没有作为s 的内嵌定义的一部分提供。换言之, 

不可将下述代码作为类定义的一部分使用(应该位于任何方法的外部):  

inst size = Integer。parseInt(getParameter(〃Size〃));  

Ticker'' s = new Ticker'size'  

可把它编译出来,但会在运行期得到一个空指针违例。但若将 getParameter()初始化移入 init(),则可正常 

工作。程序片框架会进行必要的启动工作,以便在进入 init()前收集好一些参数。  

此外,上述代码被同时设置成一个程序片和一个应用(程序)。在它是应用程序的情况下,size 参数可从命 

令行里提取出来(否则就提供一个默认的值)。  

数组的长度建好以后,就可以创建新的Ticker 对象;作为 Ticker 构建器的一部分,用于每个Ticker 的按钮 

和文本字段就会加入程序片。  

按下Start 按钮后,会在整个 Ticker 数组里遍历,并为每个 Ticker 调用 start()。记住,start()会进行必 

要的线程初始化工作,然后为那个线程调用run()。  

ToggleL 监视器只是简单地切换 Ticker 中的标记,一旦对应线程以后需要修改这个标记,它会作出相应的反 

应。  

这个例子的一个好处是它使我们能够方便地创建由单独子任务构成的大型集合,并以监视它们的行为。在这 

种情况下,我们会发现随着子任务数量的增多,机器显示出来的数字可能会出现更大的分歧,这是由于为线 

程提供服务的方式造成的。  



                                                                                       497 


…………………………………………………………Page 499……………………………………………………………

亦可试着体验一下 sleep(100)在 Ticker。run()中的重要作用。若删除 sleep(),那么在按下一个切换按钮 

前,情况仍然会进展良好。按下按钮以后,那个特定的线程就会出现一个失败的 runFlag,而且 run()会深深 

地陷入一个无限循环——很难在多任务处理期间中止退出。因此,程序对用户操作的反应灵敏度会大幅度降 

低。  



14。1。5 Daemon 线程  



 “Daemon”线程的作用是在程序的运行期间于后台提供一种“常规”服务,但它并不属于程序的一个基本部 

分。因此,一旦所有非 Daemon 线程完成,程序也会中止运行。相反,假若有任何非 Daemon 线程仍在运行 

 (比如还有一个正在运行main()的线程),则程序的运行不会中止。  

通过调用 isDaemon(),可调查一个线程是不是一个Daemon,而且能用 setDaemon()打开或者关闭一个线程的 

Daemon 状态。如果是一个 Daemon 线程,那么它创建的任何线程也会自动具备Daemon 属性。  

下面这个例子演示了Daemon 线程的用法:  

  

//: Daemons。java  

// Daemonic behavior  

import java。io。*;  

  

class Daemon extends Thread {  

  private static final int SIZE = 10;  

  private Thread'' t = new Thread'SIZE';  

  public Daemon() {   

    setDaemon(true);  

    start();  

  }  

  public void run() {  

    for(int i = 0; i 《 SIZE; i++)  

      t'i' = new DaemonSpawn(i);  

    for(int i = 0; i 《 SIZE; i++)  

      System。out。println(  

        〃t'〃 + i + 〃'。isDaemon() = 〃   

        + t'i'。isDaemon());  

    while(true)   

      yield();  

  }  

}  

  

class DaemonSpawn extends Thread {  

  public DaemonSpawn(int i) {  

    System。out。println(  

      〃DaemonSpawn 〃 + i + 〃 started〃);  

    start();  

  }  

  public void run() {  

    while(true)   

      yield();  

  }  

}  

  

public class Daemons {  

  public static void main(String'' args) {  

    Thread d = new Daemon();  

    System。out。println(  



                                                                                             498 


…………………………………………………………Page 500……………………………………………………………

      〃d。isDaemon() = 〃 + d。isDaemon());  

    // Allow the daemon threads to finish  

    // their startup processes:  

    BufferedReader stdin =  

      new BufferedReader(  

        new InputStreamReader(System。in));  

    System。out。println(〃Waiting for CR〃);  

    try {  

      stdin。readLine();  

    } catch(IOException e) {}  

  }  

} ///:~  

  

Daemon 线程可将自己的Daemon 标记设置成“真”,然后产生一系列其他线程,而且认为它们也具有 Daemon 

属性。随后,它进入一个无限循环,在其中调用yield(),放弃对其他进程的控制。在这个程序早期的一个 

版本中,无限循环会使 int计数器增值,但会使整个程序都好象陷入停顿状态。换用 yield()后,却可使程 

序充满“活力”,不会使人产生停滞或反应迟钝的感觉。  

一旦main()完成自己的工作,便没有什么能阻止程序中断运行,因为这里运行的只有 Daemon 线程。所以能 

看到启动所有Daemon 线程后显示出来的结果,System。in 也进行了相应的设置,使程序中断前能等待一个回 

车。如果不进行这样的设置,就只能看到创建 Daemon 线程的一部分结果(试试将readLine()代码换成不同 

长度的 sleep()调用,看看会有什么表现)。  



14。2 共享有限的资源  



可将单线程程序想象成一种孤立的实体,它能遍历我们的问题空间,而且一次只能做一件事情。由于只有一 

个实体,所以永远不必担心会有两个实体同时试图使用相同的资源,就象两个人同时都想停到一个车位,同 

时都想通过一扇门,甚至同时发话。  

进入多线程环境后,它们则再也不是孤立的。可能会有两个甚至更多的线程试图同时同一个有限的资源。必 

须对这种潜在资源冲突进行预防,否则就可能发生两个线程同时访问一个银行帐号,打印到同一台计算机, 

以及对同一个值进行调整等等。  



14。2。1  资源访问的错误方法  



现在考虑换成另一种方式来使用本章频繁见到的计数器。在下面的例子中,每个线程都包含了两个计数器, 

它们在 run()里增值以及显示。除此以外,我们使用了 Watcher 类的另一个线程。它的作用是监视计数器, 

检查它们是否保持相等。这表面是一项无意义的行动,因为如果查看代码,就会发现计数器肯定是相同的。 

但实际情况却不一定如此。下面是程序的第一个版本:  

  

//: Sharing1。java  

// Problems with resource sharing while threading  

import java。awt。*;  

import java。awt。event。*;  

import java。applet。*;  

  

class TwoCounter extends Thread {  

  private boolean started = false;  

  private TextField   

    t1 = new TextField(5);  

    t2 = new TextField(5);  

  private Label l =   

    new Label(〃count1 == count2〃);  

  private int count1 = 0; count2 = 0;  

  // Add the display ponents as a panel  



                                                                                     499 


…………………………………………………………Page 501……………………………………………………………

  // to the given container:  

  public TwoCounter(Container c) {  

    Panel p = new Panel();  

    p。add(t1);  

    p。add(t2);  

    p。add(l);  

    c。add(p);  

  }  

  public void start() {  

    if(!started) {  

      started = true;  

      super。start();  

    }  

  }  

  public void run() {  

    while (true) {  

      t1。setText(Integer。toString(count1++));  

      t2。setText(Integer。toString(count2++));  

      try {  

        sleep(500);  

      } catch (InterruptedException e){}  

    }  

  }  

  public void synchTest() {  

    Sha
返回目录 上一页 下一页 回到顶部 1 1
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部!
温馨提示: 温看小说的同时发表评论,说出自己的看法和其它小伙伴们分享也不错哦!发表书评还可以获得积分和经验奖励,认真写原创书评 被采纳为精评可以获得大量金币、积分和经验奖励哦!