Язык программирования C#9 и платформа .NET5 - Троелсен Эндрю
Обратите внимание, что переменные-члены объявлены с применением модификатора доступа public. Открытые (public) члены класса доступны напрямую после того, как создан объект этого типа. Вспомните, что термин объект используется для описания экземпляра заданного типа класса, который создан с помощью ключевого слова new.
На заметку! Поля данных класса редко (если вообще когда-нибудь) должны определяться как открытые. Чтобы обеспечить целостность данных состояния, намного лучше объявлять данные закрытыми (private) или возможно защищенными (protected) и разрешать контролируемый доступ к данным через свойства (как будет показано далее в главе). Тем не менее, для максимального упрощения первого примера мы определили поля данных как открытые.
После определения набора переменных-членов, представляющих состояние класса, следующим шагом в проектировании будет установка членов, которые моделируют его поведение. Для этого примера в классе Car определены методы по имени SpeedUp() и PrintState(). Модифицируйте код класса Car следующим образом:
class Car
{
// 'Состояние' объекта Car.
public string petName;
public int currSpeed;
// Функциональность Car.
// Использовать синтаксис членов, сжатых до выражений,
// который рассматривался в главе 4.
public void PrintState()
=> Console.WriteLine("{0} is going {1} MPH.", petName, currSpeed);
public void SpeedUp(int delta)
=> currSpeed += delta;
}
Метод PrintState() — простая диагностическая функция, которая выводит текущее состояние объекта Car в окно командной строки. Метод SpeedUp() увеличивает скорость автомобиля, представляемого объектом Car, на величину, которая передается во входном параметре типа int. Обновите операторы верхнего уровня в файле Program.cs, как показано ниже:
Console.WriteLine("***** Fun with Class Types *****n");
// Разместить в памяти и сконфигурировать объект Car.
Car myCar = new Car();
myCar.petName = "Henry";
myCar.currSpeed = 10;
// Увеличить скорость автомобиля в несколько раз и вывести новое состояние.
for (int i = 0; i <= 10; i++)
{
myCar.SpeedUp(5);
myCar.PrintState();
(window.adrunTag = window.adrunTag || []).push({v: 1, el: 'adrun-4-390', c: 4, b: 390})}
Console.ReadLine();
Запустив программу, вы увидите, что переменная Car (myCar) поддерживает свое текущее состояние на протяжении жизни приложения:
***** Fun with Class Types *****
Henry is going 15 MPH.
Henry is going 20 MPH.
Henry is going 25 MPH.
Henry is going 30 MPH.
Henry is going 35 MPH.
Henry is going 40 MPH.
Henry is going 45 MPH.
Henry is going 50 MPH.
Henry is going 55 MPH.
Henry is going 60 MPH.
Henry is going 65 MPH.
Размещение объектов с помощью ключевого слова new
Как было показано в предыдущем примере кода, объекты должны размещаться в памяти с применением ключевого слова new. Если вы не укажете ключевое слово new и попытаетесь использовать переменную класса в последующем операторе кода, то получите ошибку на этапе компиляции. Например, приведенные далее операторы верхнего уровня не скомпилируются:
Console.WriteLine("***** Fun with Class Types *****n");
// Ошибка на этапе компиляции! Забыли использовать new для создания объекта!
Car myCar;
myCar.petName = "Fred";
Чтобы корректно создать объект с применением ключевого слова new, можно определить и разместить в памяти объект Car в одной строке кода:
Console.WriteLine("***** Fun with Class Types *****n");
Car myCar = new Car();
myCar.petName = "Fred";
В качестве альтернативы определение и размещение в памяти экземпляра класса может осуществляться в отдельных строках кода:
Console.WriteLine("***** Fun with Class Types *****n");
Car myCar;
myCar = new Car();
myCar.petName = "Fred";
Здесь первый оператор кода просто объявляет ссылку на определяемый объект типа Car. Ссылка будет указывать на действительный объект в памяти только после ее явного присваивания.
В любом случае к настоящему моменту мы имеем простейший класс, в котором определено несколько элементов данных и ряд базовых операций. Чтобы расширить функциональность текущего класса Car, необходимо разобраться с ролью конструкторов.
Понятие конструкторов
Учитывая наличие у объекта состояния (представленного значениями его переменных-членов), обычно желательно присвоить подходящие значения полям объекта перед тем, как работать с ним. В настоящее время класс Car требует присваивания значений полям petName и currSpeed по отдельности. Для текущего примера такое действие не слишком проблематично, поскольку открытых элементов данных всего два. Тем не менее, зачастую класс содержит несколько десятков полей, с которыми надо что-то делать. Ясно, что было бы нежелательно писать 20 операторов инициализации для всех 20 элементов данных.