Книга: C# 4.0: полное руководство

Прерывание потока

Прерывание потока

Иногда поток полезно прервать до его нормального завершения. Например, отладчику может понадобиться прервать вышедший из-под контроля поток. После прерывания поток удаляется из системы и не может быть начат снова.

Для прерывания потока до его нормального завершения служит метод Thread.Abort(). Ниже приведена простейшая форма этого метода.

public void Abort()

Метод Abort() создает необходимые условия Для генерирования исключения ThreadAbortException в том потоке, для которого он был вызван. Это исключение приводит к прерыванию потока и может быть перехвачено и в коде программы, но в этом случае оно автоматически генерируется еще раз, чтобы остановить поток. Метод Abort() не всегда способен остановить поток немедленно, поэтому если поток требуется остановить перед тем, как продолжить выполнение программы, то после метода Abort() следует сразу же вызвать метод Join(). Кроме того, в самых редких случаях методу Abort() вообще не удается остановить поток. Это происходит, например, в том случае, если кодовый блок finally входит в бесконечный цикл.

В приведенном ниже примере программы демонстрируется применение метода Abort() для прерывания потока.

// Прервать поток с помощью метода Abort().
using System;
using System.Threading;
class MyThread {
  public Thread Thrd;
  public MyThread(string name) {
    Thrd = new Thread(this.Run);
    Thrd.Name = name;
    Thrd.Start();
  }
  // Это точка входа в поток,
  void Run() {
    Console.WriteLine(Thrd.Name + " начат.");
    for (int i = 1; i <= 1000; i++) {
      Console.Write(i + " ");
      if((i %10)==0) {
        Console.WriteLine();
        Thread.Sleep(250);
      }
    }
    Console .WriteLine (Thrd.Name + " завершен.");
  }
}
class StopDemo {
  static void Main() {
    MyThread mt1 = new MyThread("Мой Поток");
    Thread.Sleep(1000); // разрешить порожденному потоку начать свое выполнение
    Console.WriteLine("Прерывание потока.");
    mt1.Thrd.Abort();
    mt1.Thrd.Join(); // ожидать прерывания потока
    Console.WriteLine("Основной поток прерван.");
  }
}

Вот к какому результату приводит выполнение этой программы.

Мой Поток начат.
1 2 3 4 5 6 7 8 9 10
11 12 13 14 15 16 17 18 19 20
21 22 23 24 25 26 27 28 29 30
31 32 33 34 35 36 37 38 39 40
Прерывание потока.
Основной поток прерван.

-----------------------------

ПРИМЕЧАНИЕ

Метод Abort() не следует применять в качестве обычного средства прерывания потока, поскольку он предназначен для особых случаев. Обычно поток должен завершаться естественным образом, чтобы произошел возврат из метода, выполняющего роль точки входа в него.

-----------------------------

Другая форма метода Abort()

В некоторых случаях оказывается полезной другая форма метода Abort(), приведенная ниже в общем виде:

public void Abort(object stateInfo)

где statelnfo обозначает любую информацию, которую требуется передать потоку, когда он останавливается. Эта информация доступна посредством свойства ExceptionState из класса исключения ThreadAbortException. Подобным образом потоку можно передать код завершения. В приведенном ниже примере программы демонстрируется применение данной формы метода Abort().

// Использовать форму метода Abort (object stateInfo) .
using System;
using System.Threading;
class MyThread {
  public Thread Thrd;
  public MyThread(string name) {
    Thrd = new Thread(this.Run);
    Thrd.Name = name;
    Thrd.Start();
  }
  // Это точка входа в поток,
  void Run() {
    try {
      Console.WriteLine(Thrd.Name + " начат.");
      for (int i = 1; i <= 1000; i++)    {
        Console.Write(i + " ");
        if((i%10)==0) {
          Console.WriteLine();
          Thread.Sleep(250);
        }
      }
      Console.WriteLine(Thrd.Name + " завершен нормально.");
    } catch(ThreadAbortException exc) {
      Console.WriteLine("Поток прерван, код завершения "
               + exc.ExceptionState);
    }
  }
}
class UseAltAbort {
  static void Main() {
    MyThread mt1 = new MyThread("Мой Поток");
    Thread.Sleep(1000);// разрешить порожденному потоку начать свое выполнение
    Console.WriteLine("Прерывание потока.");
    mt1.Thrd.Abort(100);
    Console.WriteLine("Основной поток прерван.");
  }
}

