Над объектами в языке Си могут выполняться различные операции:
- операции присваивания;
- операции отношения;
- арифметические;
- логические;
- сдвиговые операции.
Результатом выполнения операции является число.
Операции могут быть бинарными или унарными. Бинарные операции выполняются над двумя объектами, унарные — над одним.
Операция присваивания
Операция присваивания обозначается символом = и выполняется в 2 этапа:
- вычисляется выражение в правой части;
- результат присваивается операнду, стоящему в левой части:
объект = выражение;
Пример:
2
3
int b;
b = a + 2; // переменной b присваивается значение 6, вычисленное в правой части
В случае если объекты в левой и правой части операции присваивания имеют разные типы используется операция явного приведения типа.
объект = (тип)выражение;
Пример:
2
3
// Перед вычислением остатка от деления a приводится к целому типу
int b = (int)a % 2; // b = 1
Операции отношения
Основные операции отношения:
- == эквивалентно — проверка на равенство;
- != не равно — проверка на неравенство;
- < меньше;
- > больше;
- <=меньше или равно;
- >= больше или равно.
Операции отношения используются при организации условий и ветвлений. Результатом этих операций является 1 бит, значение которого равно 1, если результат выполнения операции — истина, и равно 0, если результат выполнения операции — ложь.
Арифметические операции
Основные бинарные операции, расположенные в порядке уменьшения приоритета:
- * — умножение;
- / — деление;
- + — сложение;
- — — вычитание;
- % — остаток от целочисленного деления.
Основные унарные операции:
- ++ — инкрементирование (увеличение на 1);
- –– — декрементирование (уменьшение на 1);
- — — изменение знака.
Результат вычисления выражения, содержащего операции инкрементирования или декрементирования, зависит от того, где расположен знак операции (до объекта или после него). Если операция расположена до объекта, то сначала происходит изменение значения переменной на 1, а потом это значение используется для выполнения следующих операций. Если операция ++ или — расположена после переменной, то сначала выполняется операция, а потом значение переменной изменяется на 1.
Пример:
2
3
4
5
int b = 3;
int c;
c = a * ++b;
// c=8, поскольку в операции умножения уже b=4
2
3
4
5
int b = 3;
int d;
d = a * b++;
// d=6, поскольку в операции умножения b=3, следующим действием будет b=4
Бинарные арифметические операции могут быть объединены с операцией присваивания:
- объект *= выражение; // объект = объект * выражение
- объект /= выражение; // объект = объект / выражение
- объект += выражение; // объект = объект + выражение
- объект -= выражение; // объект = объект — выражение
- объект %= выражение; // объект = объект % выражение
Логические операции
Логические операции делятся на две группы:
- условные;
- побитовые.
Условные логические операции чаще всего используются в операциях проверки условия if и могут выполняться над любыми объектами. Результат условной логической операции:
- 1 если выражение истинно;
- 0 если выражение ложно.
Вообще, все значения, отличные от нуля, интерпретируются условными логическими операциями как истинные.
Основные условные логические операции:
- && — И (бинарная) — требуется одновременное выполнение всех операций отношения;
- || — ИЛИ (бинарная) — требуется выполнение хотя бы одной операции отношения;
- ! — НЕ (унарная) — требуется невыполнение операции отношения.
Побитовые логические операции оперируют с битами, каждый из которых может принимать только два значения: 0 или 1.
Основные побитовые логические операции в языке Си:
- & конъюнкция (логическое И) — бинарная операция, результат которой равен 1 только когда оба операнда единичны (в общем случае — когда все операнды единичны);
- | дизъюнкция (логическое ИЛИ) — бинарная операция, результат которой равен 1 когда хотя бы один из операндов равен 1;
- ~ инверсия (логическое НЕ) — унарная операция, результат которой равен 0 если операнд единичный, и равен 1, если операнд нулевой;
- ^ исключающее ИЛИ — бинарная операция, результат которой равен 1, если только один из двух операндов равен 1 (в общем случае если во входном наборе операндов нечетное число единиц).
Для каждого бита результат выполнения операции будет получен в соответствии с таблицей.
| a | b | a & b | a | b | ~a | a ^ b |
| 0 | 0 | 0 | 0 | 1 | 0 |
| 0 | 1 | 0 | 1 | 1 | 1 |
| 1 | 0 | 0 | 1 | 0 | 1 |
| 1 | 1 | 1 | 1 | 0 | 0 |
2
3
4
5
6
7
unsigned char b = 9; // b = 0000 1001
unsigned char c, d, e, f;
c = a & b; // c = 8 = 0000 1000
d = a | b; // d = 15 = 0000 1111
e = ~a; // e = 241 = 1111 0001
f = a ^ b; // f = 7 = 0000 0111
Побитовые операции позволяют осуществлять установку и сброс отдельных битов числа. С этой целью используется маскирование битов. Маски, соответствующие установке каждого бита в байте, представлены в таблице
| Бит | Маска |
| 0 | 0x01 |
| 1 | 0x02 |
| 2 | 0x04 |
| 3 | 0x08 |
| 4 | 0x10 |
| 5 | 0x20 |
| 6 | 0x40 |
| 7 | 0x80 |
2
a = a | 0x04; // a = 7, бит 2 установлен
2
a = a & (~0x02); // a = 1, бит 1 сброшен
Бинарные побитовые логические операции могут быть объединены с операцией присваивания:
- объект &= выражение; // объект = объект & выражение
- объект |= выражение; // объект = объект | выражение
- объект ^= выражение; // объект = объект ^ выражение
Сдвиговые операции
Операции арифметического сдвига применяются в целочисленной арифметике и обозначаются как:
- >> — сдвиг вправо;
- << — сдвиг влево.
Общий синтаксис осуществления операции сдвига:
объект = выражение сдвиг КоличествоРазрядов;
Пример:
2
3
unsigned char a=6; // a = 0000 0110
unsigned char b;
b = a >> 1; // b = 0000 0110 >> 1 = 0000 0011 = 3Арифметический сдвиг целого числа вправо >> на 1 разряд соответствует делению числа на 2.
Арифметический сдвиг целого числа влево << на 1 разряд соответствует умножению числа на 2.

