Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:

MM2 TP

.pdf
Скачиваний:
9
Добавлен:
09.02.2015
Размер:
3.63 Mб
Скачать

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

пятой выделена на отдельную строку, чтобы сделать ее более заметной.

Следует отметить, что если файл ввода не содержит никаких символов, то условие в

WHILE или FOR не выполнится при самом первом обращении к GETCHAR, и, следова-

тельно , программа выдаст нуль, т.е. правильный ответ. Это важное замечание. Одним из положительных свойств операторов WHILE и FOR является то, что они проверяют усло-

вие в начале цикла, т.е. до выполнения тела.

Следующая программа подсчитывает количество строк во введенной информации.

Предполагается, что строки ввода заканчиваются символом новой строки \N.

/* EX0009.C */ #include <stdio.h> int main() {

int c,nl;

nl = 0;

while ((c = getchar()) != EOF) if (c == '\n')

++nl; printf("%d\n", nl); return(0);

}

Тело WHILE теперь содержит оператор IF , который в свою очередь управляет опера-

тором увеличения ++NL. Оператор IF проверяет заключенное в круглые скобки условие и,

если оно истинно, выполняет следующий за ним оператор /или группу операторов, заклю-

ченных в фигурные скобки/. В тексте программы использован сдвиг вправо, чтобы пока-

зать, что чем управляет.

Удвоенный знак равенства == является обозначением в языке C для операции провер-

ки равенства. Этот символ введен для того, чтобы отличать проверку на равенство от одиночного =, ис-

пользуемого при присваивании.

Любой отдельный символ может быть записан внутри одиночных кавычек, и при этом ему соответствует значение, равное численному значению этого символа в машинном на-

боре символов; это называется символьной константой. Так, например, 'A' - символьная константа; ее значение в наборе символов ASCII /американский стандартный код для об-

мена информацией/ равно 65, внутреннему представлению символа а. Конечно, 'A' пред-

почтительнее, чем 65: его смысл очевиден.

Специальные последовательности, используемые в символьных строках, также зани-

мают законное место среди символьных констант. Так в проверках и арифметических вы-

ражениях '\N' представляет значение символа новой строки. Необходимо твердо уяснить,

что '\N' - отдельный символ, который в выражениях эквивалентен одиночному целому; с

другой стороны "\N" - это символьная строка, которая содержит только один символ.

Упражнение 6.

Написать программу для подсчета пробелов, табуляций и новых строк.

Упражнение 7.

Написать программу, которая копирует ввод на вывод, заменяя при этом каждую по-

следовательность из одного или более пробелов на один пробел.

Следующая программа подсчитывает количество строк, слов и символов, используя при этом весьма широкое определение, по которому словом является любая последова-

тельность символов, не содержащая пробелов, табуляций или новых строк.

/* EX0010.C */ #include <stdio.h> #define YES 1 #define NO 0

int main() {

int c, nc, nl, nw, inword;

inword = NO;

nl = nw = nc = 0;

while((c = getchar()) != EOF) { ++nc;

if (c == '\n') ++nl;

if (c == ' ' || c == '\n' || c == '\t') inword = NO;

else if (inword == NO) { inword = YES; ++nw;

}

}

printf("%d %d %d\n", nl, nw, nc); return(0);

}

Каждый раз, когда программа встречает первый символ слова, она увеличивает счет-

чик числа слов на единицу. Переменная INWORD следит за тем, находится ли программа в настоящий момент внутри слова или нет. Сначала этой переменной присваивается " не в слове", чему соответствует значение NO. Целесообразно отдавать предпочтение символи-

ческие константам YES и NO по сравнению с литерными значениям 1 и 0, потому что они делают программу более удобной для чтения. Кроме того в программы, где числа фигури-

руют только в качестве символьных констант, гораздо легче вносить изменения.

Строка

NL = NW = NC = 0;

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

сваивания проводятся последовательно справа налево. Таким образом, дело обстоит так,

как если бы мы написали

NC = (NL = (NW = 0));

Новая операция || Означает ИЛИ , так что строка

IF( C == ' ' || C == '\N' || C == '\T')

говорит "если С - пробел, или С - символ новой строки, или С -табуляция

..."./специальная последовательность \T является изображением символа табуляции/.

Наряду с новой операцией ||, в языке С имеется операция && (И). Выражения, связан-

ные операциями && или ||, рассматриваются слева на право, и при этом гарантируется,

что оценивание выражений будет прекращено, как только станет ясно, является ли все

выражение истинным или ложным. Так, если C оказывается пробелом, то нет никакой не-

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

Рассматриваемая программа демонстрирует оператор ELSE языка C, который указыва-

