Участник:Joparino/Совместимость C и C++: различия между версиями

Материал из Википедии — свободной энциклопедии
Перейти к навигации Перейти к поиску
Содержимое удалено Содержимое добавлено
Нет описания правки
Нет описания правки
Строка 1: Строка 1:
[[Язык программирования|Языки программирования]] [[Си (язык программирования)|C]] и [[C++]] тесно связаны, но имеют существенные различия. C++ создавался как потомок [[ANSI C|достандартизированного]] C и был разработан так, чтобы быть в основном совместимым с исходным кодом C того времени.<ref name="overview">{{cite web | url=http://www.stroustrup.com/crc.pdf | title=An Overview of the C++ Programming Language in The Handbook of Object Technology (Editor: Saba Zamir). CRC Press LLC, Boca Raton. 1999. ISBN 0-8493-3135-8. | first=Bjarne | last=Stroustrup | author-link=Bjarne Stroustrup | page=4 | format=PDF | access-date=12 August 2009 | url-status=live | archive-url=https://web.archive.org/web/20120816122304/http://www.stroustrup.com/crc.pdf | archive-date=16 August 2012 | df=dmy-all }}</ref><ref>{{cite web |url=http://www.stroustrup.com/siblings_short.pdf |title=C and C++: Siblings. The C/C++ Users Journal. July 2002.|author=B.Stroustrup |access-date=17 March 2019}}</ref> В связи с этим, средства разработки для двух языков (такие, как [[Интегрированная среда разработки|интегрированные среды разработки]] и [[Компилятор|компиляторы]]) часто интегрировались в один продукт, при этом программист может указать C или C++ в качестве исходного языка.
[[Язык программирования|Языки программирования]] [[Си (язык программирования)|C]] и [[C++]] тесно связаны, но имеют существенные различия. C++ создавался как потомок [[ANSI C|достандартизированного]] C и был разработан так, чтобы быть в основном совместимым с исходным кодом C того времени.<ref name="overview">{{cite web | url=http://www.stroustrup.com/crc.pdf | title=An Overview of the C++ Programming Language in The Handbook of Object Technology (Editor: Saba Zamir). CRC Press LLC, Boca Raton. 1999. ISBN 0-8493-3135-8. | first=Bjarne | last=Stroustrup | author-link=Bjarne Stroustrup | page=4 | format=PDF | access-date=12 August 2009 | url-status=live | archive-url=https://web.archive.org/web/20120816122304/http://www.stroustrup.com/crc.pdf | archive-date=16 August 2012 | df=dmy-all }}</ref><ref>{{cite web |url=http://www.stroustrup.com/siblings_short.pdf |title=C and C++: Siblings. The C/C++ Users Journal. July 2002.|author=B.Stroustrup |access-date=17 March 2019}}</ref> В связи с этим, средства разработки для двух языков (такие, как [[Интегрированная среда разработки|интегрированные среды разработки]] и [[Компилятор|компиляторы]]) часто интегрировались в один продукт, при этом программист может указать C или C++ в качестве исходного языка.


Однако, C ''не является'' [[Подмножество|подмножеством]] C++,<ref name="subset">{{cite web |url=http://www.stroustrup.com/bs_faq.html#C-is-subset |title=Bjarne Stroustrup's FAQ&nbsp;– Is C a subset of C++? |access-date=22 Sep 2019}}</ref>, поэтому нетривиальные программы на C не будут компилироваться на C++ без изменений. Также, C++ вводит множество возможностей, недоступных в C и на практике почти весь код, написанный на C++, не соответствует коду на C. Однако в этой статье основное внимание уделяется различиям, которые приводят к тому, что соответствующий код C является неправильно ({{lang-en|ill-formed}}) написанным кодом C++ или соответствующим / хорошо ({{lang-en|conforming/ill-formed}}) написанном на обоих языках, но вести себя по-разному на C и C++.
Однако, C ''не является'' [[Подмножество|подмножеством]] C++,<ref name="subset">{{cite web |url=http://www.stroustrup.com/bs_faq.html#C-is-subset |title=Bjarne Stroustrup's FAQ&nbsp;– Is C a subset of C++? |access-date=22 Sep 2019}}</ref>, поэтому нетривиальные программы на C не будут компилироваться на C++ без изменений. Также, C++ вводит множество возможностей, недоступных в C и на практике почти весь код, написанный на C++, не соответствует коду на C. Однако в этой статье основное внимание уделяется различиям, которые приводят к тому, что соответствующий код C является неправильно написанным ({{lang-en|ill-formed}}) кодом C++ или соответствующим/хорошо написанном ({{lang-en|conforming/ill-formed}}) на обоих языках, но вести себя по-разному на C и C++.


[[Страуструп, Бьёрн|Бьёрн Страуструп]], создатель C++, предложил<ref>{{cite web |url=http://www.stroustrup.com/compat_short.pdf |title=C and C++: A Case for Compatibility. The C/C++ Users Journal. August 2002.|author=B. Stroustrup |access-date=18 August 2013 |url-status=live |archive-url=https://web.archive.org/web/20120722012742/http://www2.research.att.com/~bs/compat_short.pdf |archive-date=22 July 2012 |df=dmy-all }}</ref> что несовместимость между C и C++ должна быть уменьшена насколько это возможно, чтобы обеспечить максимальную совместимость между двумя языками. Другие утверждают, что, поскольку C и C++ — это два разных языка, совместимость между ними полезна, но не жизненно важна; согласно им, усилия по уменьшению несовместимости не должны препятствовать попыткам улучшить каждый язык в отдельности. Официальное обоснование стандарта 1999 C ([[C99]]) "одобрить принцип сохранения наибольшего общего подмножества" между C и C++ "сохраняя при этом различие между ними и позволяя им развиваться отдельно", и заявил, что авторы были "довольны тем, что C++ стал большим и амбициозным языком."<ref>[http://www.open-std.org/jtc1/sc22/wg14/www/C99RationaleV5.10.pdf Rationale for International Standard—Programming Languages—C] {{webarchive|url=https://web.archive.org/web/20160606072228/http://www.open-std.org/jtc1/sc22/wg14/www/C99RationaleV5.10.pdf |date=6 June 2016 }}, revision 5.10 (April 2003).</ref>
[[Страуструп, Бьёрн|Бьёрн Страуструп]], создатель C++, предложил<ref>{{cite web |url=http://www.stroustrup.com/compat_short.pdf |title=C and C++: A Case for Compatibility. The C/C++ Users Journal. August 2002.|author=B. Stroustrup |access-date=18 August 2013 |url-status=live |archive-url=https://web.archive.org/web/20120722012742/http://www2.research.att.com/~bs/compat_short.pdf |archive-date=22 July 2012 |df=dmy-all }}</ref> что несовместимость между C и C++ должна быть уменьшена насколько это возможно, чтобы обеспечить максимальную совместимость между двумя языками. Другие утверждают, что, поскольку C и C++ — это два разных языка, совместимость между ними полезна, но не жизненно важна; согласно им, усилия по уменьшению несовместимости не должны препятствовать попыткам улучшить каждый язык в отдельности. Официальное обоснование стандарта 1999 C ([[C99]]) "одобрить принцип сохранения наибольшего общего подмножества" между C и C++ "сохраняя при этом различие между ними и позволяя им развиваться отдельно", и заявил, что авторы были "довольны тем, что C++ стал большим и амбициозным языком."<ref>[http://www.open-std.org/jtc1/sc22/wg14/www/C99RationaleV5.10.pdf Rationale for International Standard—Programming Languages—C] {{webarchive|url=https://web.archive.org/web/20160606072228/http://www.open-std.org/jtc1/sc22/wg14/www/C99RationaleV5.10.pdf |date=6 June 2016 }}, revision 5.10 (April 2003).</ref>
Строка 30: Строка 30:
</syntaxhighlight>
</syntaxhighlight>


|2= C++ имеет более сложные правила присваивания указателей, которые добавляют квалификаторы, поскольку он позволяет приводить <code>int **</code>
|2= C++ has more complicated rules about pointer assignments that add qualifiers as it allows the assignment of <code>int **</code>
to <code>const int *const *</code> but not the unsafe assignment to <code>const int **</code> while C allows neither of those (although compilers will usually only emit a warning).
к <code>const int *const *</code>, но не небезопасное присваивание <code>const int **</code>, в то время как C не допускает ни того, ни другого (хотя компиляторы обычно выдают только предупреждение).


|3= C++ changes some [[C standard library]] functions to add additional overloaded functions with <code>[[Const (computer programming)|const]]</code> [[type qualifier]]s, e.g. <code>strchr</code> returns <code>char*</code> in C, while C++ acts as if there were two overloaded functions <code>const char *strchr(const char *)</code> and a <code>char *strchr(char *)</code>.
|3= C++ изменяет некоторые функции [[Стандартная библиотека языка Си|стандартной библиотеки языка Си]], добавляя дополнительные перегруженные функции с [[Квалификаторы типа|квалификатором типа]] <code>[[Const (программирование)|const]]</code>, например <code>strchr</code> возвращает <code>char*</code> в C, в то время как C++ поступает так, как если бы были две перегруженные функции <code>const char *strchr(const char *)</code> и <code>char *strchr(char *)</code>.


|4= C++ также более строг в преобразованиях в перечисления: целые числа не могут быть неявно преобразованы в перечисления, как в C. Кроме того, [[Перечисляемый тип#C++
|4= C++ is also more strict in conversions to enums: ints cannot be implicitly converted to enums as in C. Also, [[enumerated type#C and syntactically similar languages|enumeration constants]] (<code>enum</code> enumerators) are always of type <code>int</code> in C, whereas they are distinct types in C++ and may have a size different from that of <code>int</code>.
|константные перечисления]] (<code>enum</code> enumerators) всегда имеют тип <code>int</code> в C, тогда как в C++ они являются различными типами и могут иметь размер, отличный от размера <code>int</code>.


|5= In C++ a <code>const</code> variable must be initialized; in C this is not necessary.
|5= В C++ переменная <code>const</code> должна быть инициализирована; в C это необязательно.


|6= Компиляторы C++ запрещают goto или switch пересекать инициализацию, как в следующем коде C99:
|6= C++ compilers prohibit goto or switch from crossing an initialization, as in the following C99 code:
<syntaxhighlight lang="c">
<syntaxhighlight lang="c">
void fn(void)
void fn(void)
Строка 50: Строка 51:
</syntaxhighlight>
</syntaxhighlight>


|7= While syntactically valid, a [[longjmp|<code>longjmp()</code>]] results in undefined behaviour in C++ if the jumped-over [[stack frame]]s include objects with nontrivial destructors.<ref>{{cite web|url=http://www.cplusplus.com/reference/csetjmp/longjmp/|title=longjmp - C++ Reference|website=www.cplusplus.com|url-status=live|archive-url=https://web.archive.org/web/20180519163512/http://www.cplusplus.com/reference/csetjmp/longjmp/|archive-date=19 May 2018|df=dmy-all}}</ref> The C++ implementation is free to define the behaviour such that destructors would be called. However, this would preclude some uses of <code>longjmp()</code> which would otherwise be valid, such as implementation of [[Thread (computing)|threads]] or [[coroutines]] switching between separate call stacks with <code>longjmp()</code> — when jumping from the lower to the upper call stack in global address space, destructors would be called for every object in the lower call stack. No such issue exists in C.
|7= Несмотря на синтаксическую корректность, функция [[longjmp|<code>longjmp()</code>]] приводит к неопределенному поведению в C++, если [[Стек вызовов|фреймы стека]] с перепрыгиванием включают объекты с нетривиальными деструкторами.<ref>{{cite web|url=http://www.cplusplus.com/reference/csetjmp/longjmp/|title=longjmp - C++ Reference|website=www.cplusplus.com|url-status=live|archive-url=https://web.archive.org/web/20180519163512/http://www.cplusplus.com/reference/csetjmp/longjmp/|archive-date=19 May 2018|df=dmy-all}}</ref> Реализация C++ может свободно определять поведение таким образом, чтобы вызывались деструкторы. Однако это мешает некоторым видам использования <code>longjmp()</code> , которые в противном случае были бы допустимы, такие как [[Поток выполнения|потоков]] или [[Сопрограмма|сопрограмм]] переключающихся между отдельными стеками вызовов с помощью <code>longjmp()</code> — при переходе из нижнего стека вызовов в верхний в глобальном адресном пространстве деструкторы будут вызываться для каждого объекта в нижнем стеке вызовов. В C такой проблемы не существует.


C допускает несколько предварительных определений одной глобальной переменной в одной единице перевода, что недопустимо как нарушение ODR в C+
|8= C allows for multiple tentative definitions of a single global variable in a single [[Translation unit (programming)|translation unit]], which is invalid as an [[One Definition Rule|ODR]] violation in C++.

|8= C допускает несколько предварительных определений одной глобальной переменной в одной [[Единица трансляции|единице трансляции]], что недопустимо в C++, так как нарушение [[Правило одного определения|правила одного определения]] ({{lang-en|One Definition Rule, ODR}}).
<syntaxhighlight lang="c">
<syntaxhighlight lang="c">
int N;
int N;
Строка 58: Строка 61:
</syntaxhighlight>
</syntaxhighlight>


|9= In C, declaring a new type with the same name as an existing <code>struct</code>, <code>union</code> or <code>enum</code> is valid, but it is invalid in C++, because in C, <code>struct</code>, <code>union</code>, and <code>enum</code> types must be indicated as such whenever the type is referenced whereas in C++, all declarations of such types carry the [[typedef]] implicitly.
|9= В C допустимо объявление нового типа с тем же именем, что и у <code>struct</code>, <code>union</code> или <code>enum</code>, но это недопустимо в C++, потому что в C типы, <code>struct</code>, <code>union</code>, и <code>enum</code> должны указываться всякий раз, когда на тип ссылаются, тогда как в C++ все объявления таких типов неявно содержат [[typedef]].
<syntaxhighlight lang="c">
<syntaxhighlight lang="c">
enum BOOL {FALSE, TRUE};
enum BOOL {FALSE, TRUE};

Версия от 16:22, 25 июля 2022

Языки программирования C и C++ тесно связаны, но имеют существенные различия. C++ создавался как потомок достандартизированного C и был разработан так, чтобы быть в основном совместимым с исходным кодом C того времени.[1][2] В связи с этим, средства разработки для двух языков (такие, как интегрированные среды разработки и компиляторы) часто интегрировались в один продукт, при этом программист может указать C или C++ в качестве исходного языка.

Однако, C не является подмножеством C++,[3], поэтому нетривиальные программы на C не будут компилироваться на C++ без изменений. Также, C++ вводит множество возможностей, недоступных в C и на практике почти весь код, написанный на C++, не соответствует коду на C. Однако в этой статье основное внимание уделяется различиям, которые приводят к тому, что соответствующий код C является неправильно написанным (англ. ill-formed) кодом C++ или соответствующим/хорошо написанном (англ. conforming/ill-formed) на обоих языках, но вести себя по-разному на C и C++.

Бьёрн Страуструп, создатель C++, предложил[4] что несовместимость между C и C++ должна быть уменьшена насколько это возможно, чтобы обеспечить максимальную совместимость между двумя языками. Другие утверждают, что, поскольку C и C++ — это два разных языка, совместимость между ними полезна, но не жизненно важна; согласно им, усилия по уменьшению несовместимости не должны препятствовать попыткам улучшить каждый язык в отдельности. Официальное обоснование стандарта 1999 C (C99) "одобрить принцип сохранения наибольшего общего подмножества" между C и C++ "сохраняя при этом различие между ними и позволяя им развиваться отдельно", и заявил, что авторы были "довольны тем, что C++ стал большим и амбициозным языком."[5]

Некоторые нововведения C99 не поддерживаются в текущем стандарте C++ или конфликтуют с особенностями C++, например массивы переменной длины, собственные комплексные типы данных и квалификатор типа restrict. С другой стороны, C99 уменьшил некоторые другие несовместимости по сравнению с C89, включив такие функции C++, как // комментарии и смешанные объявление и код.[6]

Конструкции, допустимые в C, но не в C++

C++ применяет более строгие правила типизации (никаких неявных нарушений системы статических типов[1]), и требования к инициализации (принудительное выполнение во время компиляции, чтобы переменные в области видимости не нарушали инициализацию)[7] чем C, и поэтому некоторый допустимый код C недопустим в C++. Обоснование этого приведено в Приложении C.1 к стандарту ISO C++.[8]

  • Одно из часто встречающихся отличий заключается в том, что C более слабо типизирован в отношении указателей. В частности, C позволяет присваивать указатель void* любому типу указателя без приведения, в то время как C++ этого не позволяет; эта идиома часто встречается в коде C, использующем для выделения памяти malloc,[9] или при передаче контекстных указателей на POSIX pthreads API, и другие фреймворки, включающие обратные вызовы. Например, следующее допустимо в C, но не в C++:
    void *ptr;
    /* Неявное преобразование из void* в int* */
    int *i = ptr;
    

    или аналогично:

    int *j = malloc(5 * sizeof *j);     /* Неявное преобразование из void* в int* */
    

    Чтобы заставить код компилироваться как на C, так и на C++, необходимо использовать явное приведение следующим образом (с некоторыми оговорками на обоих языках[10][11]):

    void *ptr;
    int *i = (int *)ptr;
    int *j = (int *)malloc(5 * sizeof *j);
    
  • C++ имеет более сложные правила присваивания указателей, которые добавляют квалификаторы, поскольку он позволяет приводить int ** к const int *const *, но не небезопасное присваивание const int **, в то время как C не допускает ни того, ни другого (хотя компиляторы обычно выдают только предупреждение).
  • C++ изменяет некоторые функции стандартной библиотеки языка Си, добавляя дополнительные перегруженные функции с квалификатором типа const, например strchr возвращает char* в C, в то время как C++ поступает так, как если бы были две перегруженные функции const char *strchr(const char *) и char *strchr(char *).
  • C++ также более строг в преобразованиях в перечисления: целые числа не могут быть неявно преобразованы в перечисления, как в C. Кроме того, [[Перечисляемый тип#C++ |константные перечисления]] (enum enumerators) всегда имеют тип int в C, тогда как в C++ они являются различными типами и могут иметь размер, отличный от размера int.
  • В C++ переменная const должна быть инициализирована; в C это необязательно.
  • Компиляторы C++ запрещают goto или switch пересекать инициализацию, как в следующем коде C99:
    void fn(void)
    {
        goto flack;
        int i = 1;
    flack:
        ;
    }
    
  • Несмотря на синтаксическую корректность, функция longjmp() приводит к неопределенному поведению в C++, если фреймы стека с перепрыгиванием включают объекты с нетривиальными деструкторами.[12] Реализация C++ может свободно определять поведение таким образом, чтобы вызывались деструкторы. Однако это мешает некоторым видам использования longjmp() , которые в противном случае были бы допустимы, такие как потоков или сопрограмм переключающихся между отдельными стеками вызовов с помощью longjmp() — при переходе из нижнего стека вызовов в верхний в глобальном адресном пространстве деструкторы будут вызываться для каждого объекта в нижнем стеке вызовов. В C такой проблемы не существует. C допускает несколько предварительных определений одной глобальной переменной в одной единице перевода, что недопустимо как нарушение ODR в C+
  • C допускает несколько предварительных определений одной глобальной переменной в одной единице трансляции, что недопустимо в C++, так как нарушение правила одного определения (англ. One Definition Rule, ODR).
    int N;
    int N = 10;
    
  • В C допустимо объявление нового типа с тем же именем, что и у struct, union или enum, но это недопустимо в C++, потому что в C типы, struct, union, и enum должны указываться всякий раз, когда на тип ссылаются, тогда как в C++ все объявления таких типов неявно содержат typedef.
    enum BOOL {FALSE, TRUE};
    typedef int BOOL;
    
  • Non-prototype ("K&R"-style) function declarations are invalid in C++; they are still valid in C,[13] although they have been deemed obsolescent since C's original standardization in 1990. (The term "obsolescent" is a defined term in the ISO C standard, meaning a feature that "may be considered for withdrawal in future revisions" of the standard.) Similarly, implicit function declarations (using functions that have not been declared) are not allowed in C++, and have been invalid in C since 1999.
  • In C, a function prototype without parameters, e.g. int foo();, implies that the parameters are unspecified. Therefore, it is legal to call such a function with one or more arguments, e.g. foo(42, "hello world"). In contrast, in C++ a function prototype without arguments means that the function takes no arguments, and calling such a function with arguments is ill-formed. In C, the correct way to declare a function that takes no arguments is by using 'void', as in int foo(void);, which is also valid in C++. Empty function prototypes are a deprecated feature in C99 (as they were in C89).
  • In both C and C++, one can define nested struct types, but the scope is interpreted differently: in C++, a nested struct is defined only within the scope/namespace of the outer struct, whereas in C the inner struct is also defined outside the outer struct.
  • C allows struct, union, and enum types to be declared in function prototypes, whereas C++ does not.

C99 and C11 added several additional features to C that have not been incorporated into standard C++, such as complex numbers, variable length arrays (note that complex numbers and variable length arrays are designated as optional extensions in C11), flexible array members, the restrict keyword, array parameter qualifiers, compound literals, and designated initializers.

  • Complex arithmetic using the float complex and double complex primitive data types was added in the C99 standard, via the _Complex keyword and complex convenience macro. In C++, complex arithmetic can be performed using the complex number class, but the two methods are not code-compatible. (The standards since C++11 require binary compatibility, however.)[14]
  • Variable length arrays. This feature leads to possibly non-compile time sizeof operator.[15]
    void foo(size_t x, int a[*]);  // VLA declaration
    void foo(size_t x, int a[x]) 
    {
        printf("%zu\n", sizeof a); // same as sizeof(int*)
        char s[x * 2];
        printf("%zu\n", sizeof s); // will print x*2
    }
    
  • The last member of a C99 structure type with more than one member may be a flexible array member, which takes the syntactic form of an array with unspecified length. This serves a purpose similar to variable-length arrays, but VLAs cannot appear in type definitions, and unlike VLAs, flexible array members have no defined size. ISO C++ has no such feature. Example:
    struct X
    {
        int n, m;
        char bytes[];
    }
    
  • The restrict type qualifier defined in C99 was not included in the C++03 standard, but most mainstream compilers such as the GNU Compiler Collection,[16] Microsoft Visual C++, and Intel C++ Compiler provide similar functionality as an extension.
  • Array parameter qualifiers in functions are supported in C but not C++.
    int foo(int a[const]);     // equivalent to int *const a 
    int bar(char s[static 5]); // annotates that s is at least 5 chars long
    
  • The functionality of compound literals in C is generalized to both built-in and user-defined types by the list initialization syntax of C++11, although with some syntactic and semantic differences.
    struct X a = (struct X){4, 6};  // The equivalent in C++ would be X{4, 6}. The C syntactic form used in C99 is supported as an extension in the GCC and Clang C++ compilers.
    foo(&(struct X){4, 6});         // The object is allocated in the stack and its address can be passed to a function. This is not supported in C++.
    
    if (memcmp(d, (int []){8, 6, 7, 5, 3, 0, 9}, n) == 0) {} // The equivalent in C++ would be using digits = int []; if (memcmp(d, digits{8, 6, 7, 5, 3, 0, 9}, n) == 0) {}
    
  • Designated initializers for arrays are valid only in C:
    char s[20] = { [0] = 'a', [8] = 'g' };  // allowed in C, not in C++
    
  • Functions that do not return can be annotated using a noreturn attribute in C++ whereas C uses a distinct keyword.

C++ adds numerous additional keywords to support its new features. This renders C code using those keywords for identifiers invalid in C++. For example:

struct template 
{
    int new;
    struct template* class;
};
is valid C code, but is rejected by a C++ compiler, since the keywords template, new and class are reserved.

Constructs that behave differently in C and C++

There are a few syntactic constructs that are valid in both C and C++ but produce different results in the two languages.

  • Character literals such as 'a' are of type int in C and of type char in C++, which means that sizeof 'a' will generally give different results in the two languages: in C++, it will be 1, while in C it will be sizeof(int). As another consequence of this type difference, in C, 'a' will always be a signed expression, regardless of whether or not char is a signed or unsigned type, whereas for C++ this is compiler implementation specific.
  • C++ assigns internal linkage to namespace-scoped const variables unless they are explicitly declared extern, unlike C in which extern is the default for all file-scoped entities. Note that in practice this does not lead to silent semantic changes between identical C and C++ code but instead will lead to a compile-time or linkage error.
  • In C use of inline functions requires manually adding a prototype declaration of the function using the extern keyword in exactly one translation unit to ensure a non-inlined version is linked in, whereas C++ handles this automatically. In more detail, C distinguishes two kinds of definitions of inline functions: ordinary external definitions (where extern is explicitly used) and inline definitions. C++, on the other hand, provides only inline definitions for inline functions. In C, an inline definition is similar to an internal (i.e. static) one, in that it can coexist in the same program with one external definition and any number of internal and inline definitions of the same function in other translation units, all of which can differ. This is a separate consideration from the linkage of the function, but not an independent one. C compilers are afforded the discretion to choose between using inline and external definitions of the same function when both are visible. C++, however, requires that if a function with external linkage is declared inline in any translation unit then it must be so declared (and therefore also defined) in every translation unit where it is used, and that all the definitions of that function be identical, following the ODR. Note that static inline functions behave identically in C and C++.
  • Both C99 and C++ have a boolean type bool with constants true and false, but they are defined differently. In C++, bool is a built-in type and a reserved keyword. In C99, a new keyword, _Bool, is introduced as the new boolean type. The header stdbool.h provides macros bool, true and false that are defined as _Bool, 1 and 0, respectively. Therefore, true and false have type int in C.

Several of the other differences from the previous section can also be exploited to create code that compiles in both languages but behaves differently. For example, the following function will return different values in C and C++:

extern int T;

int size(void)
{
    struct T {  int i;  int j;  };
    
    return sizeof(T);
    /* C:   return sizeof(int)
     * C++: return sizeof(struct T)
     */
}

This is due to C requiring struct in front of structure tags (and so sizeof(T) refers to the variable), but C++ allowing it to be omitted (and so sizeof(T) refers to the implicit typedef). Beware that the outcome is different when the extern declaration is placed inside the function: then the presence of an identifier with same name in the function scope inhibits the implicit typedef to take effect for C++, and the outcome for C and C++ would be the same. Observe also that the ambiguity in the example above is due to the use of the parenthesis with the sizeof operator. Using sizeof T would expect T to be an expression and not a type, and thus the example would not compile with C++.

Linking C and C++ code

While C and C++ maintain a large degree of source compatibility, the object files their respective compilers produce can have important differences that manifest themselves when intermixing C and C++ code. Notably:

  • C compilers do not name mangle symbols in the way that C++ compilers do.[17]
  • Depending on the compiler and architecture, it also may be the case that calling conventions differ between the two languages.

For these reasons, for C++ code to call a C function foo(), the C++ code must prototype foo() with extern "C". Likewise, for C code to call a C++ function bar(), the C++ code for bar() must be declared with extern "C".

A common practice for header files to maintain both C and C++ compatibility is to make its declaration be extern "C" for the scope of the header:[18]

/* Header file foo.h */
#ifdef __cplusplus /* If this is a C++ compiler, use C linkage */
extern "C" {
#endif

/* These functions get C linkage */
void foo();
 
struct bar { /* ... */ };

#ifdef __cplusplus /* If this is a C++ compiler, end C linkage */
}
#endif

Differences between C and C++ linkage and calling conventions can also have subtle implications for code that uses function pointers. Some compilers will produce non-working code if a function pointer declared extern "C" points to a C++ function that is not declared extern "C".[19]

For example, the following code:

void my_function();
extern "C" void foo(void (*fn_ptr)(void));

void bar()
{
   foo(my_function);
}

Using Sun Microsystems' C++ compiler, this produces the following warning:

 $ CC -c test.cc
 "test.cc", line 6: Warning (Anachronism): Formal argument fn_ptr of type
 extern "C" void(*)() in call to foo(extern "C" void(*)()) is being passed
 void(*)().

This is because my_function() is not declared with C linkage and calling conventions, but is being passed to the C function foo().

References

  1. 1 2 Stroustrup, Bjarne An Overview of the C++ Programming Language in The Handbook of Object Technology (Editor: Saba Zamir). CRC Press LLC, Boca Raton. 1999. ISBN 0-8493-3135-8. (PDF) 4. Дата обращения: 12 августа 2009. Архивировано 16 августа 2012 года.
  2. B.Stroustrup. C and C++: Siblings. The C/C++ Users Journal. July 2002. Дата обращения: 17 марта 2019.
  3. Bjarne Stroustrup's FAQ – Is C a subset of C++? Дата обращения: 22 сентября 2019.
  4. B. Stroustrup. C and C++: A Case for Compatibility. The C/C++ Users Journal. August 2002. Дата обращения: 18 августа 2013. Архивировано 22 июля 2012 года.
  5. Rationale for International Standard—Programming Languages—C Архивировано 6 июня 2016 года., revision 5.10 (April 2003).
  6. C Dialect Options - Using the GNU Compiler Collection (GCC). gnu.org. Архивировано 26 марта 2014 года.
  7. N4659: Working Draft, Standard for Programming Language C++. Архивировано 7 декабря 2017 года. ("It is invalid to jump past a declaration with explicit or implicit initializer (except across entire block not entered). … With this simple compile-time rule, C++ assures that if an initialized variable is in scope, then it has assuredly been initialized.")
  8. N4659: Working Draft, Standard for Programming Language C++. Архивировано 7 декабря 2017 года.
  9. IBM Knowledge Center. ibm.com.
  10. FAQ > Casting malloc - Cprogramming.com. faq.cprogramming.com. Архивировано 5 апреля 2007 года.
  11. 4.4a — Explicit type conversion (casting) (16 апреля 2015). Архивировано 25 сентября 2016 года.
  12. longjmp - C++ Reference. www.cplusplus.com. Архивировано 19 мая 2018 года.
  13. 2011 ISO C draft standard.
  14. std::complex - cppreference.com. en.cppreference.com. Архивировано 15 июля 2017 года.
  15. Incompatibilities Between ISO C and ISO C++. Архивировано 9 апреля 2006 года.
  16. Restricted Pointers Архивировано 6 августа 2016 года. from Using the GNU Compiler Collection (GCC)
  17. IBM Knowledge Center. ibm.com.
  18. IBM Knowledge Center. ibm.com.
  19. Oracle Documentation. Docs.sun.com. Дата обращения: 18 августа 2013. Архивировано 3 апреля 2009 года.

Шаблон:CProLang Шаблон:C++ programming language

Шаблон:Use dmy dates