int n = 49, m = 10, i, j, k, l;
i = n % m; /* 9 */
j = n % (-m); /* 9 */
k = (—n) % m; /* -9 */
l = (-n) % (-m); /* -9 */
К аддитивным операциям относятся сложение (+) и вычитание (-). Операнды могут быть целого или плавающего типов. В некоторых случаях над операндами аддитивных операций выполняются общие арифметические преобразования. При аддитивных операциях не контролируется переполнение результата и сообщение об ошибке не выдается.
Пример переполнения результата:
int i=30000, j=30000, k;
k=i+j;
В результате сложения переменная к получит значение, равное -5536 (принципы выполнения суммирования двоичных чисел в дополнительном двоичном коде приведены в
Результатом выполнения операции сложения является сумма двух операндов. Операнды могут быть представлены переменными целого или плавающего типа. При этом операнды могут быть доступны как непосредственно, так и при помощи указателей.
Когда производится суммирование указателя с переменной целого типа, то к содержимому указателя прибавляется произведение количества ячеек памяти, зависящее от типа указателя, на значение целочисленной переменной. Например:
char *a=5; //Объявить указатель и настроить на ячейку памяти с адресом 5
int *b=5; //Объявить указатель и настроить на ячейку памяти с адресом 5
long *c=5; //Объявить указатель и настроить на ячейку памяти с адресом 5
а=а+5; //Увеличить значение указателя на 5 (адрес 10)
t>=b+5; //Увеличить значение указателя на 5 (адрес 15)
с=с+5; //Увеличить значение указателя на 5 (адрес 25)
Когда указатель складывается с целым числом, то в результате будет содержать адрес переменной, расположенной на целое число ячеек дальше от исходного адреса. Новое значение указателя адресует тот же самый тип данных, что и исходный указатель.
1. Оба операнда — целого или плавающего типа.
2. Оба операнда являются указателями на один и тот же тип.
3. Первый операнд является указателем, а второй — целым числом.
Отметим, что операции сложения и вычитания над адресами в единицах, отличных от длины типа, могут привести к непредсказуемым результатам.
Пример работы с указателем:
double d[10],* u;
int i;
u = d+2; /*
i = u-d; /*
Операции сдвига осуществляют смещение операнда влево («) или вправо (») на число битов, задаваемое вторым операндом. Оба операнда должны быть целыми величинами. Выполняются обычные арифметические преобразования.
При сдвиге влево правые освобождающиеся биты устанавливаются в ноль. При сдвиге вправо метод заполнения освобождающихся левых битов зависит от типа первого операнда. Если тип сдвигаемой переменной unsigned, то свободные левые биты устанавливаются в ноль. В случае использования знаковой переменной они заполняются копией знакового бита. Результат операции сдвига не определен, если второй операнд отрицательный.
Преобразования, выполненные операциями сдвига, не обеспечивают обработку ситуаций переполнения. Информация теряется, если результат операции сдвига не может быть представлен типом первого операнда, после преобразования.
Отметим, что сдвиг влево соответствует умножению первого операнда на 2 в степени, равной второму операнду, а сдвиг вправо соответствует делению первого операнда на 2 в степени, равной второму операнду.
Примеры использования операции сдвига:
int i=0x1234, j, к ;
k=i«4; /* k="0x2340» */
j=i«8; /* j="0x3400» */
i=j»8; /* i = 0x0034 */
К поразрядным (побитовым) операциям относятся: операция поразрядного «И» (&), операция поразрядного «ИЛИ» (|) и операция поразрядного «исключающего ИЛИ» (^).
Операнды поразрядных операций могут быть любого целого типа. При необходимости над операндами выполняются преобразования по умолчанию, тип результата — это тип операндов после преобразования.