并发(1)中断线程,线程的状态,线程属性。

一 ,中断线程   

   获取当前线程: System.out.println(Thread.currentThread());    

   结果为:Thread[main,5,main]   指的是当前线程的字符串表示形式。Thread[main,5,main]为主线程的表示方式,中括号里的第一个值为当前主线程的名字,第二个为线程级别,第三个为线程组。Thread[Thread-0,5,main]Thread-0为另一个启动的线程,级别为5,属于线程组main

  当线程的 run 方法执行方法体中最后一条语句后, 并经由执行 return 语句返冋时,或者出现了在方法中没有捕获的异常时,线程将终止。 在 Java 的早期版本中, 还有一个 stop 方法, 其他线程可以调用它终止线程。但是, 这个方法现在已经被弃用。

       没有可以强制线程终止的方法。然而,interrupt 方法可以用来请求终止线程。

      当对一个线程调用 interrupt 方法时,线程的中断状的 boolean 标志。每个线程都应该不时地检查这个标志, 要想弄清中断状态是否被置位,首先调用静态的Thread.currentThread 方法获得当前线程, 然后调用 islnterrupted 方法:

while (!Thread.currentThread().islnterrupted() && more work to do)
{
do more work
}

• void interrupts()

向线程发送中断请求。线程的中断状态调用阻塞,那么,InterruptedException 

•static boolean interrupted()

测试当前线程(即正在执行这一命令的线程)是否被中断。注意,这是一个静态方法。这一调用会产生副作用—它将当前线程的中断状态重置为 false.

• boolean islnterrupted()

测试线程是否被终止。不像静态的中断方法,这一调用不改变线程的中断状态。

•static Thread currentThread()

返回代表当前执行线程的 Thread 

二 ,线程的状态

线程可以有如下 6 种状态:

•New (新创建)

•Runnable (可运行)

•Blocked (被阻塞)

•Waiting (等待)

•Timed waiting (计时等待)

•Terminated (被终止)

新创建线程

      当用 new 操作符创建一个新线程时,如 newThread(r), 该线程还没有开始运行。这意味着它的状态是 new。当一个线程处于新创建状态时,程序还没有开始运行线程中的代码。在线程运行之前还有一些基础工作要做。

可运行线程

       一旦调用 start 方法,线程处于 runnable 状态。一个可运行的线杆可能正在运行也可能没有运行, 这取决于操作系统给线程提供运行的时间。(Java 的规范说明没有将它作为一个单独状态。一个正在运行中的线程仍然处于可运行状态。)

       一旦一个线程开始运行,它不必始终保持运行。事实上,运行中的线程被中断,目的是为了让其他线程获得运行机会。线程调的细节依赖于操作系统提供的服务。抢占式调度系统给每一个可运行线程一个时间片来执行任务。当时间片用完,操作系统剥夺该线程的运行权, 并给另一个线程运行机会。

记住,在任何给定时刻,二个可运行的线程可能正在运行也可能没有运行(这就是为什么将这个状态称为可运行而不是运行。)

被阻塞线程和等待线程

       当线程处于被阻塞或等待状态时,它暂时不活动。它不运行任何代码且消耗最少的资源。直到线程调度器重新激活它。细节取决于它是怎样达到非活动状态的。

      (1)当一个线程试图获取一个内部的对象锁(而不是 javiutiUoncurrent 库中的锁,) 而该锁被其他线程持有, 则该线程进人阻塞状态

      (2)当线程等待另一个线程通知调度器一个条件时,它自己进入等待状态。

       ( 3 ) 有几个方法有一个超时参数。调用它们导致线程进人计时等待( timed waiting ) 状态。这一状态将一直保持到超时期或者接收到适当的通知。带有超时参数的方法有Thread.sleep 和 Object.wait、Thread.join、 Lock,tryLock 以及 Condition.await 的计时版。

       当一个线程被阻塞或等待时(或终止时,) 另一个线程被调度为运行状态。当一个线程被重新激活(例如, 因为超时期满或成功地获得了一个锁,) 调度器检查它是否具有比当前运行线程更高的优先级。如果是这样,调度器从当前运行线程中挑选一个, 剥夺其运行权,选择一个新的线程运行。

被终止的线程

线程因如下两个原因之一而被终止:
   •因为 run 方法正常退出而自然死亡。

   •因为一个没有捕获的异常终止了 run()方法而意外。

3 .线程属性

(1)优先级

       在 Java 程序设计语言中,每一个线程有一个优先级。默认情况下, 一+线程继承它的父线程的优先级。可以用 setPriority 方法提高或降低任何一个线程的优先级。可以将优先级设置为在 MIN_PRIORITY (在 Thread 类中定义为 1 ) 与 MAX_PRIORITY (义为 10 ) 之间的任何值。NORM_PRIORITY 被定义为 5。

       每当线程调度器有机会选择新线程时, 它首先选择具有较高优先级的线程。但是,线程优先级是高度依赖于系统的。当虚拟机依赖于宿主机平台的线程实现机制时, Java 线程的优先级被映射到宿主机平台的优先级上, 优先级个数也许更多,也许更少。例如,Windows 有 7 个优先级别。一些 Java 优先级将映射到相同的操作系统优先级在Oracle 为 Linux 提供的 Java 虚拟机中,线程的优先级被忽略一所有线程具有相同的优

     如果确实要使用优先级, 应该避免初学者常犯的一个错误。如果有几个高优先级的线程没有进入非活动状态, 低优先级的线程可能永远也不能执行。每当调度器决定运行一个新线程时,首先会在具有高优先级的线程中进行选择, 尽管这样会使低优先的线程完全饿死。

(2)守护线程

       可以通过调用t.setDaemon(true);

      将线程转换为守护线程(daemon thread。) 这样一个线程没有什么神奇。守护线程的唯一用途是为其他线程提供服务。计时线程就是一个例子,它定时地发送“ 计时器嘀嗒” 信号给其他线程或清空过时的高速缓存项的线程。当只剩下守护线程时, 虚拟机就退出了,由于如果只剩下守护线程, 就没必要继续运行程序了。 守护线程应该永远不去访问固有资源, 如文件、 数据库,因为它会在任何时候甚至在一个操作的中间发生中断。

 (3) 未捕获异常处理器

        线程的 run方法不能抛出任何受查异常, 但是,非受查异常会导致线程终止。在这种情况下,线程就死亡了。

        但是,不需要任何 catch子句来处理可以被传播的异常。相反,就在线程死亡之前, 异常被传递到一个用于未捕获异常的处理器。

        该处理器必须属于一个实现 Thread.UncaughtExceptionHandler 接口的类。这个接口只有—个方法。

void uncaughtException(Thread t, Throwable e)

    可以用 setUncaughtExceptionHandler 方法为任何线程安装一个处理器。也可以用 Thread类的静态方法 setDefaultUncaughtExceptionHandler 为所有线程安装一个默认的处理器。替换处理器可以使用日志 API 发送未捕获异常的报告到日志文件。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章