Heim >Java >javaLernprogramm >Java-Thread (1) – Detaillierte Erläuterung des Thread-Status und der Thread-Attribute

Java-Thread (1) – Detaillierte Erläuterung des Thread-Status und der Thread-Attribute

黄舟
黄舟Original
2017-03-01 11:38:321633Durchsuche


Im Betriebssystem verwenden wir die Time-Sharing-Methode, um mehrere Prozessaufgaben kontinuierlich auf der CPU zu wechseln und zu verarbeiten, wodurch den Menschen das Gefühl einer parallelen Verarbeitung vermittelt wird Betriebssystem Multitasking. Multitasking erweitert das Konzept des Multithreading auf einer niedrigeren Ebene, was bedeutet, dass ein Programm mehrere Threads gleichzeitig ausführt. Diese Art von Programm, das mehr als einen Thread gleichzeitig ausführen kann, wird als Multithread-Programm bezeichnet.

(1) Erstellen Sie einen Thread

1. Erstellen Sie eine Thread-Instanz

Es ist besser, es zu sehen als zu sehen Lassen Sie uns zuerst über Threads sprechen Verschiedene Zustände, warum zeigen wir nicht direkt, wie man einen neuen Thread erstellt?

package Thread;/**
 * 
 * @author QuinnNorris
 * 创建线程实例
 */public class NewThread {

    /**
     * @param args
     */
    public static void main(String[] args) {        // TODO Auto-generated method stub

        Runnable r = new Run();        //创建一个类对象
        Thread th = new Thread(r);        //由Runnable创建一个Thread对象
        th.start();        //调用start方法,运行新的线程,即运行的是th中的run方法
    }

}

Werfen wir einen Blick auf Run, eine Klasse, die die Runnable-Schnittstelle implementiert:

