Bjarne Stroustrup's C Style and Technique FAQ
Modified January 26, 2005
These are questions about C Style and Technique that people ask me often. If you have better questions or comments on the answers, feel free to email me (bs@research.att.com). Please remember that I can not spend all of My Time Improving My HomePages.
For More General Questions, See My General FAQ.
For Terminology and Concepts, See My C Glossary.
Please note that these are just a collection of questions and answers. They are not a substitute for a carefully selected sequence of examples and explanations as you would find in a good textbook. Nor do they offer detailed and precise specifications as you would find in a reference manual or the standard. See The Design and Evolution of C for questions related to the design of C . See The C Programming Language for questions about the use of C and its standard library.
Here is a chinese...........
Index
How do I write this very simple program? Why do my compiles take so long? Why do I have to put the data in my class declarations? Why are member functions not virtual by default? Why do not we have virtual constructors? Why are destructors not virtual by default? Why does not overloading work for derived classes? Can I call a virtual function from a constructor? Is there a "placement delete"? Can I stop people deriving from my class? Why does not C provide heterogenous containers? What's wrong with arrays? does "friend" violate encapsulation? Why does C have both pointers and references? Why is "this" not a reference? Why does not C have a final keyword? Should I use NULL or 0? How Are C Objects LAID OUT IN MEMORY? What's The Value Of i ? why can't i define constraints for my template parameters? why can't i assign a vector
T THE DESTRUCTOR CALLED AT THE END OE SCOPE? CAN I WRITE "void main ()"? why can't i Overload Dot, ::, sizeof, etc.? How do i convera integer to a string? How do i call A c FUNCTION FROM C ? How do i call a c function from c? IS `` Int * p; '' Right or IS `` Int * p; '' Right? Which Layout Style Is Best for My Code? SHOULD i PUT "const" before or after the type? What good is static_cast? So, what's wrong with using macros? How do you pronounce "cout"? How do you pronounce "char"? How do I write this very simple program? Often, especially at the start of semesters, I get a lot of questions about how to write very simple programs. Typically, the problem to be solved is to read in a few numbers, do something with them, and write out an answer. Here is a sample Program That Does That:
#include
#include
#include
Using namespace std;
int main ()
{
Vector
Double D;
While (cin >> d) v.push_back (d); // read elements
IF (! cin.eof ()) {// check if input failed
CERR << "Format Error / N";
Return 1; // Error Return
}
Cout << "read" << v.size () << "Elements / N";
Reverse (v.begin (), v.end ());
COUT << "Elements in Reverse Order: / N";
For (int i = 0; i Return 0; // Success Return } Here Are A Few Observations About this Program: This is a Standard ISO C program using the standard library. Standard library facilities are declared in namespace std in headers without a .h suffix. If you want to compile this on a Windows machine, you need to compile it as a "console application" . Remember to give your source file the .cpp suffix or the compiler might think that it is C (not C ) source. Yes, main () returns an int. Reading into a standard vector guarantees that you do not overflow some arbitrary buffer . Reading into an array without making a "silly error" is beyond the ability of complete novices -. by the time you get that right, you are no longer a complete novice If you doubt this claim, I suggest you read my paper "Learning Standard C as a New Language ", which you can download from my publications list. The! cin.eof () is a test of the stream's format. Specifically, it tests whether the loop ended by finding end-of-file (if not You Didn't Get Input of the Expected Type / Format). for more information, Look Up "Stream state" in your C textbook. A vector knows its size, so I do not have to count elements. This program contains no explicit memory management, and it does not leak memory. A vector keeps track of the memory it use to . store its elements When a vector needs more memory for elements, is allocates more; when a vector goes out of scope, it frees that memory Therefore, the user need not be concerned with the allocation and deallocation of memory for vector elements for.. Reading in strings, See How do I read A String from Input ?. The Program Endings Reading INPUT WHEN IT SEES "End of File". If you run the program from the keybord on a unix machineine end of file "is Ctrl-D. If you are on a Windows machine that because of a bug does not recognize an end-of-file character, you might prefer this slightly more complicated version of the program that terminates input with the word "end": #include #include #include Using namespace std; int main () { Vector Double D; While (cin >> d) v.push_back (d); // read elements IF (! cin.eof ()) {// check if input failed cin.clear (); // Clear Error State String S; Cin >> S; // Look for Terminator String IF (s! = "end") { CERR << "Format Error / N"; Return 1; // Error Return } } Cout << "read" << v.size () << "Elements / N"; Reverse (v.begin (), v.end ()); COUT << "Elements in Reverse Order: / N"; For (int i = 0; i Return 0; // Success Return } For more examples of how to use the standard library to do simple things simply, see "Tour of the standard library" chapter of TC PL3 (Available For Download). Why Do My Compiles Take So Long? You May Have a Problem With your compiler. It may be old, you may have it installed wrongly, or your computer might be an antique. I can't help you with such problems. However, it is more likely that the program that you are trying to compile is poorly designed, so that compiling it involves the compiler examining hundreds of header files and tens of thousands of lines of code. In principle, this can be avoided. If this problem is in your library vendor's design, there is not much you can do (except changing to a better library / vendor), but you can structure your own code to minimize re-compilation after changes. Designs that do that are typically better, More Maintainable, Designs Because The EXHIBIT BETER SEPARATION OF EXCONSIDER A Classical Example of An Object-Oriented Program: Class shape { Public: // Interface to users of shapes Virtual void Draw () const; Virtual Void Rotate; Int De De // ... Protected: // Common Data (for Implementers of Shapes) Point center; Color color // ... } Class Circle: Public Shape { PUBLIC: Void Draw () Const; Void rotate (int) {} // ... protected: int RADIUS; // ... } Class Triangle: Public Shape { PUBLIC: Void Draw () Const; Void Rotate (int); // ... protected: Point A, B, C; // ... } The idea is that users manipulate shapes through Shape's public interface, and that implementers of derived classes (such as Circle and Triangle) share aspects of the implementation represented by the protected members. There Are Three Serious Problems With this apparently Simple Idea: It is not easy to define shared aspects of the implementation that are helpful to all derived classes. For that reason, the set of protected members is likely to need changes far more often than the public interface. For example, even though "center" is arguably a valid concept for all Shapes, it is a nuisance to have to maintain a point "center" for a Triangle - for triangles, it makes more sense to calculate the center if and only if someone expresses interest in it The protected members are. likely to depend on "implementation" details that the users of Shapes would rather not have to depend on. For example, many (most?) code using a Shape will be logically independent of the definition of "Color", yet the presence of Color in the definition of Shape will probably require compilation of header files defining the operating system's notion of color When something in the protected part changes, users of Shape have to recompile -. even though only implementers of derived classes have ac cess to the protected members.Thus, the presence of "information helpful to implementers" in the base class that also acts as the interface to users is the source of instability in the implementation, spurious recompilation of user code (when implementation information changes), And Excess Inclusion of Heooter Files INTO User Code (Because The "Information Helpful To Implement" Needs Those Headers). This is some class problem. "BRITTLE BASE CLASS Problem. The obvious solution is to omit the "information helpful to implemeters" for classes that are used as interfaces to users That is, to make interfaces, pure interfaces That is, to represent interfaces as abstract classes:.. Class Shape { Public: // Interface to users of shapes Virtual void Draw () const = 0; Virtual Void Rotate (int De De degRees) = 0; Virtual point center () const = 0; // ... // no data } Class Circle: Public Shape { PUBLIC: Void Draw () Const; Void rotate (int) {} Point center () const {return center // ... protected: Point cent; Color color int RADIUS; // ... } Class Triangle: Public Shape { PUBLIC: Void Draw () Const; Void Rotate (int); Point center () const; // ... protected: Color color Point A, B, C; // ... } The Users Are Now Insulated from changes. I have seen this technique decrease build times by Orders of Magnitude. But what if the really is some information? Is comment (or simply to several derived class)? Simply Make That Information Class from That Also: Class shape { Public: // Interface to users of shapes Virtual void Draw () const = 0; Virtual Void Rotate (int De De degRees) = 0; Virtual point center () const = 0; // ... // no data } Struct Common { Color color // ... } Class Circle: Public Shape, Protected Common { PUBLIC: Void Draw () Const; Void rotate (int) {} Point center () const {return center // ... protected: Point cent; int RADIUS; } Class Triangle: Public Shape, Protected Common { PUBLIC: Void Draw () Const; Void Rotate (int); Point center () const; // ... protected: Point A, B, C; } Why is the size of an empty class not zero To ensure that the addresses of two different objects will be different For the same reason, "new" always returns pointers to distinct objects Consider:?.. Class Empty {}; Void f () { EMPTY A, B; IF (& a == & b) cout << "Impossible: Report Error To Compiler Suppilier; EMPTY * P1 = New Empty EMPTY * P2 = New Empty IF (p1 == p2) cout << "Impossible: Report Error to Compiler Suppilier; } There is an interesting rule thing say, one an es Empty base Class Need NOT BE REPRESENTED BY A SEPARATE BYTE: Struct x: Empty { Int a; // ... } Void f (x * p) { Void * p1 = p; Void * p2 = & p-> a; IF (p1 == p2) cout << "nice: good Optimizer"; } This optimization is safe and can be most useful. It allows a programmer to use empty classes to represent very simple concepts without overhead. Some current compilers provide this "empty base class optimization". Why do i have to put the data in my class declarations? You don't. If you don't want the interface. Put it in defines the interface. Put it in derived classes instead. See , Why do my compiles take so long ?. Sometimes, You do Want to Have Representation Data In a class. Consider Class Complex: Template PUBLIC: Complex (): Re (0), IM (0) {} Complex (Scalar R): RE (R), IM (0) {} Complex (Scalar R, Scalar I): RE (R), IM (i) {} // ... Complex & Operator = (Const Complex & A) {RE = A.RE; IM = a.im; return * this;} // ... Private: Scalar RE, IM; } This type is designed to be used much as a built-in type and the representation is needed in the declaration to make it possible to create genuinely local objects (ie objects that are allocated on the stack and not on a heap) and to ensure proper inlining of simple operations. genuinely local objects and inlining is necessary to get the performance of complex close to what is provided in languages with a built-in complex type.Why are member functions not virtual by default? Because many classes are not designed to be Used as base classes. for example, see Class Complex. Also, objects of a class with a virtual function require space needed by the virtual function call mechanism -. Typically one word per object This overhead can be significant, and can get in the way of layout compatibility with data from other languages (eg C and FORTRAN). See The Design and Evolution of C for More Design Rational. Why are destructors not virtual by default? Because many classes are not designed to be used as base classes. Virtual functions make sense only in classes meant to act as interfaces to objects of derived classes (typically allocated on a heap and accessed through pointers or references ). So when should I declare a destructor virtual? Whenever the class has at least one virtual function. Having virtual functions indicate that a class is meant to act as an interface to derived classes, and when it is, an object of a derived class may be Destroyed THROUGH a Pointer to the base. for example: Class base { // ... Virtual ~ base (); } Class Derived: public base { // ... ~ Derived (); } Void f () { Base * p = new deerid; Delete P; // Virtual Destructor Used to Ensure That ~ Derived Is Called } Had Base's destructor not been virtual, Derived's destructor would not have been called - with likely bad effects, such as resources owned by Derived not being freed.Why do not we have virtual constructors A virtual call is a mechanism to get work done given? partial information. In particular, "virtual" allows us to call a function knowing only an interfaces and not the exact type of the object. to create an object you need complete information. In particular, you need to know the exact type of what you Want to create. Consequently, a "Call to a constructor" Cannot Be Virtual. Techniques for Using An Object Are Off To Create An Object Are Off The as "Virtual Constructors". For Example, SEE TC PL3 15.6.2. For Example, Here Is A Technique for Generating An Object of An Appropriate Type Using AN Abstract Class: Struct f {// interface to Object Creation functions Virtual a * make_an_a () const = 0; Virtual B * Make_a_b () const = 0; } Void User (Const F & FAC) { A * p = fac.make_an_a (); // make an a of the appropriate Type B * Q = fac.make_a_b (); // make a b of the appropriate Type // ... } Struct fx: f { A * make_an_a () const {return new ax ();} // ax is deived from a B * Make_a_b () const {return new bx ();} // bx is deive from b } Struct fy: f { A * make_an_a () const {return new aY ();} // ay is deived from a B * Make_a_b () const {return new by ();} // by is deact from b } int main () { FX X; FY Y; User (x); // this user makes axs and bxs User (y); // this user makes ays and bys User (fx ()); // this user makes axs and bxs User (fy ()); // this user makes ays and bys // ... } This is a variant of what is often called "the factory pattern". The point is that user () is completely isolated from knowledge of classes such as AX and AY.Why does not overloading work for derived classes? That question (in many Variations Are Usually Prompted by An Example Like this: #include Using namespace std; Class b { PUBLIC: INT F (INT i) {cout << "f (int):"; Return i 1;} // ... } Class D: public b { PUBLIC: Double F (Double D) {cout << "f (double):"; Return D 1.3;} // ... } int main () { D * pd = new d; COUT << PD-> F (2) << '/ n'; Cout << PD-> f (2.3) << '/ n'; } Which Will Produnce: f (double): 3.3 f (double): 3.6 Rather Than f (int): 3 f (double): 3.6 That Some People (WRONGLY) GUESSED. In Other Words, There Is No Overload Resolution Between D and B. The Compiler Looks INTO The Scope Of D, FINDS The Single Function "Double F (Double)" AND calls it. It never bothers with the (Enclosing) Scope Of B. IN C , There is no overloading across scopes - Derived Class Scopes Are Not An Exception To this General Rule. (See D & E OR TC PL3 for Details). But what if I want to create an overload set of all my f () Functions from my base and derived class? That's Easily Done Using A Using-Declaration: Class D: public b { PUBLIC: Using b :: f; // make every f from b available Double F (Double D) {cout << "f (double):"; Return D 1.3;} // ... } Give That Modification, The Output Will Be f (int): 3 f (double): 3.6 . Can I call a virtual function from a constructor? Yes, but be careful. It may not do what you expect. In a constructor, the virtual call mechanism is disabled because overriding from derived classes has not yet happened. Objects are constructed from the Base Up, "Base Before Derived" .consideer: #include #include Using namespace std; Class b { PUBLIC: B (Const string & ss) {cout << "b constructor / n"; f (ss);} Virtual void f (const string &) {cout << "b :: f / n";} } Class D: public b { PUBLIC: D (Const String & SS): B (SS) {cout << "D Constructor / N"; Void F (const string & ss) {cout << "D:: f / n"; s = ss;} Private: String S; } int main () { D ("Hello"); } The Program Compiles and Product B Constructor B :: F D Constructor Note NOT D :: F. Consider What Wouldered So That D :: f () WAEN Called from b :: b (): Because the constructor d :: d () Hadn't yet been run, d :: f () Would try.................... .. Destruction is done "derived class before base class", so virtual functions behave as in constructors: Only the local definitions are used - and no calls are made to overriding functions to avoid touching the (now destroyed) derived class part of the object. For More Details See D & E 13.2.4.2 OR TC PL3 15.4.3. It has been suggested that this rule is an implementation artifact. It is not so. In fact, it would be noticeably easier to implement the unsafe rule of calling virtual functions from constructors exactly as from other functions. However, that would imply that no virtual Function Could Be Written to Rely on Invariants Established by Base Classes. That Would Be a Terrible Mess.is There a "Placement DELETE"? No but if you need one you can write your ows. Consider Placement New Used to Place Objects in A Set of Arenas Class arena { PUBLIC: Void * allocate (SIZE_T); Void Deallocate (Void *); // ... } Void * Operator New (Size_t Sz, Arena & A) { Return A.allocate (SZ); } Arena A1 (Some Arguments); Arena A2 (Some Arguments); Given That, We can WRITE X * p1 = new (a1) x; Y * p2 = new (a1) y; Z * p3 = new (a2) z; // ... But how can we later delete those objects correctly? The reason that there is no built-in "placement delete" to match placement new is that there is no general way of assuring that it would be used correctly. Nothing in the C type system allows US to DEDUCE THAT P1 P1 Points to an Object Allocated in Arena A1. a Pointer to anyxi x allocated anywhere can be assigned to p1. However, Sometimes The Programmer Does Know, and There IS A WAY: Template { IF (p) { P-> ~ t (); // Explicit Destructor Call A.deallocate (P); } } Now, we can write: Destroy (P1, A1); Destroy (P2, A2); Destroy (P3, A3); IF An Arena Keeps Track of What Objects It Holds, You Can Even Write Destroy () To Defend Itself Against Mistakes. IT IS Also Possible to Define a matching operator new () and operator delete () Pairs for a class Hierarchy TC PL (SE) 15.6. See Also D & E 10.4 and TC PL (SE) 19.4.5.can i stop People Deriving from My Class? Yes, But why do you want to? There is TWO Common answers: for efficiency: to avoid my function calls being virtual for safety: to ensure that my class is not used as a base class (for example, to be sure that I can copy objects without fear of slicing) In my experience, the efficiency reason is usually misplaced fear. In C , virtual function calls are so fast that their real-world use for a class designed with virtual functions does not to produce measurable run-time overheads compared to alternative solutions using ordinary function calls. Note that the virtual function call Mechanism is Typically Used Only WHEN CALING THROUGH A POINTER OR a Reference. When Calling a Function Directly for a named Object, The Virtual Function Class Overhead is Easily Optimized Away. If there is a genuine need for "capping" a class hierarchy to avoid virtual function calls, one might ask why those functions are virtual in the first place. I have seen examples where performance-critical functions had been made virtual for no good reason, Just Because "That's The Way We Usually Do IT". . The other variant of this problem, how to prevent derivation for logical reasons, has a solution Unfortunately, that solution is not pretty It relies on the fact that the most derived class in a hierarchy must construct a virtual base For example..: Class usable; Class USABLE_LOCK { Friend class usable; Private: USABLE_LOCK () {} USABLE_LOCK (const usable_lock &) {} } Class usable: public virtual usable_lock {// ... PUBLIC: USABLE (); Usable (char *); // ... } USABLE A; Class Dd: public usable {}; DD DD; // Error: Dd :: DD () Cannot Access // USABLE_LOCK :: usable_lock (): Private Member (from From D & E Sec..4.3). Why Doesn't C Provide HETEROGENOUS CONTAINERS? The C Standard Library Provides a set of useful, statically type-save, and effect containers. Examples Are Vector, List, And Map: Vector Vector List List Map Map Thase Containers Are Described in All Good C Textbooks, and Should Be Preferred over Arrays and "Home Cooked" Containers Unless there is a good reason not to. These containers are homogeneous;. That is, they hold elements of the same type If you want a container to hold elements of several different types, you must express that either as a union or (usually much better) as a container of pointers to a Polymorphic Type. The Classical Example IS: Vector Here, vi can hold elements of any type derived from Shape. That is, vi is homogeneous in that all its elements are Shapes (to be precise, pointers to Shapes) and heterogeneous in the sense that vi can hold elements of a wide variety of Shapes, Such As Circles, Triangles, ETC. So, in a sense all containers (in every language) are homogenous because to use them there must be a common interface to all elements for users to rely on. Languages that provide containers deemed heterogenous simply provide containers of elements that all provide a standard interface . For example, Java collections provide containers of (references to) Objects and you use the (common) Object interface to discover the real type of an element.The C standard library provides homogeneous containers because those are the easiest to use in the vast majority Of Cases, Gives The Best Compile-Time Error Message, and Imposes No Unnecessary Run-Time Overheads. If you need a heterogeneous container in C , Define a common interface for all the elements and make a container of those. For example: Class IO_OBJ {/ * ... * /}; // the interface neededed to take part in Object I / O Vector Vector Don't Drop to the Lowest Level of Implementation Detail Unless You Have To: Vector A good indictive That You Have "Gone Too Low Level" is this your code gets limited with Casts. Does "friend" violate encapsulation? No. It does not. "Friend" is an explicit mechanism for granting access, just like membership. You can not (in a standard conforming program) grant yourself access to a class without modifying its source. For example : Class x { INT I; PUBLIC: Void M (); // Grant x :: m () access Friend Void F (X &); // Grant F (X &) Access // ... } Void x :: m () {i ; / * x :: m () CAN access x :: i * /} void f (x <) {x.i ; / * f (x &) can access x :: i * /} For A Description on the C Protection Model, See D & E Sec 2.10 and Tc PL Sec. 1.5, 15.3, and C.11. Why does C have both pointers and references? C inherited pointers from C, so I could not remove them without causing serious compatibility problems. References are useful for several things, but the direct reason I introduced them in C was to support operator overloading. For example: Void F1 (const complex * x, const complex * y) // without references { Complex z = * x * y; // ugly // ... } Void F2 (Const Complex & x, Const Complex & Y) // with References { Complex z = x y; // better // ... } More generally, if you want to have both the functionality of pointers and the functionality of references, you need either two different types (as in C ) or two different sets of operations on a single type. For example, with a single type you need Both An Operation to Assign To the Object Referred to and an Operation To Assign To The Reference / Pointer. This Can Be Done Using Separate Operators (as in Simula). for example: REF R: = 7; // Assign to Object R: - new my_type; // assign to reasonnce Alternatively, you Could Rely on type checking (overloading) .for example: REF R = 7; // Assign to Object R = new my_type; // Assign to Reference Why is "this" not a reference? Because "this" was introduced into C (really into C with Classes) before references were added. Also, I chose "this" to follow Simula usage, rather than the (later) Smalltalk use of "Self" .what's WRONG WITH ARRAYS? In terms of time and space, an array is just about the optimal construct for accessing a sequence of objects in memory. It is, however, also a very low level data structure with a vast potential for misuse and errors and in essentailly all cases there Are Better Alternatives. by "better" I mean Easier to Write, Easier To Read, Less Error Prone, and as fast. The Two Fundamental Problems with arrays are there ARRAY DOESN'T KNOW ITS OWN SIZE The Name of an Array Converts To a Pointer To Its First Element At The Slightst ProVocation Consider Some Examples: Void F (int A [], int S) { // Do Something with a; The size of a is s For (INT I = 0; i } Int Arr1 [20]; Int Arr2 [10]; Void g () { f (arr1, 20); f (arr2, 20); } The second call will scribble all over memory that does not belong to arr2. Naturally, a programmer usually get the size right, but it's extra work and ever so often someone makes the mistake. I prefer the simpler and cleaner version using the standard library Vector: Void f (Vector { // Do Something with V For (int i = 0; i } Vector Vector Void g () { f (v1); f (v2); } Since An Array Doesn't Know ITS Size, There Can Be No Array Assignment: Void f (int A [], int b [], int size) { A = b; // not array assignment Memcpy (A, B, Size); // a = B // ...} Again, I Prefer Vector: Void G (Vector { A = B; // ... } . Void f (string a [], string b [], int size) { A = b; // not array assignment Memcpy (A, B, Size); // Disaster // ... } Void G (Vector { A = B; // ... } ARRAY IS OF A FIXED SIZE DETERMINED AT Compile Time: Const int s = 10; Void f (int S) { Int a1 [s]; // error Int a2 [s]; // ok // if I want to exchange a2, i'll have to chage to an array // allocated on free store using malloc () and use falpoc () // ... } To Contrast: Const int s = 10; Void G (int S) { Vector Vector v2.resize (v2.size () * 2); // ... } C99 Allows Variable Array Bounds for Local Arrays, But Those Vlas Have The Their Own Problems. The Way That Array Names "DECAY" INTO POINTERS IS FUNDAMENTAL TO THEIR USE IN C and C . However, Array Decay Interact Very Badly with inheritance. Consider: Class Base {void fct (); / * ... * /}; Class derived {/ * ... * /}; Void F (Base * P, Int Sz) { For (int i = 0; i } Base AB [20]; Derived ad [20]; Void g () { f (ab, 20); f (AD, 20); // disaster! } In the last call, the derived [] is what subscripting no longer Works Correctly when Sizeof (Derived)! = Sizeof (base) - As Will Be The Case In Most Cases of Interest. IF We buy Vectors Instead, The Error Would Be Caught At Compile Time: Void F (Vector { For (int i = 0; i Vector Vector Void g () { f (ab); f (ad); // error: Cannot Convert a Vector } I Find this an AN ASTONSHING NUMBER OF NoVice Programming Errors In C and C Relate To (MIS) Uses of Arrays. Why Doesn't C Have a final keyword? There Didn't (AND doesn't) Seem to Be a success. Should I use NULL or 0? In C , the definition of NULL is 0, so there is only an aesthetic difference. I prefer to avoid macros, so I use 0. Another problem with NULL is that people sometimes mistakenly believe that it is different From Pre-Standard Code, NULL WAS / IS SOMETIMES Defined to Something Unsuitable and Therefore HAD / HAS TO BE Avoided. That's Less Common these Days. How are C objects laid out in memory? Like C, C does not define layouts, just semantic constraints that must be met. Therefore different implementations do things differently. Unfortunately, the best explanation I know of is in a book that is otherwise outdated And Doesn't Describe ANY CURRENT C Implementation: The Annotated C Reference Manual (Usually Called The ARM). IT Has Diagrams of Key Layout Examples. There is a Very Brief Explanation in Chapter 2 of TC PL. Basically, C Constructs Objects Simply by ConcateNating Sub Objects. Thus Struct a {Int A, B; IS represented by Two INTS next to each other, and Struct B: a {INT C;}; IS represented by An a followed by an int; what is, by Three INTS Next TO EACH OTHER. Virtual functions are typically implemented by adding a pointer (the vptr) to each object of a class with virtual functions. This pointer points to the appropriate table of functions (the vtbl). Each class has its own vtbl shared by all objects of that class ?. .What's the value of i i It's undefined Basically, in C and C , if you read a variable twice in an expression where you also write it, the result is undefined do not do that Another example is..: v [i] = i ; Related Example: f (v [i], i ); Here, The Result is undefined Because The ORDER OF Evaluation of Function Arguments Are Undefined. Having the order of evaluation undefined is claimed to yield better performing code. Compilers could warn about such examples, which are typically subtle bugs (or potential subtle bugs). I'm disappointed that after decades, most compilers still do not warn, leaving That Job To Specialized, Separate, And Underused Tools. Why can't i define constraints for my template parameters? Well, you can, and it's quite easy and general. CONSIDER: Template Void Draw_all (Container & C) { For_each (c.begin (), c.end (), MEM_FUN (& Shape :: Draw)); } If there is a type error, it will be in the resolution of the fairly complicated for_each () call. For example, if the element type of the container is an int, then we get some kind of obscure error related to the for_each () Call (Because We can't Invoke Shape :: DRAW () for an int). To Catch Such Errors Early, i can write: Template Void Draw_all (Container & C) { Shape * p = c.front (); // accept Only Containers of Shape * S For_each (C. Segin (), C.End (), Mem_Fun (& Shape :: Draw); The initialization of the spurious variable "p" will trigger a comprehensible error message from most current compilers Tricks like this are common in all languages and have to be developed for all novel constructs In production code, I'd probably write something like..: Template Void Draw_all (Container & C) { Typedef Typename Container :: Value_Type T; CAN_copy For_each (c.begin (), c.end (), MEM_FUN (& Shape :: Draw)); } This Makes It Clear That I'm Making An Assertion. The Can_copy Template Can Be Defined Like this: Template Static void constraints (t1 a, t2 b) {t2 c = a; b = a;} CAN_COPY () {void (* p) (T1, T2) = constructs; } CAN_COPY TIME) THAT A T1 CAN Be Assigned to A T2. CAN_COPY one line to name the constraints to be checked and the types for which to check them one line to list the specific constraints checked (the constraints () function) one line to provide a way to trigger the check (the constructor) Note Also That The Definition Has The Desirable Properties That You can express constraints without declaring or copying variables, thus the writer of a constraint does not have to make assumptions about how a type is initialized, whether objects can be copied, destroyed, etc. (unless, of course, those are the properties being tested by the constraint) No code is generated for a constraint using current compilers No macros are needed to define or use constraints Current compilers give acceptable error messages for a failed constraint, including the word "constraints" (to give the reader a clue) , the name of the constraints, and the specific error that caused the failure (eg "can not initialize Shape * by double *") So why is something like Can_copy () - or something even more elegant -? not in the language D & E contains an analysis of the difficulties involved in expressing general constraints for C . Since then, many ideas have emerged for making these constraints classes easier to write and still trigger good error messages. for example, I believe th e use of a pointer to function the way I do in Can_copy originates with Alex Stepanov and Jeremy Siek I do not think that Can_copy () is quite ready for standardization -.. it needs more use Also, different forms of constraints are in use In the C Community; Thee IS Not Yet a Constraints Templates Is The MOST EFFECTIVE OVER A WIDE RANGE OF USES. However, the idea is very general, more general than language facilities that have been proposed and provided specifically for constraints checking After all, when we write a template we have the full expressive power of C available Consider..: Template Static void constraints (t * p) {b * Pb = p;} Derived_From () {void (* p) (t *) = constructs; } Template Static void constraints (t1 a, t2 b) {t2 c = a; b = a;} CAN_COPY () {void (* p) (T1, T2) = constructs; } Template Static void constraints (t1 a, t2 b) {a == b; a! = b; a
CAN_compare () {void (* p) (t1, t2) = constructs; } Template Static void constraints (t1 a, t2 b, t3 c) {c = a * b;} CAN_MULTIPLY () {void (* p) (T1, T2, T3) = constructs; } Struct b {}; Struct D: b {}; Struct Dd: D {}; Struct x {}; int main () { Derived_from Derived_from Derived_from Derived_from Derived_from CAN_Compare CAN_Compare Can_multiply CAN_MULTIPLY CAN_MULTIPLY (); CAN_COPY CAN_COPY CAN_COPY } // The classical "elements must derived from mybase *" Constraint: Template // ... } Actually, Derived_From Doesn't Check DeriVation, But That Offen A Better Constaint. Finding Good Names for Constraints Can Be Hard. Why use sort () WHEN WE HAVE "Good old qsort ()"? To a novel, Qsort (Array, Asize, SizeOf (Elem), ELEM_COMPARE); Looks pretty weird, and is harder to understand Than Sort (vec.begin (), vec.end ()); To an expert, the fact that sort () tends to be faster than qsort () for the same elements and the same comparison criteria is often significant. Also, sort () is generic, so that it can be used for any reasonable combination of Container Type, Element Type, And Comparison criterion. for example: struct record { String name; // ... } Struct Name_compare {// compare records sale "name" as the key Bool Operator () (Const Record & A, Const Record & B) Const {Return A.Name } Void F (Vector { Sort (vs.begin (), vs.end (), name_compare ()); // ... } In Addition, Most People Appreciate That () IS TYPE SAFE, THAT No Casts Are Required To Use it, and there. For a more detailed explanation, see my paper "Learning C as a new language", Which You can download from my publications list. . What is a function object? An Object That in Some Way Behaves Like A Function, Of Course. Typically, That Would Mean An Object of a class. Defines The Application Operator - Operator (). A function object is a more general concept than a function because a function object can have state that persist across several calls (like a static local variable) and can be initialized and examined from outside the object (unlike a static local variable). For example : Class Sum { Int Val; PUBLIC: SUM (INT I): VAL (i) {} Operator int () const {return val;} // extract value INT operator () (int i) {return val = i;} // Application } Void f (Vector { Sum s = 0; // Initial value 0 S = for_each (v.begin (), v.end (), s); // Gather the sum of all elementscout << "" the suuse "<< S <<" / n "; // or even: COUT << "" the suuse "<< for_each (v.begin (), v.end (), sum (0)) <<" / n "; } Note that a function object with an inline application operator inlines beautifully because there are no pointers involved that might confuse optimizers To contrast:. (? Never) current optimizers are rarely able to inline a call through a pointer to function. Function Objects Are Extensively Used to Provide Flexibility In The Standard Library. How do I deal with memory leaks? By writing code that does not have any. Clearly, if your code has new operations, delete operations, and pointer arithmetic all over the place, you are going to mess up somewhere and get leaks, stray pointers, etc. This is true independently of how conscientious you are with your allocations:. eventually the complexity of the code will overcome the time and effort you can afford It follows that successful techniques rely on hiding allocation and deallocation inside more manageable types Good. Examples are the standard containers. They manage memory for Their Elements Better Than You Could. Consider Writing this without the help of string and vector: #include #include #include #include Using namespace std; int main () // Small Program Messing Around with strings { COUT << "Enter Some Whitespace-Separated Words: / N"; Vector String S; While (cin >> s) v.push_back (s); Sort (v.begin (), v.end ()); String cat; TypedEf Vector COUT << Cat << '/ n'; } What Would Be Your Chance of getting it right the first time? And how would you know you? Didn't Have a Leak? Note the absence of explicit memory management, macros, casts, overflow checks, explicit size limits, and pointers. By using a function object and a standard algorithm, I could have eliminated the pointer-like use of the iterator, but that seemed overkill for Such a tiny program. These techniques are not perfect and it is not always easy to use them systematically. However, they apply surprisingly widely and by reducing the number of explicit allocations and deallocations you make the remaining examples much easier to keep track of. As early as 1981, I pointed out that by reducing the number of objects that I had to keep track of explicitly from many tens of thousands to a few dozens, I had reduced the intellectual effort needed to get the program right from a Herculean task to something manageable, or even easy . If your application area does not have libraries that make programming that minimizes explicit memory management easy, then the fastest way of getting your program complete and correct might be to first build such a library. Templates and the Standard Libraries make this use of containers, Resource Handles, etc., Much Easier Than IT Was Even A Few Years Ago. The Use of Exceptions Makes It Close To Essential. If you can not handle allocation / deallocation implicitly as part of an object you need in your application anyway, you can use a resource handle to minimize the chance of a leak. Here is an example where I need to return an object allocated on the free store from a function. This is an opportunity to forget to delete that object. After all, we can not tell just looking at pointer whether it needs to be deallocated and if so who is responsible for that. Using a resource handle, here the standard library auto_ptr , Makes It Clear Where The Responsibility Lies: #include #include Using namespace std; Struct s { S () {cout << "make AN s / n"; ~ S () {cout << "destroy an S / N"; S (const S &) {cout << "Copy Initialize An S / N"; S & Operator = (const S &) {cout << "Copy Assign AN S / N"; } S * f () { RETURN New S; // Who Is Responsible for DELETING S? } Auto_Ptr { Return auto_ptr } int main () { COUT << "start main / n"; S * p = f (); Cout << "after f () before g () / n"; // s * q = g (); // this error Would Be caught by the compiler Auto_ptr COUT << "exit main / n"; // Leaks * P // iMPlicitly deletes * Q } Think About Resources In General, Rather Than Simply About Memory. If systematic application of these techniques is not possible in your environment (you have to use code from elsewhere, part of your program was written by Neanderthals, etc.), be sure to use a memory leak detector as part of your standard development procedure, or plug in a garbage collector.Why can not I resume after catching an exception? in other words, why does not C provide a primitive for returning to the point from which an exception was thrown and continuing execution from there? Basically, someone resuming from an exception handler can never be sure that the code after the point of throw was written to deal with the excecution just continuing as if nothing had happened. An exception handler can not know how much context to "get right" before resuming . to get such code right, the writer of the throw and the writer of the catch need intimate knowledge of each others code and context. This creates a complicated mutual dependency that wherever it has been allowed has led to serious maintenance problems. I seriously considered the possibility of allowing resumption when I designed the C exception handling mechanism and this issue was discussed in quite some detail during standardization. See the exception handling chapter of The Design and Evolution of C . If you want to check to see if you can fix a problem before throwing an exception, call a function that checks and then throws only if the problem can not be dealt with locally. A new_handler is an example of this. I On A Somewhat Different Style As Part of A Discussion in a newsgroup. Why does not C have an equivalent to realloc ()? If you want to, you can of course use realloc (). However, realloc () is only guaranteed to work on arrays allocated by malloc () (and similar functions) containing objects without user-defined copy constructors. Also, please remember that contrary to naive expectations, realloc () occationally does copy its argument array.In C , a better way of dealing with reallocation is to use a standard library container, such as vector, And let it grow naturally. How do I use eXceptions? See THE C Programming Language Section 8.3, Chapter 14, And Appendix E. The Appendix Focuses on Techniques for Writing Exception-Safe Code in Demanding Applications, and is not write for novices. IN C , Exceptions Is Used to Signal Erros That Cannot Be Handled Locally, Such As The Failure TO Acquire A Resource In A Constructor. For Example: Class vector { Int Sz; INT * ELEM; Class Range_ERROR {}; PUBLIC: VECTOR (INT S): SZ (s) {IF (SZ <0) throw range_ERROR (); / * ... * /} // ... } Do not use exceptions as simply another way to return a value from a function Most users assume -. As the language definition encourages them to - that exception-handling code is error-handling code, and implementations are optimized to reflect that assumption. A KEY TECHNIQUE IS Resource Acquisiton Is Initialization (Sometimes Abbreviated to Raii), Which Uses Classes with destructors to impose ORDER ON Resource Management. For example: Void FCT (String S) { File_Handle F (s, "r"); // file_handle's constructor Opens the file called "s" // USE F } // Here File_Handle's Destructor Closes The File If the "use f" part of fct () throws an exception, the destructor is still thrown and the file is proPerly closed. This Contrasts to the commit (const char * s) { File * f = fopen (s, "r"); // Open the file named "s" // USE F Fclose (f); // Close the file } If The "USE F" Part of Old_fct throws an exception - or simply does a return - the file isn't closed. In c Programs, longjmp () is an affitional hazard. Why can't i assign a vector Class Apple: Public Fruit {Void Apple_FCT (); / * ... * /}; Class orange: public fruit {/ * ... * /}; // Orange Doesn't Have Apple_FCT () Vector Void F (Vector { vf.push_back (new orange); // add orange to Vector of Fruit } Void h () { f (v); // error: cannot pass a vector For (int i = 0; i } Had The Call F (v) Been Legal, We Well Have Had an Orange Pretending to Be An Apple. An alternative language design decision would have been to allow the unsafe conversion, but rely on dynamic checking. That would have required a run-time check for each access to v's members, and h () would have had to throw an exception upon encountering the Last Element OF V. Why Doesn't C Have A Universal Class Object? We do not need one: generic programming provides statically type safe alternatives in most cases Other cases are handled using multiple inheritance There is no useful universal class:... A truly universal carries no semantics of its own A "universal" class encourages sloppy . thinking about types and interfaces and leads to excess run-time checking Using a universal base class implies cost: Objects must be heap-allocated to be polymorphic; that implies memory and access cost heap objects do not naturally support copy semantics heap.. objects do not support simple scoped behavior (which complicates resource management). A universal base class encourages use of dynamic_cast and other run-time checking. Yes. I have simplified the arguments, but this is a FAQ, not an academic paper.How Do I read a string from input? You can read A Single, Whitespace Terminated Word Like this: #include #include Using namespace std; int main () { COUT << "" please enter a word: / n "; String S; CIN >> S; Cout << "you entered" << s << '/ n'; } Note That there is no expendit memory management and no fixed-size buffer this you could Possibly overflow. If you really need a whole line (and not just a single word) You can do this: #include #include Using namespace std; int main () { Cout << "please enter a line: / n"; String S; GetLine (CIN, S); Cout << "you entered" << s << '/ n'; } For a brief Introduction To Standard Library Facilities, Such As Iostream and String, See Chaper 3 of TC PL3 (Available Online). For a Detailed Comparison Of Simple Uses of C and C I / O, See "Learning Standard C As a New Language", Which You Can Download from MyPublications List Why does not C provide a "finally" construct Because C supports an alternative that is almost always better:?. The "resource acquisition is initialization" technique (TC PL3 section 14.4) The basic idea is to represent a resource by a Local Object, So That The Local Object's Destructor Will Release The Resource. That Way, The Programmer Cannot ForgeGet To Release The Resource. For Example: Class file_handle { File * p; PUBLIC: FILE_HANDLE (const char * n, const char * a) {P = fopen (n, a); if (p == 0) throw open_ERROR (Errno); FILE_HANDE (file * pp) {P = PP; if (p == 0) throw open_error (errno); ~ File_handle () {fclose (p); Operator file * () {returnit p; // ... } Void f (const char * fn) { File_Handle F (FN, "RW"); // Open Fn forreading and Writing // USE File Through F } In a system, we need a "resource handle" class for each resource. However, we do not have to have an "finally" clause for each acquisition of a resource. In realistic systems, there are far more resource acquisitions than kinds of . Also, Have A Look at The Examples of Resource Management in Appendix E of the C Programming Language. What is an auto_ptr and why is not there an auto_array? An auto_ptr is an example of very simple handle class, defined in #include Using namespace std; Struct x { Int m; // .. } Void f () { Auto_ptr X * q = new x; P-> m ; // use p Just Like a Pointer Q-> m ; // ... Delete q; } IF An Exception Is Thrown in The ... Part, The Object Held by P Is Correctly Deleted by Auto_Ptr's Destructor While The x Pointed to by q IS Leaked. See Tc Pl 14.4.2 for Details. Auto_ptr is a very lightweight class In particular, it is * not * a reference counted pointer If you "copy" one auto_ptr into another, the assigned to auto_ptr holds the pointer and the assigned auto_ptr holds 0. For example..: #include #include Using namespace std; Struct x { Int m; // .. } int main () { Auto_ptr Auto_PTR COUT << "P" << p.get () << "q" << q.get () << "/ n"; } Should Print A 0 Pointer Followed by a Non-0 Pointer. for example: P 0x0 q 0x378d0 Auto_ptr :: get () returns the help pointer. ... This "move semantics" differs from the usual "copy semantics", and can be surprising In particular, never use an auto_ptr as a member of a standard container The standard containers require the usual copy semantics For example: Std :: Vector An Auto_Ptr Holds a Pointer To An Indiviual Element, Not a Pointer To An Array: Void F (INT N) { Auto_Ptr // ... } THIS An Error Because The Destructor Will Delete The Pointer Using "DELETE" Rather Than "delete []" AND will fail to invoke the destructor for the last n-1 xs. SO SHOULD WE USE AUTO_ARRAY to HOLD ARRAYS? NO. The REAON IS THATO_ARRAY. The Reason is this. A Better Solution Is To Use a Vector: Void f (int N) { Vector // ... } Should An Exception Occur in The ... Part, V'S Destructor Will Be Correctly Invoked. CAN I MIX C-STYLE AND C Style Allocation and deallocation? Yes, in The Sense That You Can Use Malloc () and new in the same program. NOR CAN You Allocate with new and delete with free () or use realloc () on an array allocated by new. The C operators new and delete guarantee proper construction and destruction; where constructors or destructors need to be invoked, they are The C-style functions malloc (), calloc (), free (), and realloc () does not ensure that. . Furthermore, there is no guarantee that the mechanism use by new and delete to acquire and release raw memory is compatible with malloc () and free () If mixing styles works on your system, you were simply "lucky" -. for now. IF you feel the need for realoc () - And Many Do - The Consider Using A Standard Library Vector. For Example // read Words from Input Into a Vector of Strings: Vector String S; While (CIN >> S && S! = ".") Words.push_back (s); The vector expands as needed. See Also The Examples and Discussion in "Learning Standard C As a New Language", Which You Can Download from My Publications List. Why Must I Use a cast to control from void *? IN C, You CAN Implicitly Convert a void * to a t *. This is unsafe. Consider: #include int main () { CHAR i = 0; Char j = 0; Char * p = & i; Void * Q = P; INT * PP = Q; / * Unsafe, Legal C, Not C * / Printf ("% D% D / N", I, J); * PP = -1; / * Overwrite Memory Starting AT & I * / Printf ("% D% D / N", I, J); } The Effects of Using A T * That Doesn't Point To A T CAN Be Disastrous. Consequently, In C , To Get A T * from a void * You NEED AN Explicit Cast. For Example, TO Get The Undesirable Effects of the Program Above, you since t w w w: INT * PP = (int *) q; OR, USING A New Style Cast to make the unchecked Type Conversion Operation More Visible: INT * PP = static_cast Casts are best avoided. One of the MOST Common Uses of this unsafe conversion in c is to assign the result of malloc () to a suitable pointer. For example: INT * P = Malloc (sizeof (int)); IN C , Use the TypeSafe New Operator: INT * P = new int; Incidentally, The New Operator Offers Additional Advantages over Malloc (): New Can't Accidentally, New Implicitly Checks for Memory Exhaustion, and New Provides for Initialization for Example: Typedef std :: complex / * C style: * / CMPLX * P = (CMPLX *) Malloc (sizeof (int)); / * error: wrong size * / / * forgot to test for p == 0 * / IF (* p == 7) {/ * ... * /} / *Oops: Forgot to Initialize * p * / // C style: CMPLX * Q = New CMPLX (1, 2); // Will Throw Bad_alloc if Memory Is Exhaustedif (* q == 7) {/ * ... * /} How do I define an in-class constant? If You Want a constant this you can use in a constant expression, Say As an Array Bound, you have two Class x { Static const INT C1 = 7; Enum {c2 = 19}; CHAR V1 [C1]; CHAR V2 [C2]; // ... } At first glance, the declaration of c1 seems cleaner, but note that to use that in-class initialization syntax, the constant must be a static const of integral or enumeration type initialized by a constant expression That's quite restrictive.: Class y { Const Int C3 = 7; // error: not static Static Int C4 = 7; // error: not const Static const float c5 = 7; // error NOT INTEGRAL } I TEND TO Use The "Enum Trick" Because It's Portable and Doesn't Tempt Me To Use Non-Standard Extensions of The in-Class Initialization Syntax. So why do these inconvenient restrictions exist? A class is typically declared in a header file and a header file is typically included into many translation units. However, to avoid complicated linker rules, C requires that every object has a unique definition. That rule would Be Broken IF C Allowed In-Class Definition of Entities That Needed to Be Stored In Memory As Objects. See D & E for an Explanation Of C 's Design TradeOffs. You Have More Flexibility If The Const Isn't Needed for Use in a constant expression: Class z { Static char * p; // initialize in definition Const Int I; // Initialize In Constructor PUBLIC: Z (INT II): i (ii) {} } Char * z :: p = "Hello, There"; You can take the address of a static member if (and only if) it has an out-of-class definition: class ae { // ... PUBLIC: Static const INT C6 = 7; Static const INT C7 = 31; } Const Int ae :: C7; // definition INT f () { Const int * p1 = & ae :: c6; // error: c6 not an Lvalue Const Int * P2 = & ae :: C7; // OK // ... } Why doesn't delete Zero out its operand? Consider Delete P; // ... Delete P; If the ... part does not touch p then the second "delete p;". Is a serious error that a C implementation can not effectively protect itself against (without unusual precautions) Since deleting a zero pointer is harmless by definition, a simple Solution Would Be for "Delete P;" To do a "p = 0;" After IT HAS DONE wherever Else is Required. However, C Doesn't Gurantee That. One Reason Is That The Operand of Delete Need NOT BE An Lvalue. Consider: DELETE P 1; Delete f (x); Here, the implementation of delete does not have a pointer to which it can assign zero. These examples may be rare, but they do imply that it is not possible to guarantee that `` any pointer a to deleted object is 0. '' A Simpler Way of Bypassing That `` rule '' Is To Have Two Pointers to an Object: T * p = new t; T * Q = P; Delete P; Delete q; // ooch! C Explicitly Allows An Implement Of Delete To Zero Out An Lvalue Operand, and i Had Hoped That Implementations Would Do That, But That Idea Doesn't Seem To Have Become Popular with iMplementers. If You Consider Zeroing Out Pointers Important, Consider Using A Destroy Function: Template Consider this yet-another reason to minimize explicit use of new and delete by relying on stadard library containers, handles, etc.Note that passing the pointer as a reference (to allow the pointer to be zero'd out) has the added benefit of Preventing Destroy () from Being Called for An Rvalue: INT * f (); INT * P; // ... DESTROY (f ()); // error: Trying to pass an rvalue by non-const reference Destroy (p 1); // Error: Trying to pass An rvalue by non-const revance Why isn't the destructor called at the end of scope? The Simple Answer IS "of Course It Is!" Void f () { X * p = new x; // USE P } That IS, There Was Some (Mistaken) Assumption That The Object Created by "New" Would Be Destroyed At The End of a Function. Basically, You Should Only Use "New" if you want an object to live beyond the lifetime of the scope you create it in. That Done, you need to use "delete" to destroy it. For example: X * g (int i) {/ * ... * / return new x (i);} // the x output The call of g () Void H (INT i) { X * p = g (i); // ... Delete P; } If you want an object to live in a scope only, don't use "new" But Simply Define a variable: { ClassName X; // USE X } The Variable Is Implicitly Destroyed at the end of the scope. Code That Creates An Object Using New and The Deletes It at The End of The Same Scope IS Ugly, Error-Prone, And INEfficient. For example: Void fct () // ugly, error-prone, and inefficient { X * p = new x; // USE P Delete P; } CAN I WRITE "Void Main ()"? The definition Void main () {/ * ... * /} IS NOT AND NEVER HAS BEEN C. See The ISO C Standard 3.6.1 [2] or The ISO C Standard 5.1.2.2.1. A Conforming Implementation AcceptsInT main () {/ * ... * /} and INT Main (int Argc, char * argv []) {/ * ... * /} A conforming implementation may provide more versions of main (), but they must all have return type int. The int returned by main () is a way for a program to return a value to "the system" that invokes it. On systems that does not provide such a facility the return value is ignored, but that does not make "void main ()" legal C or legal C. Even if your compiler accepts "void main ()" avoid it, or risk being considered ignorant BY C and C Programmers. IN C , main () NEED NOT Contain An Explicit Return Statement. In That Case, The Value Returned IS 0, Meaning Successful Execution. For example: #include int main () { Std :: Cout << "This Program Returns The Integer Value 0 / N"; } Note Also That Neither ISO C Nor c99 allows you to Leave the Type Out of a Declaration. That IS, IN Contrast to C89 and ARM C , "Int" is not assumed WHERE A TYPE IS MISSING IN A DECLATION. Consequently: #include Main () {/ * ... * /} Is An Error Because The Return Type of Main () is Missing. Why can't i overload dot, ::, sizeof, etc .?most Operators can be overloaded by a programmer. The Exceptions Are (DOT) ::?: SIZEOF THERE IS NO FUNDAMENTAL REASON TO DISALLOW OVERLOADING OF?:. I Just Didn't See The Need To Introduce The Special Case Of Overloading a Ternary Operator. Note That A Function Overloading EXPR1? EXPR2: EXPR3 Would Not Be Able To Guarantee That Only ONE onPer2 and EXPR3 WAS EXECUTED. Sizeof Cannot Be overloaded Because Built-in Operations, Such As Incrementing a Pointer Into An Array Implicitly Depends on It. Consider: X a [10]; X * p = & a [3]; X * Q = & a [3]; P ; // p points to a [4] // thus the integer value of p Must Be // sizeof (x) larger Than the integer value of q Thus, Sizeof (x) Could Not Be Given A New and Different Meaning by The Programmer Without Violating Basic Language Rules. In N :: m neither N nor m are expressions with values; N and m are names known to the compiler and :: performs a (compile time) scope resolution rather than an expression evaluation One could imagine allowing overloading of x :: y. where x is an object rather than a namespace or a class, but that would - contrary to first appearences - involve introducing new syntax (to allow expr :: expr) It is not obvious what benefits such a complication would bring.. . Operator (dot) could in principle be overloaded using the same technique as used for -> However, doing so can lead to questions about whether an operation is meant for the object overloading or an object referred to by For example.:.. Class y { PUBLIC: Void f (); // ... } Class X {// Assume That You can overload. Y * p; Y & operator. () {Return * p;} Void f (); // ... } Void G (X & X) { X.f (); // x :: f or y :: f or error? } This Problem Can Be Solved in Several Ways. At The Time of Standardization, IT WAS NOT OBVIOS WAY WOULD BEST. For More Details, SEE D & E. How do i communication an integer to a string? The Simplest Way Is To Use A StringStream: #include #include #include String Itos (INT I) // Convert int to string { Stringstream S; s << i; Return S.STR (); } int main () { INT i = 127; String ss = ITOS (i); Const char * p = ss.c_str (); COUT << SS << "" << p << "/ n"; } Naturally, this Technique Works for Converting Any Type That You Can Output Using << To a string. For a description of string streams, see 21.5.3 of OF STREAMS, SEE 21.5.3 of THE C Programming Language. HOW Do I CALL A C FUNCTION FROM C ? Just Declare The C Function `` Extern "C" '' (in Your C Code) AND Call IT (from your C or C code). For example: // c code EXTERN "C" void f (int); // one Way EXTERN "C" {// Another Way INT G (double); Double h (); } Void Code (INT I, DOUBLE D) { f (i); INT II = g (d); Double DD = H (); // ... } The Definitions of the Functions MAY LOOK LIKE THIS: / * C code: * / Void f (INT i) { / * ... * / } INT G (Double D) { / * ... * / } Double H () { / * ... * / } Note That C Type Rules, Not C Rules, Are Used. So you can't call function declared `` Extern "C" '' with the wrong number of argument. For example: // c code Void more_code (int i, double d) { Double DD = H (i, d); // error: unExpected Arguments // ... } HOW DO I CALL A C Function from C? Just Declare The C Function `` EXTERN "C" '' (in Your C Code) and call it (from your C or C code). For example: // c code: EXTERN "C" void f (int); Void f (INT i) { // ... } Now f () can be used like this: / * C code: * / Void f (int); Void CC (INT i) { f (i); / * ... * / } Naturally, this Works Only for Non-Member Functions. If You Want To Call MEMBER FUNCTIONS (INCL. Virtual Functions) from C, You NEED TO Provide A Simple Wrapper. For example: // c code: Class C { // ... Virtual Double F (char); } Extern "C" Double Call_c_f (c * p, int i) // wrapper function { Return P-> f (i); } Now c :: f () can be used like this: / * C code: * / Double Call_c_f (Struct C * P, INT I); Void CCC (Struct C * P, INT i) { Double D = CALL_C_F (P, I); / * ... * / } If You Want to Call Overloaded Functions from C, You Must Provide Wrappers with Distinct Names for the c code to use. For example: // c code: Void f (int); Void f (double); Extern "C" Void F_i (INT I) {f (i);} Extern "c" void f_d (double d) {f (d);} Now the f () Functions Can Be Used Like this: / * C code: * / Void f_i (int); Void f_d (double); Void CCCC (INT I, DOUBLE D) { F_i (i); f_d (d); / * ... * / } Note That Sestniques Can Be Used to Call A C Library from C C code Even if You Cannot (or Do Not Want to) Modify The C Headers. IS `` Int * p; '' Right or IS `` Int * P; '' Right? Both Are "Right" in The Sense That Both Are Valid C and C and Both Have Exactly the Same Meaning. As far as the language Definitions and the compilers are concerned we could just as well. `` Int * p; '' or `` Int * p; '' The choice between `` int * p; '' and `` int * p; '' is not about right and wrong, but about style and emphasis C emphasized expressions;.. Declarations were often considered little more than a necessary evil C , On The Other Hand, Has A Heavy Emphasis On Types.a `Typical C Programmer '' Writes` `Int * P; '' And Explains It` `* P Is What Is The Int '' EMPHASIZING SYNTAX, AND Y Point To THE C (AND C ) Declaration grammar to argue for the correctness of the style. Indeed, The * binds to the name p in the grammar. A `` Typical C Programmer '' Writes `` Int * P; '' and explains it `` P IS a Pointer to an int '' Emphasizing Type. Indeed The Type of P Is Int *. I Clearly Prefer That Emphasis and See IT As Important for Using The More Advanced Parts Of C Well. The critical confusion comes (ONLY) WHEN People Try to Declare Several Pointers with a Single Declaration: INT * P, P1; // Probable error: p1 is not an int * Placing the * closer to the name does not make this kind of error significantly less like. INT * P, P1; / / PROBABLE ERROR? Declaring One name per declaration minimizes the probem - in Particular When We Initialize The Variables. People Arefi Likely to Write: INT * P = & I; INT P1 = p; // error: int initialized by int * And if the do, the compiler will company. Whenever something can be done in two ways, someone will be confused. Whenever something is a matter of taste, discussions can drag on forever. Stick to one pointer per declaration and always initialize variables and the source of confusion disappears. See The Design and Evolution of C for a longer discussion of the C declaration syntax.Which layout style is the best for my code? Such style issues are a matter of personal taste. Often, opinions about code layout are strongly held, but probably consistency matters more than any particular Style. Like Most People, I'd Have a Hard Time Construction A Solid Logical Argument for My Preferences. I Personally use what is offen called "k & r" style. When You Add Conventions for Constructs Not Found In C, That Becomes What Is Sometimes Called "Stroustrup" style. For example: Class C: public b { PUBLIC: // ... } Void f (int * p, int max) { IF (p) { // ... } For (INT I = 0; i // ... } } This style conserves vertical space better than most layout styles, and I like to fit as much as is reasonable onto a screen. Placing the opening brace of a function on a new line helps me distinguish function definition from class definitions at a glance. Indentation is very important. Design issues, such as the use of abstract classes for major interfaces, use of templates to present flexible type-safe abstractions, and proper use of exceptions to represent errors, are far more important than the choice of layout style. SHOULD I PUT "const" before or after the Type? I put it before, but what's a matter of taste. "Const T" and "t const" WERE ALWAYS (BOTH) Allowed and Equivalent. For Example: Const int a = 1; // okint const b = 2; // Also OK My Guess Is That Using The First Version Will Confuse Fewer Programmers (`` is more idiomatic ''). Why? When I invented "const" (initially named "readonly" and had a corresponding "writeonly"), I allowed it to go before or after the type because I could do so without ambiguity. Pre-standard C and C imposed few ( IF any) ORDERING RULES ON Specifiers. I don't recolved discussions about the order at the time. A few of the early user - notably me - simply liked the Look of Const Int C = 10; Better Than INT const c = 10; At the time. I May Have Been Influenced by The Fact That My Earliest Examples WERE WRITTEN Using "Readonly" and Readonly INT C = 10; Does Read Better Than INT Readonly C = 10; The Earliest (C or C ) code using "const" APPEARS to have been create (by me) by a global substitude of "const" for "readonly". I Remember Discussing Syntax Alternatives with Several People - Incl. Dennis Ritchie - But I Don n't Remember Which Languages i Looked At Then. Note That In Const Pointers, "const" always comes after the "*". For example: INT * const p1 = q; // constant Pointer to int Variable INT const * p2 = q; // Pointer to constant int Const Int * P3 = Q; // Pointer to Constant Int What good is static_cast? Casts are generally best avoided. With the exception of dynamic_cast, their use implies the possibility of a type error or the truncation of a numeric value. Even an innocent-looking cast can become a serious problem if, during development or MAINTENANCE, ONE OF THE THEPES INVOLVED IS CHANGED. FOR EXAMPLE, What does this mean?: x = (t) y; We don't know. It depends on the Type T and the Types of x and y. T Could Be The name of a class, a typedef, or maybe a template parameter. Maybe x and y area scalar variables and (t) represents a value conversion. Maybe x is of a class derived from y's class and (T) is a downcast. Maybe x and y are unrelated pointer types. Because the C-style cast (T) can be used to express many logically different operations, the compiler has only the barest chance to catch misuses. For the same reason, a programmer may not know exactly what a cast does. This is sometimes considered an advantage by novice programmers and is a source of subtle errors when the novice guessed wrong. The "new-style casts" with the body to give programs More Clearly and for the compiler to catch more errors. For example: INT A = 7; Double * p1 = (double *) & a; // ok (But a doesn't point to a double) Double * p2 = static_cast Double * p2 = reinterpret_cast Const Int C = 7; INT * Q1 = & C; // Error INT * Q2 = (int *) & c; // ok (but * q2 = 2; is still invalid code and may fail) INT * Q3 = static_cast int * q4 = const_cast A secondary reason for introducing the new-style cast was that C-style casts are very hard to spot in a program. For example, you can not conveniently search for casts using an ordinary editor or word processor. This near-invisibility of C -style casts is especially unfortunate because they are so potentially damaging. An ugly operation should have an ugly syntactic form. That observation was part of the reason for chosing the syntax for the new-style casts. A further reason was for the new-style . Maybe, Because Static_Cast Is So Ugly and So Relatively Hard to Type, You're More Likely to Think Twice Before Using One? That Wouldrn C . So, what's wrong with using macros? Macros do not obey the C scope and type rules. This is often the cause of subtle and not-so-subtle problems. Consequently, C provides alternatives that fit better with the rest of C , such as Inline functions, templates, and namespaces. CONSIDER: #include "someheader.h" Struct s { Int alpha; Int Beta; } If Someone (Unwisely) HAS Written A Macro Called "Beta" This May Not Compile OR (Worse) Compile Into Something UnExpected. For Example, "SomeHeader.h" May Contain: #define alpha 'a' #define beta b [2] Conventions such as having macros (and only macros) in ALLCAPS helps, but there is no language-level protection against macros For example, the fact that the member names were in the scope of the struct did not help:. Macros operate on a program as a stream of characters before the compiler proper sees it This, incidentally, is a major reason for C and C program development environments and tools have been unsophisticated:. the human and the compiler see different things. Unfortunately, you can not assume that other programmers consistently avoid what you consider "really stupid" For example, someone recently reported to me that they had encountered a macro containing a goto I have seen that also and heard arguments that might -.. In a weak Moment - Appear to make sense. for example: #define prefix get_ready (); int RET__ #define return (i) RET __ = i; do_something (); goto exit #define suffix exit: cleanup (); Return Ret__ Void f () { Prefix; // ... Return (10); // ... Return (x ); // ... SUFFIX; } Imagine Being Presented with That As a Maintenance Programmer; "Hiding" The Macros in A Header - As IS NOT UNCOMMON - MAKES THIS KIND OF "MAGIC" HARDER TO SPOT. One of the MOST Common Subtle Problems Is That A Function-Style Macro Doesn't Obey The Rules of Function Argument Passing. For example: #define Square (x) (x * x) Void F (Double D, INT i) { Square (d); //fine Square (i ); // Ouch: Means (i * i ) Square (D 1); // OUCH: Means (D 1 * D 1); That IS, (D D 1) // ... } THE "D 1" Problem is Solved by Adding Parentheses in the "call" or in the macro definition: #DEfine Square (x) ((x) * (x)) / * better * / Double Evaluation of i Remains. And yes, I do know that there are things known as macros that does not suffer the problems of C / C preprocessor macros. However, I have no ambitions for improving C macros. Instead, I recommend the use of facilities from the C language Proper, Such As Inline functions, Templates, Constructors (for Initialization), Destructors (For Cleanup), Exceptions (for Exitation Contexts), ETC. How do you pronounce "cout"? "Cout" is pronounced "see-out". The "c" Stands for "Character" Because Iostreams Map Values To and from Byte (char) Representations. How do you pronounce "char"? "Char" is usually pronounced "tchar", not "kar". This may seem illogical because "character" is pronounced "ka-rak-ter", but nobody ever accused English pronounciation and spelling of Being Logical.
g () (new s); // Explicitly Transfer Responsibility for deletring this s q = g ();