Спецификаторы формата в форматной строке, такие как %c
и %s
, определяют способ использования аргументов. Как показано выше, это может привести к ужасным последствиям. В языке C++ неопределенных аргументов лучше избегать.
Код на языке С++ часто используется наряду с кодом на языке С в одной и той же программе; иначе говоря, одни части бывают написаны на языке С++ (и скомпилированы с помощью компилятора языка С++), а другие — на языке С (и скомпилированы с помощью компилятора языка С). Для того чтобы воспользоваться этой возможностью, язык С++ предлагает программистам
extern "C" void callable_from_C(int);
В качестве альтернативы ее можно применить ко всем объявлениям в блоке.
extern "C" {
void callable_from_C(int);
int and_this_one_also(double, int*);
/* ... */
}
Детали можно найти в разделе 27.2.3.
В языке С нет возможности перегружать функции, поэтому можете поместить спецификацию связи с языком С только в одной версии перегруженной функции.
Есть два способа определить новый (пользовательский) тип: в виде класса (class
, struct
и union
; см. раздел A.12) и в виде перечисления (enum
; см. раздел A.11).
Программист может определить смысл большинства операторов, принимающих операнды пользовательского типа. Изменить стандартный смысл операторов для встроенных типов или ввести новый оператор невозможно. Имя оператора, определенного пользователем (перегруженного оператора), состоит из символа оператора, которому предшествует ключевое слово 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
Функции, определяющие следующие операторы, должны быть членами класса:
= [ ] ( ) –>
Все остальные операторы можно определить и как члены-функции, и как самостоятельные функции.
Обратите внимание на то, что каждый пользовательский тип имеет оператор =
(присваивание и инициализация), &
(взятие адреса) и ,
(запятая), определенные по умолчанию.
При перегрузке операторов следует проявлять умеренность и придерживаться общепринятых соглашений.
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.
class X {
public:
// пользовательский интерфейс
private:
// реализация
};
Переменные, функции и типы, определенные в объявлении класса, называются