ет то действие, которое должно выполняться, если условие, содержащееся в операторе IF,

окажется ложным.

Общая форма условного оператора такова:

IF (выражение)

оператор-1 ELSE оператор-2

Выполняется один и только один из двух операторов, связанных с конструкцией IFELSE. Если выражение истинно, выполняется оператор-1; если нет - выполняется опера-

тор-2.

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

Упражнение 8.

Написать программу, которая будет печатать слова из ввода, причем по одному на строку.

Следующая программа предназначена для подсчета числа символов раздельно по сле-

дующим категориям: раздельно каждая цифра; символы пустых промежутков/пробел, та-

буляции, новая строка/; всех остальных символов (всего 12 категорий). Такая задача не-

сколько искусственна, но она позволяет проиллюстрировать в одной программе сразу не-

сколько аспектов языка C.

Для хранения количества появлений символов каждой категории можно использовать

12 разных переменных. Однако в подобных случаях удобнее использовать специальную структуру данных, которая в программировании называется массив. Формально массив это именованная n – мерная упорядоченная совокупность однотипных элементов. Термин

УПОРЯДОЧЕННАЯ говорит о том, что для обращения к элементу массива необходимо указать имя массива и числовую координату элемента, в частности, номер по порядку.

Итак, переходим к рассмотрению одного из возможных вариантов программы:

/* EX0011.c */ #include <stdio.h> int main() {

int c, i, nwhite, nother; int ndigit[10];

nwhite = nother = 0; for (i = 0; i < 10; ++i)

ndigit[i] = 0;

while ((c = getchar()) != EOF) if (c >= '0' && c <= '9')

++ndigit[c - '0'];

else if(c == ' ' || c == '\n' || c == '\t') ++nwhite;

else

++nother;

printf("digits =");

for (i = 0; i < 10; ++i)

printf(" %d", ndigit[i]);

printf("\nwhite space = %d, other = %d\n", nwhite, nother); return(0);

}

Описание

INT NDIGIT[10];

объявляет, что NDIGIT является массивом из десяти целых. В языке C индексы масси-

ва всегда начинаются с нуля /а не с 1/, так что элементами массива являются NDIGIT[0],

NDIGIT[1],..., NDIGIT[9]. эта особенность отражена в циклах FOR, которые инициализи-

руют и печатают массив.

Индекс может быть любым целым выражением, которое, конечно, может включать целые переменные, такие как I, и целые константы.

Эта конкретная программа сильно опирается на свойства символьного представления цифр. Так, например, в программе проверка

IF( C >= '0' && C <= '9')

определяет, является ли символ в C цифрой, и если это так, то численное значение этой цифры определяется по формуле / C - '0'/. Такой способ работает только в том случае,

если значения символьных констант '0', '1' и т.д. положительны и расположены в порядке возрастания и нет ничего, кроме цифр, между константами '0' и '9'. На практике это спра-

ведливо для всех общепринятых наборов символов.

По определению перед проведением арифметических операций, в которых участвуют переменные типа CHAR и INT, все они преобразуются к типу INT, так что в арифметиче-

ских выражениях переменные типа CHAR по существу идентичны переменным типа INT.

Это вполне естественно и удобно; например, C -'0'- это целое выражение со значением между 0 и 9 в соответствии с тем, какой символ от '0' до '9' хранится в 'C', и, следова тель-

но, оно является подходящим индексом для массива NDIGIT.

Выяснение вопроса, является ли данный символ цифрой, символом пустого промежут-

ка или чем-либо еще, осуществляется последовательностью операторов

IF (C >= '0' && C <= '9') ++NDIGIT[C-'0'];

ELSE IF(C == ' ' \!\! C == '\N' \!\! C == '\T') ++NWHITE;

ELSE ++NOTHER;

Конструкция

IF (условие)

оператор

ELSE IF (условие)

оператор

ELSE

оператор

часто встречаются в программах как средство выражения ситуаций, в которых осуще-

ствляется выбор одного из нескольких возможных решений.

Программа движется сверху вниз до тех пор, пока не удовлетворится какое-нибудь ус-

ловие; тогда выполняется соответствующий 'оператор', и вся конструкция завершается.

Конечно, 'оператор' может состоять из нескольких операторов, заключенных в фигур-

ные скобки.

Если ни одно из условий не удовлетворяется , то выполняется 'оператор', стоящий по-

сле заключительного ELSE, если оно присутствует. Если последнее ELSE и соответст-

вующий 'оператор' опущены, то никаких действий не производится. Между начальным IF

и конечным ELSE может помещаться произвольное количество групп

