Смекни!
smekni.com

Методические указания к выполнению контрольных работ по дисциплине "Основы программирования" (стр. 17 из 40)

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

Пример 3-2. Примером может служить функция atoi, которая ставит в соответствие строке цифр ее численный эквивалент.

atoi(char s[]) // Преобразование s в целое

{

int i, n;

n = 0;

for ( i = 0; s[i]>='0' && s[i]<='9'; ++i)

n = 10 * n + s[i] - '0';

return(n);

}

Kак уже обсуждалось в главе 2, выражение s[i]-'0' имеет численное значение находящегося в s[i]символа, потому что значение символов '0', '1' и т.д. образуют возрастающую последовательность расположенных подряд целых положительных чисел.

Пример 3-3. Другой пример преобразования char в int дает функция lower, преобразующая данную прописную букву в строчную. Если выступающий в качестве аргумента символ не является прописной буквой, то lower возвращает его неизменным. Приводимая ниже программа справедлива только для набора символов ASCII.

lower(int c) // Преобразование прописных в строчные

{

if ( c >= 'A' && c <= 'Z' )

return( c + 'a' - 'A');

else // @ записано вместо 'a' строчного

return(c);

}

Эта функция правильно работает при коде ASCII, потому что численные значения, соответствующие в этом коде прописным и строчным буквам, отличаются на постоянную величину, а каждый алфавит является сплошным – между a и z нет ничего, кроме букв. Это последнее замечание для набора символов EBCDIC систем IBM 370 оказывается несправедливым, в силу чего эта программа на таких системах работает неправильно – она преобразует не только буквы.

При преобразовании символьных переменных в целые возникает один тонкий момент. Дело в том, что сам язык не указывает, должны ли переменным типа char соответствовать численные значения со знаком или без знака. Может ли при преобразовании char в int получиться отрицательное целое? К сожалению, ответ на этот вопрос меняется от машины к машине, отражая расхождения в их архитектуре. На некоторых машинах (PDP-11, например) переменная типа char, крайний левый бит которой содержит 1, преобразуется в отрицательное целое («знаковое расширение»). На других машинах такое преобразование сопровождается добавлением нулей с левого края, в результате чего всегда получается положительное число.

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


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

char c;

c = getchar();

if ( c == eof )

...

На машине, которая не осуществляет знакового расширения, переменная c всегда положительна, поскольку она описана как char, а так как eof отрицательно, то условие никогда не выполняется. Чтобы избежать такой ситуации, мы всегда предусмотрительно использовали int вместо char для любой переменной, получающей значение от getchar.

Основная же причина использования int вместо char не связана с каким-либо вопросом о возможном знаковом расширении. просто функция getchar должна передавать все возможные символы (чтобы ее можно было использовать для произвольного ввода) и, кроме того, отличающееся значение eof. Следовательно значение eof не может быть представлено как char, а должно храниться как int.

Другой полезной формой автоматического преобразования типов является то, что выражения отношения, подобные i>j, и логические выражения, связанные операциями && и ||, по определению имеют значение 1, если они истинны, и 0, если они ложны. Таким образом, присваивание:

isdigit = c >= '0' && c <= '9';

полагает isdigit равным 1, если c – цифра, и равным 0 в противном случае. (В проверочной части операторов if, while, for и т.д. «истина» просто означает «не нуль»).

Неявные арифметические преобразования работают в основном, как и ожидается. В общих чертах, если операция типа + или *, которая связывает два операнда (бинарная операция), имеет операнды разных типов, то перед выполнением операции «низший» тип преобразуется к «высшему» и получается результат «высшего» типа. Более точно, к каждой арифметической операции применяется следующая последовательность правил преобразования.

1. Типы char и short преобразуются в int, а float в double.

2. Затем, если один из операндов имеет тип double, то другой преобразуется в double, и результат имеет тип double.


3. В противном случае, если один из операндов имеет тип long, то другой преобразуется в long, и результат имеет тип long.

4. В противном случае, если один из операндов имеет тип unsigned, то другой преобразуется в unsigned и результат имеет тип unsigned.

5. В противном случае операнды должны быть типа int, и результат имеет тип int.

Подчеркнем, что все переменные типа float в выражениях преобразуются в double; в языке «C» вся плавающая арифметика выполняется с двойной точностью.

