WikiDer > Аргумент по умолчанию

Default argument

В компьютерное программирование, а аргумент по умолчанию является аргумент к функция что программист не обязан указывать. В большинстве языков программирования функции могут принимать один или несколько аргументов. Обычно каждый аргумент необходимо указывать полностью (как в случае Язык программирования C[1]). Более поздние языки (например, в C ++) позволяют программисту указывать аргументы по умолчанию, которые всегда имеют значение, даже если оно не указано при вызове функции.

Аргументы по умолчанию в C ++

Рассмотрим следующее объявление функции:

int MyFunc(int а, int б, int c = 12);

Эта функция принимает три аргумента, из которых последний имеет значение по умолчанию - двенадцать. Программист может вызвать эту функцию двумя способами:

int результат = MyFunc(1, 2, 3);результат = MyFunc(1, 2);

В первом случае значение аргумента, называемого c указано как обычно. Во втором случае аргумент опускается, а значение по умолчанию 12 будет использоваться вместо этого.

Невозможно узнать, был ли аргумент указан вызывающей стороной или использовалось значение по умолчанию.

Вышеупомянутый метод особенно полезен, когда нужно установить критерии по умолчанию, чтобы функция могла вызываться с параметрами или без них. Примите во внимание следующее:

пустота Распечатать(стандартное::течь& транслировать = стандартное::cout) {  // Это выводит сообщение в указанный поток.  транслировать << "Привет, мир!";}

Вызов функции:

Распечатать();

по умолчанию будет печатать "Привет, мир! "в стандартный вывод std :: cout (обычно экран). С другой стороны, любой объект типа std :: ostream теперь можно передать в ту же функцию, и функция будет печатать в заданный поток, а не в стандартный вывод.

Распечатать(стандартное::Cerr);

Поскольку значения аргументов по умолчанию «заполняются» на месте вызова, а не в теле вызываемой функции, виртуальные функции берут свои значения аргументов по умолчанию из статического типа указателя или ссылки, через которую выполняется вызов, а не из динамического типа объекта, предоставляющего тело виртуальной функции.

структура Основание {  виртуальный стандартное::пара<int, int> Фу(int Икс = 1) {    возвращаться {Икс, 1};  }};структура Полученный : общественный Основание {  стандартное::пара<int, int> Фу(int Икс = 2) отменять {    возвращаться {Икс, 2};  }};int главный() {  Полученный d;  Основание& б = d;  утверждать(d.Фу() == стандартное::make_pair(2, 2));  утверждать(б.Фу() == стандартное::make_pair(1, 2));}

Перегруженные методы

Некоторые языки, например Ява, не имеют аргументов по умолчанию. Однако такое же поведение можно смоделировать, используя перегрузка метода создавать перегруженные одноименные методы, принимающие разное количество аргументов; а версии с меньшим количеством аргументов просто вызывают версии с большим количеством аргументов с аргументами по умолчанию в качестве отсутствующих аргументов:

int MyFunc(int а, int б) { возвращаться MyFunc(а, б, 12); }int MyFunc(int а, int б, int c) { / * здесь основная реализация * / }

Однако помимо несколько других недостатков, поскольку аргументы по умолчанию не моделируются в системе типов, тип обратного вызова (он же функция высшего порядка) не может выразить, что он принимает любую из перегрузок, или имитирует аргументы по умолчанию с перегруженными функциями. В то время как, в JavaScript определение неперегруженной функции может заменить значение по умолчанию, когда входное значение неопределенный (независимо от того, был ли он неявно неопределенный через отсутствие аргумента на сайте вызова или явно переданный неопределенный ценить); который моделируется как необязательный тип параметра аргумента ?: в TypeScript. Решение JavaScript не разрешается статически (т.е. не во время компиляции, поэтому TypeScript моделирует только дополнительные возможности, а не значения по умолчанию в сигнатуре типа функции), таким образом, влечет дополнительные накладные расходы времени выполнения, хотя он обеспечивает большую гибкость в том, что обратные вызовы могут независимо контролировать свои значения по умолчанию, а не централизованно продиктовано (подпись типа обратного вызова в) подписью типа функции, которая вводит обратный вызов. Решение TypeScript может быть смоделировано на Java с Необязательный тип, кроме аналога неявного неопределенный для каждого отсутствующего аргумента явное Необязательно. absent () на месте звонка.

Оценка

Для каждого вызова функции значения аргумента по умолчанию должны быть переданы вызываемой функции.

Если значение аргумента по умолчанию содержит побочные эффекты, это важно при оценке этих побочных эффектов - один раз для всей программы (во время синтаксического анализа, компиляции или загрузки) или один раз на вызов функции, во время вызова.

Python - известный язык, который оценивает выражения в аргументах по умолчанию один раз во время оценки объявления функции. Если требуется оценка для каждого вызова функции, ее можно воспроизвести, задав аргументом по умолчанию контрольное значение, Такие как Никто, а затем наличие в теле функции оценки побочных эффектов значения по умолчанию, только если было передано значение дозорного.

Например:

импорт случайныйdef жаждущий(а=случайный.случайный()):    возвращаться аИкс = жаждущий()у = жаждущий()утверждать Икс == уdef ленивый(а=Никто):    если а является Никто:        а = случайный.случайный()    возвращаться аИкс = ленивый()у = ленивый()утверждать Икс != у

Степень

Обычно аргумент по умолчанию будет вести себя идентично аргументу, переданному параметром, или локальной переменной, объявленной в начале функции, и будет иметь то же объем и степень (время жизни) в качестве параметра или другой локальной переменной, а именно автоматическая переменная который освобождается при завершении функции.

В других случаях вместо этого статически может быть назначен аргумент по умолчанию. Если переменная изменяема, она будет сохранять свое значение во всех вызовах функций, как с статическая переменная.

Такое поведение встречается в Python для изменяемых типов, таких как списки. Как и в случае с оценкой, чтобы гарантировать ту же степень, что и у локальной переменной, можно использовать контрольное значение:

def жаждущий(а=[]):    возвращаться аИкс = жаждущий()Икс += [1]утверждать жаждущий() == [1]def ленивый(а=Никто):    если а является Никто:        а = []    возвращаться аИкс = ленивый()Икс += [1]утверждать ленивый() == []

Рекомендации

  1. ^ Лесли, Мартин. «Параметры по умолчанию». Справочник по программированию на C ++. Архивировано из оригинал 9 октября 2011 г.. Получено 13 января 2012.