package Thread;/**
 * 
 * @author QuinnNorris
 * Run类实现了Runnable接口
 */public class Run implements Runnable {

    @Override
    public void run() {        // TODO Auto-generated method stub
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

}

Das Erstellen und Ausführen eines Threads sieht wahrscheinlich wie folgt aus: Erstellen Sie zunächst eine zu implementierende Klasse Es ist die ausführbare Schnittstelle, in der die einzige Ausführungsmethode in der ausführbaren Schnittstelle implementiert ist. Diese Methode wird automatisch aufgerufen, nachdem der Thread gestartet wurde. Dann erstellen wir ein Objekt der Run-Klasse in der Hauptklasse und übergeben dieses Objekt als Parameter des Konstruktors. Anschließend rufen wir die Startmethode mit den Parametern des Thread-Objekts auf und beginnen Sie mit der Ausführung der run-Anweisung in der Methode. Zu diesem Zeitpunkt wird ein Thread erstellt und gestartet.

2.Start-Methode und Ausführungsmethode

Es ist zu beachten, dass wir die Ausführungsmethode nicht direkt aufrufen können, da dadurch kein neuer Thread erstellt, sondern aufgerufen wird Mitten in einem vorhandenen Thread erstellt start einen neuen Thread für Sie und ruft automatisch run auf. Wir müssen die run-Methode unter keinen Umständen selbst manuell aufrufen.
Nachdem wir das gesagt haben, werfen wir einen Blick auf die Startmethode in der API:

public void start( )
Bewirken, dass der Thread mit der Ausführung beginnt;
Die Java Virtual Machine ruft die Ausführungsmethode des Threads auf.
Das Ergebnis ist, dass zwei Threads gleichzeitig ausgeführt werden; der aktuelle Thread (kehrt vom Aufruf zur Startmethode zurück) und der andere Thread (führt seine Ausführungsmethode aus).
Es ist illegal, einen Thread mehrmals zu starten. Insbesondere wenn die Ausführung eines Threads abgeschlossen ist, kann er nicht neu gestartet werden.

3.Runnable-Klasse

Wir wissen bereits, wie man einen Thread Schritt für Schritt erstellt, aber das befriedigt nicht unsere Neugier. Wir müssen es direkt tun Schauen Sie sich diese beiden Klassen genauer an, um zu sehen, welche Methoden und Eigenschaften sie enthalten. Die Runnable-Klasse enthält keine Attribute, sondern nur eine Ausführungsmethode. Diese Schnittstelle sollte von Klassen implementiert werden, die beabsichtigen, ihre Instanzen über einen Thread auszuführen. Die Thread-Klasse ist komplizierter.

(2) Thread-Klasse

1 Die Thread-Klasse implementiert die Runnable-Schnittstelle

Im obigen Beispiel also die Die Run-Klasse verfügt über eine natürliche Schreibweise, die die Thread-Klasse erbt, da Thread Runnable implementiert. Beide Schreibweisen überschreiben tatsächlich die Run-Methode. Aber Interviewfragen stellen Ihnen sehr gerne die Frage: Welche der beiden Möglichkeiten zur Implementierung von Multithreading ist besser, Thread oder Runnable? Zu diesem Zeitpunkt sollten Sie die Runnable-Methode beantworten. Der Grund ist auch sehr einfach. Es kann nur eine Vererbung geben, aber es kann viele Schnittstellen geben, die die Flexibilität der Vererbung dieser Klasse nicht beeinträchtigen.

2. Thread-Konstruktor

Es gibt viele Konstruktoren für Thread, und es gibt hauptsächlich vier Parameter:

public Thread(ThreadGroup group,Runnable target, String name,long stackSize)

Andere Konstruktoren Sie sind alle Schauen wir uns also die Bedeutung dieser vier Parameter an:

1. ThreadGroup-Gruppe

ThreadGroup bedeutet Thread-Gruppe Es kann klassifizieren und Verwalten Sie einen Stapel Threads. Die Steuerung und Verwaltung der Thread-Gruppe bedeutet, dass gleichzeitig der Thread-Stapel in der Thread-Gruppe gesteuert wird. Alle vom Benutzer erstellten Threads gehören zur angegebenen Thread-Gruppe. Wenn die angegebene Thread-Gruppe nicht angezeigt wird, gehört der Thread zur Standard-Thread-Gruppe (dh zur Haupt-Thread-Gruppe). Standardmäßig befinden sich der untergeordnete Thread und der übergeordnete Thread in derselben Thread-Gruppe. Die Thread-Gruppe, zu der ein Thread gehört, kann nur beim Erstellen angegeben werden. Die Thread-Gruppe, zu der er gehört, kann nicht geändert werden, während ein Thread ausgeführt wird. Das heißt, sobald ein Thread die Thread-Gruppe angibt, zu der er gehört. Es wird nicht geändert, bis der Thread endet. Die Struktur zwischen Thread-Gruppen und Threads ähnelt einer Baumstruktur. Ein Thread kann nur auf die Informationen seiner eigenen Thread-Gruppe zugreifen, nicht jedoch auf die Informationen in der übergeordneten Thread-Gruppe oder anderen Thread-Gruppen.

2.Ausführbares Ziel

Ziel ist das Objekt, dessen Ausführungsmethode aufgerufen wird. Wenn dieser Parameter null ist, muss die Ausführungsmethode in der Thread-Klasse neu geschrieben werden. Andernfalls ruft die Ausführungsmethode in der Thread-Klasse die Ausführungsmethode in der Runnable-Klasse auf.

3.String-Name

Jeder Thread hat seinen eigenen Namen. Tatsächlich wird dieser Parameter selten verwendet . Wenn wir diesen Parameter jedoch nicht ausfüllen, generiert der Thread automatisch einen neuen Namen. Der automatisch generierte Name hat die Form „Thread-“+n, wobei n eine ganze Zahl ist.

4.long stackSize

这是一种具有平台依赖性的参数,stackSize能指定堆栈的大小。 在某些平台上,指定一个较高的 stackSize 参数值可能使线程在抛出 StackOverflowError 之前达到较大的递归深度。stackSize 参数的值与最大递归深度和并发程度之间的关系细节与平台有关。在某些平台上,stackSize 参数的值无论如何不会起任何作用。 作为建议,可以让虚拟机自由处理 stackSize 参数。

Thread中还有很多很多实用的方法,我们在涉及到具体概念的时候再给予介绍。

(三)中断线程

我们既然创建了线程,那么这个线程在什么时候终止呢?有两种情况:

  1. 当线程的run方法执行方法体中最后一条语句后,正常退出而自然死亡。

  2. 出现了在方法中没有捕获的异常,此时终止run方法,意外死亡。

从这两点中,我们可以看出,其实我们并没有特殊的手段可以人为的去在中间干涉线程的中断(Thread中的stop方法或许有这种作用,但是这个方法已经被废弃,我们不要使用它)。虽然没有强制结束线程的方法,但是我们可以用interrupt方法请求终止线程。要注意“请求”这个词,没有任何语言方面的表述是要求一个被中断的线程应该被终止。我们去用interrupt方法中断一个线程不过是引起他的注意。被中断的线程可以决定如何去响应这个中断的请求。

当对一个线程调用interrupt方法时,线程的中断状态将被置位。这个中断状态是每个线程都有的boolean标志。每个线程都会不时地检查这个布尔值,判断这个线程是否被中断。
我们可以使用:

Thread.currentThread().isIntertrupted()

isInterrupt方法和interrupt很像,它是用来判断线程是否被中断。
我们可以通过interrupt方法来中断一个线程,但是值得注意的是,如果这个线程处于wait和sleep或者在join方法调用过程中,中断线程将会报一个InterruptedException异常,我们可以通过try-catch块来捕获这种异常。

(四)线程状态

线程一共有6种状态,这六种状态不是我们规定的,而是在Thread中的内部嵌套类State中规定的。这六种状态分别是New,Runnable,Blocked,Waiting,Timed waiting,Terminated六种,我们来逐个分析一下这几种状态的含义:

1.New 新创建的线程

这里的创建新的线程真的是仅仅new了一个线程。

new Thread(r);

创建新的线程,是指刚刚new出来的线程,这个线程没有通过start的方法来启动。

2.Runnable 可运行

那么一旦我们调用了start方法,这个线程开始工作。这是他就处于可运行状态,这个可运行状态不只是包含线程运行的时候,线程在中断的时候也被算为可运行状态。一个可运行状态的线程可能在运行也可能没在运行,我们不要因为一个线程在可运行的状态下没运行而急躁,很有可能这个线程的终止只是为了让其他的线程获得机会。

3.Blocked 被阻塞

当一个线程试图去获得一个内部锁时,但这个内部锁被其他的线程持有,这个时候,为了等待去使用这个内部锁,这个线程将会暂时处在被阻塞的状态。当其他线程释放锁的时候,这个线程获得了内部锁,并且从阻塞状态转变为非阻塞状态。

4.Wait 等待

当一个线程等待另一个线程通知调度器一个条件时,这个线程自己进入等待状态。等待状态和阻塞状态很类似,但是他们是存在本质区别的。如果另一个线程通知调度器结束,那么这个线程进行工作,等待状态也随之结束。

5.Timed waiting 计时等待

计时等待和等待是比较相似的,计时等待是表示他有一个超时参数。调用他们导致线程会进入计时等待。这个状态将一直保持到超市期满或者接收到适当的通知。相比较直接的等待,变得更加的安全。

6.Terminated 终止

线程的终止,我们在上面的线程中断中有所提及。线程终止理论上只有两种情况:当线程的run方法执行方法体中最后一条语句后,正常退出而自然死亡。2. 出现了在方法中没有捕获的异常,此时终止run方法,意外死亡。

(五)线程属性

线程中有很多的属性,尽管我在API中只看到一些其中一小部分的字段,但是线程的这种理念,在方法中也有所体现。线程的属性有以下这些:线程优先级,守护线程,处理未捕获异常的处理器。

1.线程优先级

在java中,每个线程都有一个优先级,我们需要知道的是,在没经过特殊处理的时候,所有的线程优先级都是一样的。默认的,我们把优先级分成1到10之间,高优先级的线程会先被操作。说到这里不由得让我们想起了操作系统中的进程优先级,和一些类似老化这样的名词。实际上,java虚拟机也确实是用进程的优先级来类比出线程的优先级,这样做最大的一个问题就在于,每个操作系统对于进程优先级的处理并不相同,java的线程优先级也因此而具有平台变化。实际上,我们不应该把程序的正确性依赖于线程优先级,我们应该尽量少的使用线程优先级。我们在这里介绍了线程优先级,但是我们并不建议去使用它。

2.守护线程

守护线程是指通过:

t.setDaemon(ture); //将线程转换为守护线程

这样的写法,将一个线程转换为守护线程。守护线程的作用是为其他的线程提供服务,如果其他所有的线程都被退出,只剩下守护线程,那么程序也就结束了。没有去单独运行守护线程的必要。比如说其他线程的计时器,我们就可以将它设置为一个守护线程。

3.处理未捕获异常的处理器

run方法不能抛出任何被检查,不被检测的异常会导致线程终止。在这种情况下,线程就死亡了。在这种情况下,我们需要实现一个Thread.UncaughtExceptionHandler。我们可以通过这种方式来知道,真正让我们的run意外死亡的问题在哪里。

(六)总结

在java的并发程序中,我们先了解了这些线程的状态和属性,下面我们就可以来研究关于线程之间同步的问题。如何能几个线程同时运行,让他们之间配合的好。

在操作系统中,我们通过分时的方法在CPU上不断地切换处理多个进程任务,给人并行处理的感觉,这种方法在操作系统中叫做多任务。多任务在较低层次上扩展出多线程的概念,也就是指一个程序同时执行多个线程。这种可以同时运行一个以上的线程的程序,我们叫做多线程程序。

 以上就是java线程(一)—线程状态及属性详解的内容,更多相关内容请关注PHP中文网(www.php.cn)!


Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn