Поделиться через


Статические конструкторы (Руководство по программированию в C#)

Статический конструктор используется для инициализации любых статических данных или для выполнения определенного действия, которое требуется выполнить только один раз. Он вызывается автоматически до создания первого экземпляра или ссылки на статические элементы. Статический конструктор вызывается по крайней мере один раз.

class SimpleClass
{
    // Static variable that must be initialized at run time.
    static readonly long baseline;

    // Static constructor is called at most one time, before any
    // instance constructor is invoked or member is accessed.
    static SimpleClass()
    {
        baseline = DateTime.Now.Ticks;
    }
}

Существует несколько действий, которые являются частью статической инициализации. Эти действия выполняются в следующем порядке:

  1. Статические поля имеют значение 0. Среда выполнения обычно выполняет эту инициализацию.
  2. Запуск инициализаторов статических полей. Инициализаторы статических полей выполняются в наиболее производном типе.
  3. Запуск инициализаторов статических полей базового типа. Инициализаторы статических полей, начиная с непосредственной базой через каждый базовый тип до System.Object.
  4. Любой статический конструктор исполняется. Все статические конструкторы, начиная с конечного базового класса Object.Object, выполняются последовательно через каждый базовый класс до выполнения типа. Порядок вызова статического конструктора не определён. Однако все статические конструкторы в иерархии выполняются перед созданием любого из экземпляров.

Внимание

Существует одно важное исключение из правила, согласно которому статический конструктор запускается перед созданием любого экземпляра. Если инициализатор статических полей создает экземпляр типа, этот инициализатор выполняется (включая любой вызов конструктора экземпляра) перед запуском статического конструктора. Это наиболее распространено в однотонном шаблоне, как показано в следующем примере:

public class Singleton
{
    // Static field initializer calls instance constructor.
    private static Singleton instance = new Singleton();

    private Singleton()
    { 
        Console.WriteLine("Executes before static constructor.");
    }

    static Singleton()
    { 
        Console.WriteLine("Executes after instance constructor.");
    }

    public static Singleton Instance => instance;
}

Инициализатор модуля может быть альтернативой статическому конструктору. Дополнительные сведения см. в спецификации для инициализаторов модулей.

Замечания

Статические конструкторы обладают следующими свойствами.

  • Статический конструктор не принимает модификаторы доступа и не имеет параметров.
  • Класс или структура могут иметь только один статический конструктор.
  • Статические конструкторы не могут быть унаследованы или перегружены.
  • Статический конструктор не может вызываться напрямую и должен вызываться только средой CLR. Он вызывается автоматически.
  • Пользователь не управляет временем, в течение которого статический конструктор выполняется в программе.
  • Статический конструктор вызывается автоматически. Инициализируется класс перед созданием первого экземпляра или объявлением любых статических элементов в этом классе (не в его базовых классах). Статический конструктор выполняется раньше, чем конструктор экземпляра. Если инициализаторы статических полей присутствуют в классе статического конструктора, они выполняются в текстовом порядке, в котором они отображаются в объявлении класса. Инициализаторы выполняются сразу перед статическим конструктором.
  • Если вы не предоставили статический конструктор для инициализации статических полей, все статические поля инициализируются значениями по умолчанию, которые указаны в разделе Значения по умолчанию типов C#.
  • Если статический конструктор создает исключение, среда выполнения не вызывает его во второй раз, а тип остается неинициализируемым в течение времени существования домена приложения. Чаще всего исключение TypeInitializationException вызывается, когда статический конструктор не может создать экземпляр типа или в статическом конструкторе создается необработанное исключение. Для статических конструкторов, не определенных явным образом в исходном коде, может потребоваться проверка кода промежуточного языка (IL).
  • Наличие статического конструктора не позволяет добавлять атрибут типа BeforeFieldInit. Это ограничивает возможности оптимизации во время выполнения.
  • Поле, объявленное как static readonly может быть назначено только в рамках его объявления или в статическом конструкторе. Если явный статический конструктор не требуется, инициализируйте статические поля при объявлении, а не через статический конструктор, чтобы оптимизировать производительность во время выполнения.
  • Среда выполнения вызывает статический конструктор не более одного раза для каждого домена приложения. Этот вызов выполняется в заблокированной области на основе конкретного типа класса. В теле статического конструктора не требуются дополнительные механизмы блокировки. Чтобы избежать риска взаимоблокировки, не блокируйте текущий поток в статических конструкторах и инициализаторах. Например, не стоит ожидать выполнения задач, потоков, обработчиков ожидания или событий, не следует также создавать блокировки или исполнять блокирующие параллельные операции, такие как параллельные циклы и запросы Parallel LINQ.

Примечание.

Наличие явного статического конструктора следует документировать, несмотря на отсутствие прямого доступа к нему, поскольку это важно для устранения неполадок с исключениями при инициализации.

Использование

  • Типичным использованием статических конструкторов является случай, когда класс использует файл журнала и конструктор применяется для добавления записей в этот файл.
  • Статические конструкторы также полезны при создании классов-оболочек для неуправляемого кода, когда конструктор может вызвать метод LoadLibrary.
  • Статические конструкторы также удобно использовать для выполнения проверок во время работы программы с параметрами типа, которые нельзя проконтролировать на этапе компиляции посредством ограничений параметров типа.

Пример

В этом примере класс Bus имеет статический конструктор. При создании первого экземпляра класса Bus (bus1) для инициализации класса вызывается статический конструктор. Выходные данные этого примера проверяют, что статический конструктор выполняется только один раз, даже если создаются два экземпляра класса Bus, и что он выполняется перед выполнением конструктора экземпляра.

public class Bus
{
    // Static variable used by all Bus instances.
    // Represents the time the first bus of the day starts its route.
    protected static readonly DateTime globalStartTime;

    // Property for the number of each bus.
    protected int RouteNumber { get; set; }

    // Static constructor to initialize the static variable.
    // It is invoked before the first instance constructor is run.
    static Bus()
    {
        globalStartTime = DateTime.Now;

        // The following statement produces the first line of output,
        // and the line occurs only once.
        Console.WriteLine($"Static constructor sets global start time to {globalStartTime.ToLongTimeString()}");
    }

    // Instance constructor.
    public Bus(int routeNum)
    {
        RouteNumber = routeNum;
        Console.WriteLine($"Bus #{RouteNumber} is created.");
    }

    // Instance method.
    public void Drive()
    {
        TimeSpan elapsedTime = DateTime.Now - globalStartTime;

        // For demonstration purposes we treat milliseconds as minutes to simulate
        // actual bus times. Do not do this in your actual bus schedule program!
        Console.WriteLine($"{this.RouteNumber} is starting its route {elapsedTime.Milliseconds:N2} minutes after global start time {globalStartTime.ToShortTimeString()}.");
    }
}

class TestBus
{
    static void Main()
    {
        // The creation of this instance activates the static constructor.
        Bus bus1 = new Bus(71);

        // Create a second bus.
        Bus bus2 = new Bus(72);

        // Send bus1 on its way.
        bus1.Drive();

        // Wait for bus2 to warm up.
        System.Threading.Thread.Sleep(25);

        // Send bus2 on its way.
        bus2.Drive();

        // Keep the console window open in debug mode.
        Console.WriteLine("Press any key to exit.");
        Console.ReadKey();
    }
}
/* Sample output:
    Static constructor sets global start time to 3:57:08 PM.
    Bus #71 is created.
    Bus #72 is created.
    71 is starting its route 6.00 minutes after global start time 3:57 PM.
    72 is starting its route 31.00 minutes after global start time 3:57 PM.
*/

Спецификация языка C#

Дополнительные сведения см. в разделе Статические конструктор в спецификации языка C#.

См. также