Эта программа дает следующий результат.

Мой Поток начат.
1 2 3 4 5 6 7 8 9 10
11 12 13 14 15 16 17 18 19 20
21 22 23 24 25 26 27 28 29 30
31 32 33 34 35 36 37 38 39 40
Прерывание потока.
Поток прерван, код завершения 100
Основной поток прерван.

Как следует из приведенного выше результата, значение 100 передается методу Abort() в качестве кода прерывания. Это значение становится затем доступным посредством свойства ExceptionState из класса исключения ThreadAbortException, которое перехватывается потоком при его прерывании.

Отмена действия метода Abort()

Запрос на преждевременное прерывание может быть переопределен в самом потоке. Для этого необходимо сначала перехватить в потоке исключение ThreadAbortException, а затем вызвать метод ResetAbort(). Благодаря этому исключается повторное генерирование исключения по завершении обработчика исключения, прерывающего данный поток. Ниже приведена форма объявления метода ResetAbort().

public static void ResetAbort()

Вызов метода ResetAbort() может завершиться неудачно, если в потоке отсутствует надлежащий режим надежной отмены преждевременного прерывания потока.

В приведенном ниже примере программы демонстрируется применение метода ResetAbort().

// Использовать метод ResetAbort().
using System;
using System.Threading;
class MyThread {
  public Thread Thrd;
  public MyThread(string name) {
    Thrd = new Thread(this.Run);
    Thrd.Name = name;
    Thrd.Start();
  }
  // Это точка входа в поток,
  void Run() {
    Console.WriteLine(Thrd.Name + ".начат.");
    for (int i = 1; i<=1000; i++) {
      try {
        Console.Write(i + " ");
        if((i %10)==0) {
          Console.WriteLine();
          Thread.Sleep(250);
        }
      } catch(ThreadAbortException exc) {
        if((int)exc.ExceptionState ==0) {
          Console.WriteLine("Прерывание потока отменено! " +
                 "Код завершения " + exc.ExceptionState);
          Thread.ResetAbort();
        }
        else
          Console.WriteLine("Поток прерван, код завершения "
                + exc.ExceptionState);
      }
    }
    Console.WriteLine(Thrd.Name + " завершен нормально.");
  }
}
class ResetAbort {
  static void Main() {
    MyThread mt1 = new MyThread("Мой Поток");
    Thread.Sleep(1000); // разрешить порожденному потоку начать свое выполнение
    Console.WriteLine("Прерывание потока.");
    mt1.Thrd.Abort(0); // это не остановит поток
    Thread.Sleep(1000); // разрешить порожденному потоку выполняться подольше
    Console.WriteLine("Прерывание потока.");
    mt1.Thrd.Abort(100); // а это остановит поток
    mt1.Thrd.Join(); // ожидать прерывания потока
    Console.WriteLine("Основной поток прерван.");
  }
}

Ниже приведен результат выполнения этой программы.

Мой Поток.начат.
1 2 3 4 5 6 7 8 9 10
11 12 13 14 15 16 17 18 19 20
21 22 23 24 25 26 27 28 29 30
31 32 33 34 35 36 37 38 39 40
Прерывание потока.
Прерывание потока отменено! Код завершения 0
41 42 43 44 45 46 47 48 49 50
51 52 53 54 55 56 57 58 59 60
61 62 63 64 65 66 67 68 69 70
71 72 73 74 75 76 77 78 79 80
Прерывание потока.
Поток прерван, код завершения 100
Основной поток прерван.

Если в данном примере программы метод Abort() вызывается с нулевым аргументом, то запрос на преждевременное прерывание отменяется потоком, вызывающим метод ResetAbort(), и выполнение этого потока продолжается. Любое другое значение аргумента приведет к прерыванию потока.

Оглавление книги


Генерация: 0.048. Запросов К БД/Cache: 0 / 0
поделиться
Вверх Вниз