Полное руководство. С# 4.0
Шрифт:
Обнуляемый тип — это особый вариант типа значения, представленный структу рой. Помимо значений, определяемых базовым типом, обнуляемый тип позволяет хранить пустые значения (null). Следовательно, обнуляемый тип имеет такой же диа пазон представления чисел и характеристики, как и его базовый тип. Он предоставляет дополнительную возможность обозначить значение, указывающее на то, что перемен ная данного типа не инициализирована. Обнуляемые типы являются объектами типа System.Nullаble<Т>, где Т — тип значения, которое не должно быть обнуляемым.
ПРИМЕЧАНИЕ Обнуляемые эквиваленты могут быть только у типов значений.
Обнуляемый
И во-вторых, обнуляемый тип объявляется более кратким и поэтому чаще исполь зуемым способом с указанием знака ? после имени базового типа. В приведенном ниже примере демонстрируется более распространенный способ объявления обну ляемых переменных типа int и bool. int? count; bool? done;
Когда в коде применяются обнуляемые типы, создаваемый обнуляемый объект обычно выглядит следующим образом. int? count = null;
В данной строке кода переменная count явно инициализируется пустым значени ем (null). Это вполне соответствует принятому правилу: прежде чем использовать переменную, ей нужно присвоить значение. В данном случае присваиваемое значение означает, что переменная не определена.
Значение может быть присвоено обнуляемой переменной обычным образом, по скольку преобразование базового типа в обнуляемый определено заранее. Например, в следующей строке кода переменной count присваивается значение 100. count = 100;
Определить, имеет переменная обнуляемого типа пустое или конкретное значение, можно двумя способами. Во-первых, можно проверить переменную на пустое значе ние. Так, если переменная count объявлена так, как показано выше, то в следующей строке определяется, имеет ли эта переменная конкретное значение. if (count != null) // переменная имеет значение
Если переменная count не является пустой, то она содержит конкретное значение. И во-вторых, можно воспользоваться доступным только для чтения свойством HasValue типа Nullable<T>, чтобы определить, содержит ли переменная обнуляе мого типа конкретное значение. Это свойство показано ниже. bool HasValue
Свойство HasValue возвращает логическое значение true, если экземпляр объекта, для которого оно вызывается, содержит конкретное значение, а иначе оно возвраща ет логическое значение false. Ниже приведен пример, в котором конкретное значе ние обнуляемого объекта count определяется вторым способом с помощью свойства HasValue. if(count.HasValue) // переменная имеет значение
Если обнуляемый объект содержит конкретное значение, то получить это значение можно с помощью доступного только для чтения свойства Value типа Nullable. Т Value
Свойство Value возвращает экземпляр обнуляемого объекта, для которо го оно вызывается. Если же попытаться получить с помощью этого свойства зна чение пустой переменной, то в итоге будет сгенерировано исключение System. InvalidOperationException. Кроме того, значение экземпляра обнуляемого объек та можно получить путем приведения к его базовому типу.
В следующей программе демонстрируется основной механизм обращения с обну ляемым типом. // Продемонстрировать применение обнуляемого типа. using System; class NullableDemo { static void Main { int? count = null; if(count.HasValue) Console.WriteLine("Переменная count имеет следующее значение: " + count.Value); else Console.WriteLine("У переменной count отсутствует значение"); count = 100; if(count.HasValue) Console.WriteLine("Переменная count имеет следующее значение: " + count.Value); else Console.WriteLine("У переменной count отсутствует значение"); } }
Вот к какому результату приводит выполнение этой программы. У переменной count отсутствует значение Переменная count имеет следующее значение: 100 Применение обнуляемых объектов в выражениях
Обнуляемый объект может использоваться в тех выражениях, которые являются действительными для его базового типа. Более того, обнуляемые объекты могут со четаться с необнуляемыми объектами в одном выражении. И это вполне допустимо благодаря предопределенному преобразованию базового типа в обнуляемый. Когда обнуляемые и необнуляемые типы сочетаются в одной операции, ее результатом ста новится значение обнуляемого типа.
В приведенной ниже программе демонстрируется применение обнуляемых типов в выражениях. // Использовать обнуляемые объекты в выражениях. using System; class NullableDemo { static void Main { int? count = null; int? result = null; int incr = 10; // переменная incr не является обнуляемой // переменная result содержит пустое значение. // переменная оказывается count пустой. result = count + incr; if(result.HasValue) Console.WriteLine("Переменная result имеет следующее значение: " + result.Value); else Console.WriteLine("У переменной result отсутствует значение"); // Теперь переменная count получает свое значение, и поэтому // переменная result будет содержать конкретное значение. count = 100; result = count + incr; if(result.HasValue) Console.WriteLine("Переменная result имеет следующее значение: " + result.Value); else Console.WriteLine("У переменной result отсутствует значение"); } }
При выполнении этой программы получается следующий результат. У переменной result отсутствует значение Переменная result имеет следующее значение: 110 Оператор ??
Попытка преобразовать обнуляемый объект в его базовый тип путем при ведения типов обычно приводит к генерированию исключения System. InvalidOperationException, если обнуляемый объект содержит пустое значение. Это может произойти, например, в том случае, если значение обнуляемого объекта присваивается переменной его базового типа с помощью приведения типов. Появле ния данного исключения можно избежать, если воспользоваться оператором ??, на зываемым нулеобъединяющим оператором. Этот оператор позволяет указать значение, которое будет использоваться по умолчанию, если обнуляемый объект содержит пу стое значение. Он также исключает потребность в приведении типов.
Ниже приведена общая форма оператора ??. обнуляемый_объект ?? значение_по_умолчанию
Если обнуляемыйобъект содержит конкретное значение, то результатом опера ции ?? будет именно это значение. В противном случае результатом операции ?? ока жется значениепо_умолчанию.
Например, в приведенном ниже фрагменте кода переменная balance содержит пустое значение. Вследствие этого переменной currentBalance присваивается зна чение 0.0, используемое по умолчанию, и тем самым устраняется причина для гене рирования исключения. double? balance = null; double currentBalance; currentBalance = balance ?? 0.0;