// может быть, что-то еще
int x = 'x';
printf("hello, world!");
printf("print a char '%c'\n",x); // печатает целое число x как
// символ
printf("print a string \"%s\"",x); // "выстрел себе в ногу"
Спецификаторы формата в форматной строке, такие как
%c
и
%s
, определяют способ использования аргументов. Как показано выше, это может привести к ужасным последствиям. В языке C++ неопределенных аргументов лучше избегать.
A.9.4. Спецификации связей
Код на языке С++ часто используется наряду с кодом на языке С в одной и той же программе; иначе говоря, одни части бывают написаны на языке С++ (и скомпилированы с помощью компилятора языка С++), а другие — на языке С (и скомпилированы с помощью компилятора языка С). Для того чтобы воспользоваться этой возможностью, язык С++ предлагает программистам спецификации связей (linkage specifications), указывающие, что та или иная функция может быть вызвана из модуля, написанного на языке С. Спецификацию связи с языком С можно поместить перед объявлением функции.
extern "C" void callable_from_C(int);
В качестве альтернативы ее можно применить ко всем объявлениям в блоке.
extern "C" {
void callable_from_C(int);
int and_this_one_also(double, int*);
/* ... */
}
Детали можно найти в разделе 27.2.3.
В языке С нет возможности перегружать функции, поэтому можете поместить спецификацию связи с языком С только в одной версии перегруженной функции.
A.10. Типы, определенные пользователем
Есть два способа определить новый (пользовательский) тип: в виде класса (
class
,
struct
и
union
; см. раздел A.12) и в виде перечисления (
enum
; см. раздел A.11).
A.10.1. Перегрузка операций
Программист может определить смысл большинства операторов, принимающих операнды пользовательского типа. Изменить стандартный смысл операторов для встроенных типов или ввести новый оператор невозможно. Имя оператора, определенного пользователем (перегруженного оператора), состоит из символа оператора, которому предшествует ключевое слово
operator
; например, имя функции, определяющей оператор
+
, выглядит как
operator +
.
Matrix operator+(const Matrix&, const Matrix&);
Примеры можно найти в определениях классов
std::ostream
(см. главы 10-11),
std::vector
(см. главы 17–19 и раздел Б.4),
std::complex
(см. раздел Б.9.3) и
Matrix
(см. главу 24).
Перегрузить можно все операторы за исключением следующих:
?: . .* :: sizeof typeid
Функции, определяющие следующие операторы, должны быть членами класса:
= [ ] ( ) –>
Все остальные операторы можно определить и как члены-функции, и как самостоятельные функции.
Обратите внимание на то, что каждый пользовательский тип имеет оператор
=
(присваивание и инициализация),
&
(взятие адреса) и
,
(запятая), определенные по умолчанию.
При перегрузке операторов следует проявлять умеренность и придерживаться общепринятых соглашений.
A.11. Перечисления
Перечисление (enumeration) определяет тип, содержащий набор именованных значения (перечислителей).
enum Color { green, yellow, red };
По умолчанию первый перечислитель равен нулю
0
, так что
green==0
, а остальные значения увеличиваются на единицу, так что
yellow==1
и
red==2
. Кроме того, можно явно определить значение перечислителя.
enum Day { Monday=1,Tuesday,Wednesday };
Итак,
Monday==1
,
Tuesday==2
и
Wednesday==3
.
Отметим, что перечислители принадлежат не области видимости своего перечисления, а охватывающей области видимости.
int x = green; // OK
int y = Color::green; // ошибка
Перечислители и значения перечислений неявно преобразовываются в целые числа, но целые числа не преобразовываются в типы перечислений неявно.
int x = green; // OK: неявное преобразование Color в int
Color c = green; // OK
c = 2; // ошибка: нет неявного преобразования
// int в Color
c = Color(2); // OK: (непроверяемое) явное преобразование
int y = c; // OK: неявное преобразование Color в int
Использование перечислений обсуждается в разделе 9.5.
A.12. Классы
Класс (class) — это тип, для которого пользователь определил представление его объектов и операции, допустимые для этих объектов.
class X {
public:
// пользовательский интерфейс
private:
// реализация
};
Переменные, функции и типы, определенные в объявлении класса, называются членами этого класса. Технические детали изложены в главе 9.