Си — инструмент, острый, как бритва: с его помощью можно создать и элегантную программу, и кровавое месивоC (Си) — язык программирования разработанный в 1969-1973 годах в Bell Labs Кеном Томпсоном и Деннисом Ритчи. Создан как развитие языка B (Би) и был предназначен для ОС UNIX. С тех пор считается самым каноничным языком программирования.
Брайан У. Керниган
Язык Си — компилируемый. Это значит что программа-компилятор переводит Си-программу в бинарный код, который затем выполняется напрямую процессором. Это дает прирост в скорости выполнения программного кода по сравнению с интерпретацией.
Программа
int main(){return 0;} //самая простая программа на Си, которая возвращает 0 операционной системе после своего запуска.Важно заметить, что все строки между фигурными скобками (которые ограничивают блоки операторов) должны заканчиваться точкой с запятой.
Особенность языка, его тогдашняя инновационность, заключалась в том, что код разделялся на функции. И это разделение удобно (в отличие от языка PL/M). И все функции программы будут вызываться из главной функции main, которая в программе должна быть обязательно.
Создать функцию просто: тип имя_функции(аргументы}{тело функции}. Появление вызова функции в программе должно идти после её описания (когда указывается тело функции), или же указать в начале файла все функции, которые у нас будут в программе.
Функция может возвращать значение определенного типа (оператором return) или же ничего не возвращать (тогда на месте типа пишут void и такая функция становится процедурой).
4 типа целых чисел:
Типы автоматически не преобразуются один в другой, т. к. это потребовало бы дополнительных вливаний вычислений в код.
Конечно же можно создать массив однотипных данных. В том числе и неопределенного размера. И в отличие от BASIC нумерация элементов массива начинается с 0. Элементы локально-объявленного массива инициализировать нельзя, а глобального — можно.
Работа с переменной начинается с её объявления: int a. Объявление может происходить одновременно с инициализацией: int a=0;
Замечание: размеры в байтах указаны для архитектуры x86 и являются архитектуро-зависимыми.
+= -= *= /= %= <<= >>= &= ^= |= хитрые присваивания. Их задача уменьшить количество действий с памятью.
Нельзя не сказать и о операторах ++ и --.
while(уловие) оператор1. Цикл выполняет оператор1 до тех пор, пока условие не ноль. for(оператор1; условие; оператор2). Сначала выполнится оператор1, затем будет выполняться оператор2 до тех пор, пока условие не станет ложным.
break используется для предварительного выхода из цикла.
Строго говоря, на этом встроенные функции языка Си исчерпываются. Да, функции ввода-вывода не являются стандартными.
printf(строка, список переменных). Функция форматированного вывода. Количество аргументов для функции переменно и зависит от того, сколько переменных вы хотите вывести. Вывод строки не вызовет никаких проблем: printf("Hello world!"). Чтобы вывести строку с переносом строки нужно написать на нужной позиции \n (new line), например printf("Kill\nall\nhumans\n"). А чтобы вывести значение переменной нужно приложить немного усилий: printf("%d", a). %d означает, что переменная, подготовленная к выводу, целочисленная. %s — строка, %d — целое число.
scanf(строка, переменные). Функция форматированного ввода, аналогична printf. В конец строки автоматически добавляется нулевой символ (\0).
Например, нам в функции нам нужно изменить значение переданной переменной.
Это происходит и-за того, что в функцию передается копия значения переменной,а не она сама. Чтобы избежать этого, модифицируем программу таким образом:
Механизм указателей позволяет быстро и энергоэффективно проворачивать хитрые операции, но это тема отдельной статьи.
Другой важной директивой является #define строка значение. Компилятор ищет строку в файле и заменяет её на значение. Это позволяет легко выполнять требование отсутствия в коде «магических констант»: #define BIG_NUM 256.
Особенность языка, его тогдашняя инновационность, заключалась в том, что код разделялся на функции. И это разделение удобно (в отличие от языка PL/M). И все функции программы будут вызываться из главной функции main, которая в программе должна быть обязательно.
Создать функцию просто: тип имя_функции(аргументы}{тело функции}. Появление вызова функции в программе должно идти после её описания (когда указывается тело функции), или же указать в начале файла все функции, которые у нас будут в программе.
Функция может возвращать значение определенного типа (оператором return) или же ничего не возвращать (тогда на месте типа пишут void и такая функция становится процедурой).
Переменные
Типы данных в Си приближены к процессору.4 типа целых чисел:
- signed short int — короткое целое со знаком (2 байта)
- unsigned short int — короткое целое без знака
- signed long int или просто int — длинное целое со знаком (4 байта)
- unsigned long int — длинное целое без знака
- float — число с плавающей точкой (4 байта),
- double — число с двойной точностью (8 байт).
- enum — перечисления,
- struct — структуры,
- union — объединение.
Типы автоматически не преобразуются один в другой, т. к. это потребовало бы дополнительных вливаний вычислений в код.
Конечно же можно создать массив однотипных данных. В том числе и неопределенного размера. И в отличие от BASIC нумерация элементов массива начинается с 0. Элементы локально-объявленного массива инициализировать нельзя, а глобального — можно.
Работа с переменной начинается с её объявления: int a. Объявление может происходить одновременно с инициализацией: int a=0;
Замечание: размеры в байтах указаны для архитектуры x86 и являются архитектуро-зависимыми.
Константы
\n — новая строка, \t — табуляция, \b — шаг назад, \r — возврат каретки, \\ - косая черта, \' - апостроф, \" - кавычки.Операторы присваивания
= простое присваивание.+= -= *= /= %= <<= >>= &= ^= |= хитрые присваивания. Их задача уменьшить количество действий с памятью.
Нельзя не сказать и о операторах ++ и --.
Арифметические операторы
- + - сложение
- - - вычитание
- * - умножение
- / - деление нацело
- % - остаток от деления нацело
- ~ побитовое НЕ
- & побитовое И
- | побитовое ИЛИ
- ^ побитовое исключающее ИЛИ
Операторы отношений
- == равенство
- != неравенство
- < > <= >= различные операторы отношения
Логичесике операторы
- ! логическое НЕ
- && логическое И
- || логическое ИЛИ
Функции
if(условие) оператор1 else оператор2. Если условие истинно (не ноль), то выполняется оператор1, а если нет (условие 0) — то оператор2. Это позволяет проще записывать сравнения с нулем, например if(x!=0) <=> if(x) и if(x==0) <=> if(!x).while(уловие) оператор1. Цикл выполняет оператор1 до тех пор, пока условие не ноль. for(оператор1; условие; оператор2). Сначала выполнится оператор1, затем будет выполняться оператор2 до тех пор, пока условие не станет ложным.
break используется для предварительного выхода из цикла.
Строго говоря, на этом встроенные функции языка Си исчерпываются. Да, функции ввода-вывода не являются стандартными.
printf(строка, список переменных). Функция форматированного вывода. Количество аргументов для функции переменно и зависит от того, сколько переменных вы хотите вывести. Вывод строки не вызовет никаких проблем: printf("Hello world!"). Чтобы вывести строку с переносом строки нужно написать на нужной позиции \n (new line), например printf("Kill\nall\nhumans\n"). А чтобы вывести значение переменной нужно приложить немного усилий: printf("%d", a). %d означает, что переменная, подготовленная к выводу, целочисленная. %s — строка, %d — целое число.
scanf(строка, переменные). Функция форматированного ввода, аналогична printf. В конец строки автоматически добавляется нулевой символ (\0).
Указатели
Механизм, который вызывает трепет уважения у понявших, ненависть у не понявших и пренебрежении у тех, кто до этого писал на ассемблере.Например, нам в функции нам нужно изменить значение переданной переменной.
void set(int x){x=14;}
int main(){
int a=88;
set(a);
printf("%d\n", a); //выведется 88
return 0;
}
Это происходит и-за того, что в функцию передается копия значения переменной,а не она сама. Чтобы избежать этого, модифицируем программу таким образом:
void set(int *x){ //функция получает адрес переменной типа int
*x=14; //и записывает по этому адресу 14
}
int main(){
int a=88; set(&a); //передаем адрес на переменную
printf("%d\n", a); //выведется 14
return 0;
}
Механизм указателей позволяет быстро и энергоэффективно проворачивать хитрые операции, но это тема отдельной статьи.
Библиотеки
Библиотеки добавляют множество функций в систему языка. Причислим самые часто используемые:- math.h — математическая библиотека содержит модуль, тригонометрические функции, арк-функции, гиперболический функции, логарифмы, возведение в степень и некоторыми другими.
- signals.h — библиотека работы с сигналами в ОС Unix.
- stdio.h — библиотека содержащая функции для функций стандартного ввода-вывода. Содержит функции работы с файлами (fopen, fclose, remove, rename), fgetc (получить символ), fgets (получить строку), fputc (записать символ), fputs (записать строку), getchar (получает символ со стандартного устройства ввода), gets (получает строку со стандартного устройства ввода), fprintf, fcanf (уже были обсуждены) и проч.
- stdlib.h — стандартная библиотека с функциями:
- преобразования типов (XtoY, например atoi — строка в целое число),
- генерации псевдослучайных чисел (rand, srand),
- выделения памяти (Xalloc, например malloc),
- контроль выполнения (exit — завершить программу, abort — экстренно завершить программу),
- сортировка (qsort) и проч.
- string.h — библиотека работы со строками содержит функции копирования, сравнения, конкатенации (дописывания, сложения строк), поиска символов, лексикографического сравнения, определения длины строки и проч.
Директивы компилятора
Директива компилятора начинается с символа # и означает некоторое действие, которое компилятор обязан выполнить, например #include имя_файла означает включить содержимое указанного файла в текущий. Таким образом функции, объявленные в этом, подключенном файле, становятся «видны» в нашей программе.Другой важной директивой является #define строка значение. Компилятор ищет строку в файле и заменяет её на значение. Это позволяет легко выполнять требование отсутствия в коде «магических констант»: #define BIG_NUM 256.
Компиляторы
- GNU Compiler Collection включает компилятор Си. Знает все диалекты и может компилировать код под огромное количество платформ.
- MinGW — Минималистичный GNU для Windows. Содержит библиотеки WinAPI.
- MS C — старый компилятор от MS под DOS.
- TCC — маленький компилятор Си с возможностью интерпретации (!) Си-программ.
Диалекты
- K&RC (Си Крнигана и Ритчи). Опубликован вместе с книгой «Язык программирования Си» в 1978.
- ISO C или ANSI C. Стандарт языка разработанный ANSI в 1989. Например, теперь функция main должна возвращать хоть какое-то значение (не быть void).
- СC99. (легко догадаться до года - 1999) Некоторые добавления: возможность объявлять локальные переменные в любом месте (как в C++): for(int i=0; …) вместо int i; for(i=0; …), однострочные комментарии (как в C++).
- C++11. (2011 год, спасибо Кэп!) Введены современные мэйнстримовые фичи: многопоточность, юникод, и проч.
Комментариев нет:
Отправить комментарий