public static void SendAPersonByValue(Person p) {
// Изменяет ли это возраст 'р'?
p.age = 92;
// Увидит ли вызывающая сторона такие изменения?
p
}
Обратите внимание на то, что метод SendAPersonByValue пытается изменить получаемую ссылку Person на новый объект, а также изменить некоторые данные состояния. Давайте проверим работу этого метода, используя следующий метод Main.
static void Main(string[] args) {
// Передача ссылочных типов по значению.
Console.WriteLine("*** Передача объекта Person по значению ***");
Person fred = new Persоn("Фред", 2);
Console.WriteLine("Person до вызова по значению");
fred.PrintInfo
SendAPersonByValue(fred);
Console.WriteLine("Persоn после вызова по значению");
fred.PrintInfо;
}
На рис. 3.15 показан соответствующий вывод.
Рис. 3.15. Передача ссылочных типов по значению блокирует соответствующую ссылку
Как видите, значение возраста (age) изменяется. Кажется, такое поведение при передаче параметра противоречит самому термину "по значению". Если вы способны изменить состояние получаемого объекта Person, что же все-таки копируется? Ответ здесь следующий: в объект вызывающей стороны копируется
Передача ссылочных типов по ссылке
Теперь предположим, что у нас есть метод SendAPersonByReference, который передает ссылочный тип по ссылке (обратите внимание на то, что здесь присутствует модификатор параметра ref).
public static void SendAPersonByReference(ref Person p) {
// Изменение некоторых данных 'р'.
p.age = 122;
// Теперь 'р' указывает на новый объект в динамической памяти!
р = new Person("Никки", 222);
}
Как вы можете догадаться сами, это обеспечивает вызывающей стороне полную гибкость в управлении входными параметрами. Вызывающая сторона не только может изменить состояние объекта, но и переопределить ссылку так, чтобы она указывала на новый тип Person. Рассмотрите следующий вариант.
static void Main(string[] args) {
// Передача ссылочных типов по ссылке.
Console.WriteLine("\n*** Передача объекта Person по ссылке ***");
Person mel = new Person("Мэл", 23);
Console.WriteLine("Person до вызова по ссылке:");
mel.PrintInfo;
SendAPersonByReference(ref mel);
Console.WriteLine("Person после вызова по ссылке:");
mel.PrintInfо;
}
Из рис. 3.16 видно, что тип с именем Мэл возвращается после вызова как тип с именем Никки.
Рис. 3.16. Передача ссылочных типов по ссылке позволяет перенаправить ссылку
Золотым правилом при передаче ссылочных типов по ссылке является следующее.
• Если ссылочный тип передается по ссылке, то вызывающая сторона может изменить не только состояние данных соответствующего объекта, но
Исходный код. Проект RefTypeValTypeParams размещен в подкаталоге, соответствующем главе 3.
Типы, характеризуемые значениями, и ссылочные типы: заключительные замечания
Чтобы завершить обсуждение данной темы, изучите информацию табл. 3.8, в которой приводится краткая сводка основных отличий между типами, характеризуемыми значением, и ссылочными типами.
Таблица 3.8. Сравнение типов, характеризуемых значением, и ссылочных типов