Ранее в главе объяснялось, что ссылочные типы и типы значений могут передаваться методам как параметры. Тем не менее, передача ссылочного типа (например, класса) по ссылке совершенно отличается от его передачи по значению. Чтобы понять разницу, предположим, что есть простой класс Person
FunWithRefTypeValTypeParams:class Person
{
public string personName;
public int personAge;
// Constructors.
public Person(string name, int age)
{
personName = name;
personAge = age;
}
public Person{}
public void Display
{
Console.WriteLine("Name: {0}, Age: {1}", personName, personAge);
}
}
А что если мы создадим метод, который позволит вызывающему коду передавать объект Person
ref)?static void SendAPersonByValue(Person p)
{
// Изменить значение возраста в р?
p.personAge = 99;
// Увидит ли вызывающий код это изменение?
p = new Person("Nikki", 99);
}
Здесь видно, что метод SendAPersonByValue
Person новый объект Person, а также изменить некоторые данные состояния. Протестируем этот метод с помощью следующего кода:// Передача ссылочных типов по значению.
Console.WriteLine("***** Passing Person object by value *****");
Person fred = new Person("Fred", 12);
Console.WriteLine("\nBefore by value call, Person is:");
// Перед вызовом с передачей по значению
fred.Display;
SendAPersonByValue(fred);
Console.WriteLine("\nAfter by value call, Person is:");
// После вызова с передачей по значению
fred.Display;
Console.ReadLine;
Ниже показан результирующий вывод:
***** Passing Person object by value *****
Before by value call, Person is:
Name: Fred, Age: 12
After by value call, Person is:
Name: Fred, Age: 99
Легко заметить, что значение PersoneAge
Person прошла успешно, возникает вопрос: что же тогда было скопировано? Ответ: была получена копия ссылки на объект из вызывающего кода. Следовательно, раз уж метод SendAPersonByValue указывает на тот же самый объект, что и вызывающий код, становится возможным изменение данных состояния этого объекта. Нельзя лишь переустанавливать ссылку так, чтобы она указывала на какой-то Передача ссылочных типов по ссылке
Предположим, что имеется метод SendAPersonByReference
ref):static void SendAPersonByReference(ref Person p)
{
// Изменить некоторые данные в р.
p.personAge = 555;
// р теперь указывает на новый объект в куче!
p = new Person("Nikki", 999);
}
Как и можно было ожидать, вызываемому коду предоставлена полная свобода в плане манипулирования входным параметром. Вызываемый код может не только изменять состояние объекта, но и переопределять ссылку так, чтобы она указывала на новый объект Person
// Передача ссылочных типов по ссылке.
Console.WriteLine("***** Passing Person object by reference *****");
...
Person mel = new Person("Mel", 23);
Console.WriteLine("Before by ref call, Person is:");
// Перед вызовом с передачей по ссылке
mel.Display;
SendAPersonByReference(ref mel);
Console.WriteLine("After by ref call, Person is:");
Бьёрн Страуструп , Ирина Сергеевна Козлова , Бьерн Страуструп , Валерий Федорович Альмухаметов
Программирование, программы, базы данных / Базы данных / Программирование / Учебная и научная литература / Образование и наука / Книги по IT