Some unmentioned incompatibilities I've encountered that makes a C header not directly usable in C++:<p>- C `_Atomic(T)` and C++ `std::atomic<T>`. C++23 has C compatible header `stdatomic.h` that defines `_Atomic(T)`, but it's still problematic<p>- C `_Noreturn/noreturn` and C++ `[[noreturn]]`. C23 `[[noreturn]]` makes them compatible<p>- C inline and C++ inline are different. Good news is their `static inline` are the same<p>- C has anonymous struct. C++ doesn't. Both have anonymous union though
C is dealing with bits and bytes. C++ is building a tower of abstractions, one on top of another. For this dramatic difference in approach, the incompatibilities are surprisingly small.
I wrote this after repeatedly seeing experienced C programmers hit the same sharp edges while moving into modern C++ codebases.<p>Many of these differences are intentional and defensible from the C++ side. But some are still surprising because they invalidate patterns that were historically common, performant, or idiomatic in C.<p>The interesting part to me isn’t "C vs C++," but where the languages diverged philosophically: object lifetime vs raw storage, stronger type systems, implicit conversions, ABI and optimization assumptions, and the boundary between "portable" and "works on my compiler."<p>I’d also be curious which C constructs people still genuinely miss in modern C++. For me, restrict is still near the top of the list.
Not sure if you're aware, but defer is proposed for C2Y [1]. It's already available in Clang behind a compiler flag. It <i>is</i> interesting how the languages continue to diverge.<p>[1] <a href="https://www.open-std.org/JTC1/SC22/WG14/www/docs/n3734.pdf" rel="nofollow">https://www.open-std.org/JTC1/SC22/WG14/www/docs/n3734.pdf</a>
Because the communities aren't the same.<p>C++ is 1990's Typescript for C++, while C folks still think is a portable Assembly instead of designed to an abstract machine model.<p>As such C++ community embraces high level abstractions and type systems improvements, whereas C wants to still code as targeting classical hardware.
I appreciate that restrict isn't there, because it is yet another UB source, programmer knows not to do errors kind of attitude, and secondly no one seems to care enough to write a language proposal for it.
From the article:<p><pre><code> In 2019 I wrote a short survey of C constructs that do not
work in C++. The point was not that C is sloppy or that C++
is superior. The point was that C++ is not a superset of C,
and that C programmers crossing the border should know
where the checkpoints are.
</code></pre>
C++ was a superset of C 30-ish years ago. Now, as the author correctly identifies, it is not as both have taken different evolutionary paths.
> restrict: a C promise, not a C++ contract<p>This takes the cake.