The following points are to apply for all C programmers. The reason why they say they are because these points mentioned are that you usually can't find in the C book or on the website. Such as: pointing to members, this is where many materials are not willing to mention, and it is also a place where it is wrong, or even some advanced C programmers. The main point here is not only how to explain how to write better code, but more is something inside the language rules. Obviously, they are permanent in C programmers. I believe this article will make you harvest.
First, I will return some questions from different levels of C programmers. I am surprised to find a lot of experienced programmers, I haven't realized that Hymbols should also appear in the standard header file.
Key 1:
Many C programmers are still using
Because these substantially different, you cannot confuse these two libraries in a program. As a habit, in the new code, it is generally
Point 2: Places that should be noted when using reference to pass parameters
When passing parameters with reference, it is best to declare the reference to a const type. The advantage of this is: telling the procedure that this parameter cannot be modified. In this example, the function f () is the passed reference: Void F (Const Int & I); int Main () {f (2); / * ok * /} This program passes a parameter 2 to f () . At runtime, C creates a temporary variable of the Int type value of 2, and delivers its reference to f (). This temporary variable and its reference from F () called start creation and exist until the function returns. When you return it, you will be removed immediately. Note that if we don't add a Const qualified word before reference, the function f () may change the value of its parameters, which is more likely to make the program unexpected behavior. So don't forget const.
This point is also applicable to user-defined objects. You can add a reference to the temporary object if it is const type: struct a {}; void f (const same & a); int main () {f (a ()); // OK, passed a temporary A CONST reference}
Key 3: "Command Separation" Expression Form
The "comma separation" expression is inherited from C, in the For- and While-cycles. Of course, this grammar rule is considered not intuitive. First, let's take a look at what is "comma separation" expression form.
An expression consists of one or more other expressions, separated by a comma, such as: if ( x, --y, cin.good ()) // Three expressions this IF condition contains three comma Separated expression. C will calculate each expression, but the result of the complete "comma separation" expression is the value of the rightmost expression. Therefore, only when cin.good () returns True, the value of the IF condition is TRUE. Here is another example: int J = 10; INT i = 0; while ( i, --j) {// until j = 0, the loop ends, when cycles, I continuously} points 4, Use the constructor of the global object to call the function before the program starts
Some applications need to call other functions before starting the main program. For example: the transmissive process function, the registration function function must be called before the actual program is running. The easiest way is to call these functions through a constructor of a global object. Because global objects are constructed before the main program begins, these functions will return results before main (). Such as: class logger {public: logger () {activate_log (); // Translator Note: Call the function}}} you need to run in the constructor; Logger log; // A global instance int main () {record * PREC = read_log (); // Translator Note: Read the log file data // .. Program code} The global object log is constructed before the main () runs, and the log calls the function activate_log (). Therefore, when main () is executed, it can read data from the log file.
There is no doubt that memory management in C programming is the most complicated and easier to show bugs. Direct access to raw memory, dynamic allocation storage, and maximum playing C instruction efficiency, making you try to avoid BUG related to memory. Point 5: Avoid pointers with a pointing function using complex constructors
The pointer to the function is one of the worstable syntax of C . Can you tell me what the following statement is? Void (* p [10]); P is a "10 pointers constituted by 10 pointers to a returning Void type and points to another array of functions without returning and arithmetic functions." This troublesome syntax is really difficult to identify, isn't it? You can actually declare the functionality equivalent to the above statement via TypedEf. First, use the typedef declaration "Pointer": typedef void (* pfv) (): typedef void (* pfv) (); then, declared "another function pointer to no return and using PFV": typedef void (* pf_taking_pfv (PFV); Now, a array of pointers composed of 10 tops: PF_TAKING_PFV P [10]; with Void (* p [10]) (void (*)) reaches the same effect. But this is not more readable!
Key 6: Pointing to the member
A class has two basic members: functions members and data members. Similarly, there are two pointers that point to members: pointing to the pointer of the function member and pointer to the data member. It is not commonly used because the class generally does not contain public data members, which is only used when the coordination structure and class (Class) are used when inheriting CLASS.
The pointer to the member is one of the most difficult constructs of the C syntax, but this is also a C most powerful feature. It allows you to call a class of function members without having to know the name of this function. This very agile calling tool. Similarly, you can also check and change this data without knowing its members' name by using a pointer to the data member. Pointing to data members Although the grammar of the pointer pointing to the member will make you a little confused, but you will soon find that it is actually similar to ordinary pointers, but it is the * number :: symbol And the name of the class, an example: Define a pointer pointing to the int * Pi; defines a data member pointing to the INT type class: int A :: * PMI; // PMI is an INT type of class A Members you can initialize it: class a {public: int Num; int x;}; int A :: * PMI = & a :: Num; the above code is a NUM member that pointing a INT type pointing to class A and Initialize it to this NUM member's address. By adding * you can use and change the value of the NUM member of Category A: A A1, A2; INT N = A1. * PMI; // Put a1.Num Assign the value to Na1. * PMI = 5; // Assign 5 assignments to the A1.Num A2. * PMI = 6; // Put 6 assignment to 6A2.NUM If you define a pointer to class A, then above Must use -> * operator instead: a * pa = new a; int N = pa-> * PMI; Pa -> * PMI = 5;
Pointer to function member
It consists of the data type returned by the function member, and the class name followed by :: symbol, pointer name, and function's parameter list. For an example: a pointer to a function member of class A (this function returns an int type): Class a {public: int func ();}; int (a :: * pmf) (); the above definition is to say PMF is a pointer to the function member of the class A. In fact, this pointer and a pointer to a normal pointing function have no difference, just it contains the name of the class and :: symbol. You can call this function in any * PMF (): PMF = & a :: func; a a; (a. * Pmf) (); // call a.func () If you define one first Pointer to the object, then the above operation is used to use -> * instead: a * pa = & a; (pa-> * pmf) (); // Call PA-> FUNC () pointing to the pointer to the function member to consider polymorphism Sex. So, this call will be dynamically recycled when you call a virtual function member through a pointer. Another place to pay attention to, you can't take a class constructor and the address of the destructor. Point 7 to avoid memory fragmentation
There will often be such a situation: your application will leak the memory every time you run once due to the defects of the program, and you repeat your programs during the period, and you can imagine that it will also Crash the system. But how can I prevent it? First, try less dynamic memory. In most cases, you may use static or automatic storage or STL containers. Second, try to allocate a large block of memory rather than only a small amount of memory. For example: allocate the memory required by an array instance, not a memory of an array element at a time. Total 8, is DELETE or delete []
There is an absurd statement in the programmer: use Delete instead of delete [] to delete an array type! For example: int * p = new int [10]; delete p; // error, it should be: delete [] p The above program is completely wrong. In fact, using delete instead of delete [] on a platform may not cause the system to crash, but that is purely luck. You can't guarantee that your app is compiled on another compiler, run on another platform, so please use delete []. Point 9. Optimized members' arrangement of a class can be changed in the following manner: struct a {bool a; int b; bool c;}; // sizeof (a) == 12 on my computer SIZEOF (a) Equal to 12. This result may surprise you because the total number of members of A is 6 bytes: 1 4 1 byte. Where is the other 6 bytes? The compiler is inserted behind each BOOL member to ensure that each member is arranged in 4 bytes for the boundary. You can reduce the size of A, through the following: struct b {bool a; bool c; int b;}; // sizeof (b) == 8 this time, the compiler only inserts 2 bytes after the member C . Because B accounts for 4 bytes, it is naturally arranged as a word in the form of a word, and the size of A and C 1 1 = 2, plus 2 bytes just press two words. Form arrangement B.
Total 10, why inherited a class without a false argument function is dangerous?
A class without a false aromacy means that it cannot be used as a base class. Such as std :: string, std :: intrlex, and std :: vector are all like this. Why inherited a class without a false argument function? When you have inheritively created a related class inherited from the base class, pointing to the pointer in the new class object and the reference actually points to the origin of the object. Because the destructor is not a virtual function, when you delete such a class, C will not call the destructor chain. For example: Class a {public: ~ a () // is not a virtual function {// ...}}; class b: public a // fault; A no false argument function {public: ~ b () {// ...}}; int main () {a * p = new b; // looks right delete p; // fault, b's sectors are not called}
Point 11, category nested by friends class
When you declare a nesting class with a friend class, the friend declared behind the nested category declaration, not the front. Class A {Private: INT I; Public: Class B // Nested Category Declaration in the Former {Public: B (A & A) {Ai = 0;};}; Friend Class B; // Friendly Class Declaration} If you put the friend class statement in front of the declaration nested class, the compiler will abandon other statements after the friend class.