Преобразования возникают и при присваиваниях; значение правой части преобразуется к типу левой, который и является типом результата. Символьные переменные преобразуются в целые либо со знаковым расширением, либо без него, как описано выше. Обратное преобразование int в char ведет себя хорошо – лишние биты высокого порядка просто отбрасываются. Таким образом, если:

int i;

char c;

i = c;

c = i;

то значение символа c не изменяется. Это верно независимо от того, вовлекается ли знаковое расширение или нет.

Если х типа float, а i типа int, то как

х = i;

так и

i = х;

приводят к преобразованиям; при этом float преобразуется в int отбрасыванием дробной части. Тип double преобразуется во float округлением. Длинные целые преобразуются в более короткие целые и в переменные типа char посредством отбрасывания лишних битов высокого порядка.

Так как аргумент функции является выражением, то при передаче функциям аргументов также происходит преобразование типов: в частности, char и short становятся int, а float становится double. Именно поэтому мы описывали аргументы функций как int и double даже тогда, когда обращались к ним с переменными типа char и float.


Наконец, в любом выражении может быть осуществлено («принуждено») явное преобразование типа с помощью конструкции, называемой перевод (cast). В этой конструкции, имеющей вид:

(имя типа) выражение

выражение преобразуется к указанному типу по правилам преобразования, изложенным выше. Фактически точный смысл операции перевода можно описать следующим образом: выражение как бы присваивается некоторой переменной указанного типа, которая затем используется вместо всей конструкции. Например, библиотечная процедура sqrt ожидает аргумента типа double и выдаст бессмысленный ответ, если к ней по небрежности обратятся с чем-нибудь иным. Таким образом, если n – целое, то выражение:

sqrt((double)n)

до передачи аргумента функции sqrt преобразует n к типу double.

Отметим, что операция перевод преобразует значение n в надлежащий тип; фактическое содержание переменной n при этом не изменяется. Операция перевода имрация перевода имеет тот же уровень старшинства, что и другие унарные операции, как указывается в таблице в конце этой главы.

Упражнение 3-2. Составьте программу для функции htoi(s), которая преобразует строку шестнадцатеричных цифр в эквивалентное ей целое значение. При этом допустимыми цифрами являются цифры от 1 до 9 и буквы от а до f.

3.8. Операции увеличения и уменьшения

В языке «C» предусмотрены две необычные операции для увеличения и уменьшения значений переменных. Операция увеличения ++ добавляет 1 к своему операнду, а операция уменьшения -- вычитает 1. Мы часто использовали операцию ++ для увеличения переменных, как, например, в

if(c == '&bsol;n')

++i;

Необычный аспект заключается в том, что ++ и -- можно использовать либо как префиксные операции (перед переменной, как в ++n), либо как постфиксные (после переменной: n++). Эффект в обоих случаях состоит в увеличении N. Но выражение ++n увеличивает переменную N до использования ее значения, в то время как n++ увеличивает переменную N после того, как ее значение было использовано. Это означает, что в контексте, где используется значение переменной, а не только эффект увеличения, использование ++n и n++ приводит к разным результатам. Если n=5, то:

х = n++;

устанавливает х равным 5, а

х = ++n;

полагает х равным 6. В обоих случаях n становится равным 6. Операции увеличения и уменьшения можно применять только к переменным; выражения типа х=i+j)++ являются незаконными.

В случаях, где нужен только эффект увеличения, а само значение не используется, как, например, в:

if ( c == '&bsol;n' )

nal++;

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

Пример 3-4. Рассмотрим функцию squeeze(s,c), которая удаляет символ 'ж' из строки s, каждый раз, как он встречается. Обращение к функции может быть следующим: squeeze(s,'ж'). Текст функции выглядит так:

squeeze(char s[],int c)

{

int i, j;

for ( i = j = 0; s[i] != '&bsol;0'; i++)

if ( s[i] != c )

s[j++] = s[i];

s[j] = '&bsol;0';

}

Каждый раз, как встречается символ, отличный от 'ж', он копируется в текущую позицию j, и только после этого j увеличивается на 1, чтобы быть готовым для поступления следующего символа. Это в точности эквивалентно записи

if ( s[i] != c )

{

s[j] = s[i];

j++;

}

Пример 3-5. Подобную конструкцию дает функция getline, которую мы запрограммировали в главе 2, где можно заменить: