×
Traktatov.net » Философия Java » Читать онлайн
Страница 352 из 395 Настройки

}

Thread-0(5) Thread-1(4) Thread-2(3) Thread-3(2) Thread-4(1) */// ~

В целом происходящее не так уж сильно отличается от наследования от Thread, разве что синтаксис получается чуть более громоздким. Однако реализация интерфейса позволяет наследовать от другого класса, тогда как в варианте с Thread это невозможно.

Обратите внимание на вызов start() в конструкторе. Приведенный пример очень прост, поэтому, скорее всего, в нем такое решение безопасно, но вы должны знать, что запуск потоков в конструкторе может создать изрядные проблемы — до завершения конструктора может быть запущена на выполнение другая задача, которая обратится к объекту в нестабильном состоянии. Это еще одна причина, по которой использование Executor предпочтительнее явного создания объектов Thread.

Иногда бывает разумно спрятать потоковый код внутри класса с помощью внутреннего класса, как показано здесь:

// concurrency/ThreadVariations java

// Создание потоков с использованием внутренних классов.

import java.util.concurrent.*,

import static net mindview.util.Print.*;

// Используем именованный внутренний класс, class InnerThreadl {

private int countDown = 5; private Inner inner, private class Inner extends Thread { Inner(String name) { super(name); startO,

}

public void run() { try {

while(true) {

print(this);

if(--countDown == 0) return; sleep(lO);

}

} catchdnterruptedException e) { »print("interrupted"):

}

}

public String toStringO {

return getNameO + ": " + countDown;

>

}

public InnerThreadKString name) { inner = new Inner(name);

Thread-0(4), Thread-КЗ), Thread-2(2), Thread-3(1),

Thread-1(5). Thread-2(4). Thread-3(3). Thread-4(2).

Thread-0(3), Thread-1(2), Thread-2(1), Thread-4(5),

Thread-0(2). Thread-Id). Thread-3(5). Thread-4(4),

Thread-Od). Thread-2(5). Thread-3(4), Thread-4(3).

// Используем безымянный внутренний класс: class InnerThread2 {

private int countDown = 5; private Thread t;

public InnerThread2(String name) { t = new Thread(name) {

public void run() { try {

while(true) {

print(this).

if(--countDown == 0) return, sleep(lO).

}

} catch(InterruptedException e) {

printCsleepO interrupted");

}

}

public String toStringO {

return getNameO + ". " + countDown;

}

}:

t startO;

}

}

// Используем именованную реализацию Runnable. class InnerRunnablel {

private int countDown = 5; private Inner inner,

private class Inner implements Runnable { Thread t;

Inner(String name) {

t = new Thread(this. name); t.startO;

}

public void runO { try {

while(true) {

print(this);

if(--countDown == 0) return; Ti mellnit .MILLISECONDS. si eep( 10);

}

} catch(InterruptedException e) {

printCsleepO interrupted");

}

}

public String toStringO {

return t.getNameO + ". " + countDown;

}

}

public InnerRunnableKString name) { inner = new Inner(name),

// Используем анонимную реализацию Runnable-class InnerRunnable2 {

private int countDown = 5;

private Thread t;

public InnerRunnable2(String name) {

t = new Thread(new RunnableO { public void run() { try {

while(true) {

print(this);

if(--countDown == 0) return; Ti mellnit. MI LLISECONDS. s 1 eep( 10);

}

} catchdnterruptedException e) {

printCsleepO interrupted");