ELSE IF (условие)

оператор

Упражнение 9.

Написать программу, печатающую гистограмму длин слов из ввода.

Гистограмму начертить горизонтально.

2.6Функции.

Возврат_к_оглавлению

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

ним содержанием. Использование функций является способом справиться с потенциаль-

ной сложностью больших программ.

Впредыдущих примерах использовались только стандартные функции: PRINTF, GETCHAR и PUTCHAR; теперь рассмотрим написание собственных.

ВC нет операции воз ведения в степень. Следующий пример иллюстрирует определе-

ние функции на при мере функции POWER(M,N), возводящей целое M в целую положи-

тельную степень N. Так значение POWER(2,5) равно 32. Конечно, эта функция не выпол-

няет всей работы операции возведения в степень, поскольку она действует только с поло-

жительными степенями небольших чисел.

Ниже приводится функция POWER и вызывающая ее основная программа.

/* EX0012.C/ */ int main() {

int i;

for(i = 0; i < 10; ++i)

printf("%d %d %d\n",i,power(2,i),power(-3,i)); return(0);

}

int power(x,n) int x,n;

{

int i, p; p = 1;

for (i = 1; i <= n; ++i) p = p * x;

return(p);

}

Все функции имеют одинаковый вид:

ИМЯ (СПИСОК АРГУМЕНТОВ, ЕСЛИ ОНИ ИМЕЮТСЯ)

ОПИСАНИЕ АРГУМЕНТОВ, ЕСЛИ ОНИ ИМЕЮТСЯ

{

ОПИСАНИЯ

ОПЕРАТОРЫ

}

Эти функции могут быть записаны в любом порядке и находиться в одном или двух исходных модулях.

В данном случае обе функции (MAIN и POWER) находятся в одном и том же исход-

ном модуле.

Функции, как и переменные, имеют тип. В данном случае POWER имеет целый тип,

т.к. имеется описание типа перед именем функции: INT POWER.

Функция POWER вызывается дважды в строке

PRINTF("%D %D %D\N",I,POWER(2,I),POWER(-3,I));

при каждом обращении функция POWER, получив два аргумента, возвращает целое значение, которое печатается в заданном формате. В выражениях POWER(2,I) является точно таким же целым, как 2 и I.

Аргументы, которые указываются при вызове функции называются фактическими ар-

гументами, те, которые указываются при описании функции – формальными.

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

INT X,N;

которая следует за именем функции.

Описания аргументов помещаются между списком аргументов и открывающейся ле-

вой фигурной скобкой; каждое описание заканчивается точкой с запятой. Имена, исполь-

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

Это верно и для переменных I и P; I в функции POWER никак не связано с I в функции

MAIN.

Значение, вычисленное функцией POWER, передаются в MAIN с помощью оператора

RETURN.

Аргументом оператора RETURN может быть любое выражение.

Упражнение 10.

Написать программу преобразования прописных букв из ввода в строчные, используя при этом функцию OWER(C).

2.7Аргументы функций. Передача по значению.

Возврат_к_оглавлению

В языке C все аргументы функций передаются "по значению". Это означает, что при вызове функции ее фактические аргументы копируются в локальные переменные.

Следствием такого соглашения является то, что в C вызванная функция не может из-

менить ПЕРЕМЕННУЮ из вызывающей функции; она может менять только свою собст-

венную временную копию.

Передача по значению, однако, не помеха, а весьма ценное качество. Оно обычно при-

водит к более компактным программам, содержащим меньше не относящихся к делу пе-

ременных, потому что с аргументами можно обращаться как с удобно инициализирован-

ными локальными переменными вызванной процедуры. Рассмотрим, например, вариант функции POWER использующей это обстоятельство

int power(x,n) int x,n;

{

int p;

for (p = 1; n > 0; --n) p = p * x;

return (p);

}

Аргумент N используется как временная переменная; из нее вычитается единица до тех пор, пока ее значение не станет нулем.

Переменная I здесь больше не нужна. Чтобы ни происходило с N внутри POWER это никак не влияет на аргумент, с которым первоначально обратились к функции POWER.

При необходимости все же можно добиться, чтобы функция изменила переменную из вызывающей программы. Эта достигается с помощью использования такого понятия как УКАЗАТЕЛЬ, которое очень близко по смыслу с понятием адрес .

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

сив, а адрес его первого элемента. Очевидно, что этого достаточно для доступа к любому элементу массива при указании номера индекса. Следовательно, при строгом формальном выполнении передачи параметров по значению, массивы фактически передаются по ссыл-

ке и модификация элементов массива внутри вызванной функции ощущается в вызвавшей функции.

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]