myXLapp.Range["C1", "С3"].Value = "OK"; В этом случае значения интервала ячеек электронной таблицы передаются с ис пользованием синтаксиса индексаторов, а заполнитель Туре.Missing уже не нужен, поскольку данный параметр теперь задается по умолчанию. Как правило, при определении в методе параметра ref приходится передавать ссылку на этот параметр. Но, работая с моделью СОМ, можно передавать параметру ref значение, не заключая его предварительно в оболочку объекта. Дело в том, что компилятор будет автоматически создавать временный аргумент, который уже заклю чен в оболочку объекта, и поэтому указывать параметр ref в списке аргументов уже не нужно. ## Дружественные сборки Одну сборку можно сделать дружественной по отношению к другой. Такой сборке доступны закрытые члены дружественной ей сборки. Благодаря этому средству стано вится возможным коллективное использование членов выбранных сборок, причем эти члены не нужно делать открытыми. Для того чтобы объявить дружественную сборку, необходимо воспользоваться атрибутом InternalsVisibleTo. ## Разные ключевые слова В заключение этой главы в частности и всей части I вообще будут вкратце представ лены ключевые слова, определенные в C# и не упоминавшиеся в предыдущих главах данной книги. ### Ключевое слов lock Ключевое слово lock используется при создании многопоточных программ. Под робнее оно рассматривается в главе 23, где речь пойдет о многопоточном программи ровании. Но ради полноты изложения ниже приведено краткое описание этого клю чевого слова. Программа на C# может состоять из нескольких потоков исполнения. В этом случае программа считается многопоточной, и отдельные ее части выполняются параллельно, т.е. одновременно и независимо друг от друга. В связи с такой организацией програм мы возникает особого рода затруднение, когда два потока пытаются воспользоваться ресурсом, которым можно пользоваться только по очереди. Для разрешения этого за труднения можно создать критический раздел кода, который будет одновременно вы полняться одним и только одним потоком. И это делается с помощью ключевого слова lock. Ниже приведена общая форма этого ключевого слова:
lock(obj) { // критический раздел кода } где obj обозначает объект, для которого согласуется блокировка кода. Если один поток уже вошел в критический раздел кода, то второму потоку придется ждать до тех пор, пока первый поток не выйдет из данного критического раздела кода. Когда же первый поток покидает критический раздел кода, блокировка снимается и предоставляется второму потоку. С этого момента второй поток может выполнять критический раздел кода. **ПРИМЕЧАНИЕ** Более подробно ключевое слово lock рассматривается в главе 23. ### Ключевое слово readonly Отдельное поле можно сделать доступным в классе только для чтения, объявив его как readonly. Значение такого поля можно установить только с помощью инициали затора, когда оно объявляется или же когда ему присваивается значение в конструк торе. После того как значение доступного только для чтения поля будет установлено, оно не подлежит изменению за пределами конструктора. Следовательно, поле типа readonly удобно для установки фиксированного значения с помощью конструктора. Такое поле можно, например, использовать для обозначения размера массива, кото рый часто используется в программе. Допускаются как статические, так и нестатиче ские поля типа readonly. **ПРИМЕЧАНИЕ** Несмотря на кажущееся сходство, поля типа readonly не следует путать с полями типа const, которые рассматриваются далее в этой главе. Ниже приведен пример применения поля с ключевым словом readonly.
// Продемонстрировать применение поля с ключевым словом readonly. using System;
class MyClass { public static readonly int SIZE = 10; }
class DemoReadOnly { static void Main { int[] source = new int[MyClass.SIZE]; int[] target = new int[MyClass.SIZE]; // Присвоить ряд значений элементам массива source. for(int i=0; i < MyClass.SIZE; i++) source[i] = i; foreach(int i in source) Console.Write(i + " "); Console.WriteLine; // Перенести обращенную копию массива source в массив target. for(int i = MyClass.SIZE-1, j = 0; i > 0; i--, j++) target[j] = source[i]; foreach(int i in target) Console.Write(i + " "); Console.WriteLine; // MyClass.SIZE = 100; // Ошибка!!! He подлежит изменению! }