Чтение онлайн

на главную - закладки

Жанры

Полное руководство. С# 4.0
Шрифт:

// Вывести символьную строку в обратном порядке, используя рекурсию. using System;

class RevStr { // Вывести символьную строку в обратном порядке. public void DisplayRev(string str) { if(str.Length > 0) DisplayRev(str.Substring(1, str.Length-1)); else return; Console.Write(str[0]); } }

class RevStrDemo { static void Main { string s = "Это тест"; RevStr rsOb = new RevStr; Console.WriteLine("Исходная строка: " + s); Console.Write("Перевернутая строка: "); rsOb.DisplayRev(s); Console.WriteLine; } } Вот к какому результату приводит выполнение этого кода.

Исходная строка: Это тест Перевернутая строка: тсет отЭ Всякий раз, когда вызывается метод DisplayRev, в нем происходит проверка длины символьной строки, представленной аргументом str. Если длина строки не равна нулю, то метод DisplayRev вызывается рекурсивно с новой строкой, кото рая меньше исходной строки на один символ. Этот процесс повторяется до тех пор, пока данному методу не будет передана строка нулевой длины. После этого начнет ся раскручиваться в

обратном порядке механизм всех рекурсивных вызовов метода DisplayRev. При возврате из каждого такого вызова выводится первый символ строки, представленной аргументом str, а в итоге вся строка выводится в обратном порядке. Рекурсивные варианты многих процедур могут выполняться немного медленнее, чем их итерационные эквиваленты из-за дополнительных затрат системных ресурсов на неоднократные вызовы метода. Если же таких вызовов окажется слишком много, то в конечном итоге может быть переполнен системный стек. А поскольку параметры и локальные переменные рекурсивного метода хранятся в системном стеке и при каж дом новом вызове этого метода создается их новая копия, то в какой-то момент стек может оказаться исчерпанным. В этом случае возникает исключительная ситуация, и общеязыковая исполняющая среда (CLR) генерирует соответствующее исключение. Но беспокоиться об этом придется лишь в том случае, если рекурсивная процедура выполняется неправильно. Главное преимущество рекурсии заключается в том, что она позволяет реализовать некоторые алгоритмы яснее и проще, чем итерационным способом. Например, ал горитм быстрой сортировки довольно трудно реализовать итерационным способом. А некоторые задачи, например искусственного интеллекта, очевидно, требуют именно рекурсивного решения. При написании рекурсивных методов следует непременно указать в соответству ющем месте условный оператор, например if, чтобы организовать возврат из мето да без рекурсии. В противном случае возврата из вызванного однажды рекурсивного метода может вообще не произойти. Подобного рода ошибка весьма характерна для реализации рекурсии в практике программирования. В этом случае рекомендуется пользоваться операторами, содержащими вызовы метода WriteLine, чтобы сле дить за происходящим в рекурсивном методе и прервать его выполнение, если в нем обнаружится ошибка. ## Применение ключевого слова static Иногда требуется определить такой член класса, который будет использоваться не зависимо от всех остальных объектов этого класса. Как правило, доступ к члену клас са организуется посредством объекта этого класса, но в то же время можно создать член класса для самостоятельного применения без ссылки на конкретный экземпляр объекта. Для того чтобы создать такой член класса, достаточно указать в самом начале его объявления ключевое слово static. Если член класса объявляется как static, то он становится доступным до создания любых объектов своего класса и без ссылки на какой-нибудь объект. С помощью ключевого слова static можно объявлять как пере менные, так и методы. Наиболее характерным примером члена типа static служит метод Main, который объявляется таковым потому, что он должен вызываться опе рационной системой в самом начале выполняемой программы. Для того чтобы воспользоваться членом типа static за пределами класса, доста точно указать имя этого класса с оператором-точкой. Но создавать объект для этого не нужно. В действительности член типа static оказывается доступным не по ссылке на объект, а по имени своего класса. Так, если требуется присвоить значение 10 перемен ной count типа static, являющейся членом класса Timer, то для этой цели можно воспользоваться следующей строкой кода.

Timer.count = 10; Эта форма записи подобна той, что используется для доступа к обычным перемен ным экземпляра посредством объекта, но в ней указывается имя класса, а не объекта. Аналогичным образом можно вызвать метод типа static, используя имя класса и оператор-точку. Переменные, объявляемые как static, по существу, являются глобальными. Ког да же объекты объявляются в своем классе, то копия переменной типа static не создается. Вместо этого все экземпляры класса совместно пользуются одной и той же переменной типа static. Такая переменная инициализируется перед ее примене нием в классе. Когда же ее инициализатор не указан явно, то она инициализируется нулевым значением, если относится к числовому типу данных, пустым значением, если относится к ссылочному типу, или же логическим значением false, если отно сится к типу bool. Таким образом, переменные типа static всегда имеют какое-то значение. Метод типа static отличается от обычного метода тем, что его можно вызывать по имени его класса, не создавая экземпляр объекта этого класса. Пример такого вызова уже приводился ранее. Это был метод Sqrt типа static, относящийся к классу System.Math из стандартной библиотеки классов С#. Ниже приведен пример программы, в которой объявляются переменная и метод типа static.

// Использовать модификатор static. using System;

class StaticDemo { // Переменная типа static. public static int Val = 100; // Метод типа static. public static int ValDiv2 { return Val/2; } }

class SDemo { static void Main { Console.WriteLine("Исходное значение переменной " + "StaticDemo.Val равно " + StaticDemo.Val); StaticDemo.Val = 8; Console.WriteLine("Текущее значение переменной" + "StaticDemo.Val равно " + StaticDemo.Val); Console.WriteLine("StaticDemo.ValDiv2: " + StaticDemo.ValDiv2); } } Выполнение этой программы приводит к следующему результату.

Исходное значение переменной StaticDemo.Val равно 100 Текущее значение переменной StaticDemo.Val равно 8 StaticDemo.ValDiv2: 4 Как следует из приведенного выше результата, переменная типа static инициа лизируется до создания любого объекта ее класса. На применение методов типа static накладывается ряд следующих ограничений. * В методе типа static должна отсутствовать ссылка this, поскольку такой метод не выполняется относительно какого-либо объекта. * В методе типа static допускается непосредственный вызов только других методов типа static, но не метода экземпляра из того самого же класса. Дело в том, что методы экземпляра оперируют конкретными объектами, а метод типа static не вызывается для объекта. Следовательно, у такого метода отсутствуют объекты, которыми он мог бы оперировать. * Аналогичные ограничения накладываются на данные типа static. Для метода типа static непосредственно доступными оказываются только другие данные типа static, определенные в его классе. Он, в частности, не может оперировать переменной экземпляра своего класса, поскольку у него отсутствуют объекты, которыми он мог бы оперировать. Ниже приведен пример класса, в котором недопустим метод ValDivDenom типа static.

class StaticError { public int Denom = 3; // обычная переменная экземпляра public static int Val = 1024; // статическая переменная / Ошибка! Непосредственный доступ к нестатической переменной из статического метода недопустим. / static int ValDivDenom { return Val/Denom; // не подлежит компиляции! } } В данном примере кода Denom является обычной переменной, которая недоступна из метода типа static. Но в то же время в этом методе можно воспользоваться пере менной Val, поскольку она объявлена как static. Аналогичная ошибка возникает при попытке вызвать нестатический метод из ста тического метода того же самого класса, как в приведенном ниже примере.

using System;

class AnotherStaticError { // Нестатический метод. void NonStaticMeth { Console.WriteLine("В методе NonStaticMeth."); } / Ошибка! Непосредственный вызов нестатического метода из статического метода недопустим. / static void staticMeth { NonStaticMeth; // не подлежит компиляции! } } В данном случае попытка вызвать нестатический метод (т.е. метод экземпляра) из статического метода приводит к ошибке во время компиляции. Следует особо подчеркнуть, что из метода типа static нельзя вызывать мето ды экземпляра и получать доступ к переменным экземпляра его класса, как это обычно делается посредством объектов данного класса. И объясняется это тем, что без указания конкретного объекта переменная или метод экземпляра оказываются недоступными. Например, приведенный ниже фрагмент кода считается совершенно верным.

class MyClass { // Нестатический метод. void NonStaticMeth { Console.WriteLine("В методе NonStaticMeth."); } / Нестатический метод может быть вызван из статического метода по ссылке на объект. / public static void staticMeth(MyClass ob) { ob.NonStaticMeth; // все верно! } } В данном примере метод NonStaticMeth вызывается из метода staticMeth по ссылке на объект ob типа MyClass. Поля типа static не зависят от конкретного объекта, и поэтому они удобны для хранения информации, применимой ко всему классу. Ниже приведен пример про граммы, демонстрирующей подобную ситуацию. В этой программе поле типа static служит для хранения количества существующих объектов.

// Использовать поле типа static для подсчета // экземпляров существующих объектов. using System;

class CountInst { static int count = 0; // Инкрементировать подсчет, когда создается объект. public CountInst { count++; } // Декрементировать подсчет, когда уничтожается объект. ~Countlnst { count--; } public static int GetCount { return count; } }

class CountDemo { static void Main { CountInst ob; for(int i=0; i < 10; i++) { ob = new CountInst; Console.WriteLine("Текущий подсчет: " + CountInst.GetCount); } } } Выполнение этой программы приводит к следующему результату.

Текущий подсчет: 1 Текущий подсчет: 2 Текущий подсчет: 3 Текущий подсчет: 4 Текущий подсчет: 5 Текущий подсчет: 6 Текущий подсчет: 7 Текущий подсчет: 8 Текущий подсчет: 9 Текущий подсчет: 10 Всякий раз, когда создается объект типа CountInst, инкрементируется поле count типа static. Но всякий раз, когда такой объект утилизируется, поле count декре ментируется. Следовательно, поле count всегда содержит количество существующих в настоящий момент объектов. И это становится возможным только благодаря исполь зованию поля типа static. Аналогичный подсчет нельзя организовать с помощью переменной экземпляра, поскольку он имеет отношение ко всему классу, а не только к конкретному экземпляру объекта этого класса. Ниже приведен еще один пример применения статических членов класса. Ранее в этой главе было показано, как объекты создаются с помощью фабрики класса. В том примере фабрика была нестатическим методом, а это означало, что фабричный метод можно было вызывать только по ссылке на объект, который нужно было предвари тельно создать. Но фабрику класса лучше реализовать как метод типа static, что даст возможность вызывать этот фабричный метод, не создавая ненужный объект. Именно это улучшение и отражено в приведенном ниже измененном примере программы, реализующей фабрику класса.

Поделиться:
Популярные книги

Возвышение Меркурия

Кронос Александр
1. Меркурий
Фантастика:
героическая фантастика
попаданцы
аниме
5.00
рейтинг книги
Возвышение Меркурия

Смертник из рода Валевских. Книга 1

Маханенко Василий Михайлович
1. Смертник из рода Валевских
Фантастика:
фэнтези
рпг
аниме
5.40
рейтинг книги
Смертник из рода Валевских. Книга 1

Начальник милиции

Дамиров Рафаэль
1. Начальник милиции
Фантастика:
попаданцы
альтернативная история
5.00
рейтинг книги
Начальник милиции

Счастливый торт Шарлотты

Гринерс Эва
Любовные романы:
любовно-фантастические романы
5.00
рейтинг книги
Счастливый торт Шарлотты

Я снова не князь! Книга XVII

Дрейк Сириус
17. Дорогой барон!
Фантастика:
юмористическое фэнтези
попаданцы
аниме
5.00
рейтинг книги
Я снова не князь! Книга XVII

Вечный Данж VI

Матисов Павел
6. Вечный Данж
Фантастика:
фэнтези
7.40
рейтинг книги
Вечный Данж VI

Последний попаданец 3

Зубов Константин
3. Последний попаданец
Фантастика:
фэнтези
юмористическое фэнтези
рпг
5.00
рейтинг книги
Последний попаданец 3

Не грози Дубровскому!

Панарин Антон
1. РОС: Не грози Дубровскому!
Фантастика:
фэнтези
попаданцы
аниме
5.00
рейтинг книги
Не грози Дубровскому!

Возвышение Меркурия. Книга 12

Кронос Александр
12. Меркурий
Фантастика:
героическая фантастика
попаданцы
аниме
5.00
рейтинг книги
Возвышение Меркурия. Книга 12

Найди меня Шерхан

Тоцка Тала
3. Ямпольские-Демидовы
Любовные романы:
современные любовные романы
короткие любовные романы
7.70
рейтинг книги
Найди меня Шерхан

Ваше Сиятельство 2

Моури Эрли
2. Ваше Сиятельство
Фантастика:
фэнтези
альтернативная история
аниме
5.00
рейтинг книги
Ваше Сиятельство 2

Адепт: Обучение. Каникулы [СИ]

Бубела Олег Николаевич
6. Совсем не герой
Фантастика:
фэнтези
попаданцы
9.15
рейтинг книги
Адепт: Обучение. Каникулы [СИ]

Комбинация

Ланцов Михаил Алексеевич
2. Сын Петра
Фантастика:
попаданцы
альтернативная история
5.00
рейтинг книги
Комбинация

Мой любимый (не) медведь

Юнина Наталья
Любовные романы:
современные любовные романы
7.90
рейтинг книги
Мой любимый (не) медведь