Здравствуйте можете объяснить пожалуйста по подробнее про строку:
a = (i >> 3) & 0x01;
просто данной строкой и ещё несколькими у меня реализуется в программе таблица истинности для логического выражения но я не совсем понял как это работает.
Могу для удобства скинуть весь код он не большой.
Значение i сдвигается на 3 разряда справа, после чего маской 0x01 гасятся все биты кроме последнего. В младшем бите a получаем то значение, которое было в бите 3 переменной i.
Можете ещё объяснить
2
3
4
b = i & 0x04;
c = i & 0x02;
d = i & 0x01;
Анализируем четыре младших бита переменной i по отдельности
А какой тип данных должел быть, чтобы применить логические операции?
У меня так получилось:
Если я пишу
char ddd = 1;
.
.
ddd = ~ ddd;
if(ddd==0)…..
то ИФ не реагирует на этот ноль, видно стало ddd = 11111110, а это не ноль.
А когда написал :
char ddd = 0;
.
.
ddd = ~ ddd;
ddd = ~ ddd;
if(ddd)….
ИФ отреагировал. Что изменилось?
Так и есть. Тильда означает побитовую инверсию. Используйте восклицательный знак.
Здравствуйте
2
3
for(x; x <= 2.0; x += 0.1)
printf("%lf\n", x);
Почему здесь не выводится 2.0, если по условию оно подходит?
Это из-за разрядной сетки чисел типа float.
https://prog-cpp.ru/c-data-types/
Попробуйте представить свой шаг 0,1 в разрядной сетке, а затем добавлять его к x.
И не забудьте, что число типа float не может быть точно равно нулю — разрядная сетка не позволит.
Добрый день!
Кратко: Си умножение 1 000 000 на 1 000 001,
подскажите пожалуйста, как вывести результат ?
2
3
4
5
6
7
a = 1 000 000;
b = 1 000 001;
a = a * b;
printf("%d", a);
return(0);
Можно попробовать использовать данные типа long вместо int. Если это не поможет, придется прибегнуть к длинной арифметике: https://prog-cpp.ru/long-mult/
Добрый день! А как понять запись, когда одно число 5 к примеру, а второе 8. И дальше идёт следующая запись: 5<<8?
Эта запись означает, что число 5 сдвигается на 8 разрядов влево.
Сдвиг влево на 1 разряд соответствует умножению числа на 2.
Сдвиг влево на 8 разрядов соответствует умножению числа на 28=256.
В итоге получим 5*256 = 1280.
Очень понятно написано, спасибо. А нельзя ли расписать побитовые операции с присваиванием. Как раз за этим полез в поиск.
a = a | 0x04;
то же самое, что
a |= 0x04;
Ничем не отличается от арифметических операций с присваиванием
а что делает операция -> ?
Обращается к полю структуры, объявленной через указатель
не понял первый пример, где там деление?
2
int b = (int)a % 2; // b = 1
Тут нет деления. Только операция получения остатка от деления — %.
Благодарю за понятную статью! Искал материал на тему операций в языке С так вот у вас написано как раз так что любой чайник поймет. И теперь можно смело углубляться в тему. Правда не помешало бы в конце статьи писать конкретные примеры использования содержимого статьи в реальных задачах а то не совсем понятно применение конкретно побитовых логических операций а так все хорошо!