C++ function inlining
1. Replace macro code with inlining
The C++ language supports function inlining, whose purpose is to improve the execution efficiency (speed) of the function.
In C programs, macro codes can be used to improve execution efficiency. The macro code itself is not a function, but it behaves like a function. The preprocessor replaces function calls by copying macro codes, eliminating the need for pushing parameters onto the stack, generating CALL calls in assembly language, returning parameters, and executing return, thus improving speed. The biggest disadvantage of using macro code is that it is error-prone. The preprocessor often produces unexpected side effects when copying macro code.
For example the
#define MAX(a, b) (a) > (b) ? (a) : (b)
statement
result = MAX(i, j) + 2 ;
will be interpreted by the preprocessor as
result = (i) > (j) ? (i) : (j) + 2 ;
due to the operator ' + 'Than operator' :' has a higher priority, so the above statement is not equivalent to the expected
result = ( (i) > (j) ? (i) : (j) ) + 2 ;
If you rewrite the macro code as
#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
, you can solve the problem caused by priority mistake. But even using modified macro code is not foolproof, for example the statement
result = MAX(i++, j);
will be interpreted by the preprocessor as
result = (i++) > (j) ? (i++) : (j);
For C++, there is another disadvantage of using macro code : Private data members of the class cannot be operated, which means that the macro code is basically for public or global operations.
#Let’s see how C++’s “function inlining” works. For any inline function, the compiler places the function's declaration (including name, parameter types, and return value type) in the symbol table. If the compiler finds no error in the inline function, the code for the function is also placed in the symbol table. When calling an inline function, the compiler first checks whether the call is correct (performing type safety checks, or performing automatic type conversion, of course the same for all functions). If correct, the code for the inline function will directly replace the function call, thus eliminating the overhead of the function call. This process is significantly different from preprocessing, because the preprocessor cannot perform type safety checks or perform automatic type conversions. If the inline function is a member function, the address of the object (this) will be placed in the appropriate place, which is something that the preprocessor cannot do.
#The function inlining mechanism of C++ language not only has the efficiency of macro code, but also increases security, and can freely operate the data members of the class. Therefore, in C++ programs, all macro codes should be replaced by inline functions. "assert" is probably the only exception. assert is a macro that only works in the Debug version. It is used to check for situations that "shouldn't" happen. In order not to cause differences between the Debug and Release versions of the program, assert should not have any side effects. If assert is a function, since the function call will cause changes in memory and code, there will be a difference between the Debug version and the Release version. So assert is not a function, but a macro.
2. Programming style of inline function
The keyword inline must be placed together with the function definition body. To make a function inline, simply putting inline in front of the function declaration has no effect. The function Foo in the following style cannot become an inline function:
inline void Foo(int x, int y); // inline 仅与函数声明放在一起,不起任何作用 void Foo(int x, int y) { … }
And the function Foo in the following style becomes an inline function:
void Foo(int x, int y); inline void Foo(int x, int y) // inline 与函数定义体放在一起 { … }
So, inline It is a "keyword for implementation" rather than a "keyword for declaration". Generally, users can read the declaration of a function, but cannot see the definition of the function. Although the inline keyword is added before the declaration and definition body of inline functions in most textbooks, I think inline should not appear in the declaration of a function. Although this detail will not affect the functionality of the function, it reflects a basic principle of high-quality C++/C programming style: declaration and definition cannot be confused, and users do not need to, and should not, know whether a function needs to be inlined.
Member functions defined in the class declaration will automatically become inline functions, for example
class A { public: void Foo(int x, int y) { … } // 自动地成为内联函数 }
will Although placing the definition body of the member function in the class declaration brings convenience in writing, it is not a good programming style. The above example should be changed to:
// 头文件 class A { public: void Foo(int x, int y); } // 定义文件 inline void A::Foo(int x, int y) { … }
3. Use inlining with caution
Inlining can improve the execution efficiency of functions. Why not define all functions as inline functions?
#If all functions are inline functions, is the keyword "inline" still needed?
Inline is at the expense of code expansion (duplication), and only saves the overhead of function calls, thus improving the execution efficiency of the function. If the time to execute the code in the function body is larger than the overhead of function calls, then the efficiency gains will be very small. On the other hand, each inline function call requires copying the code, which will increase the total code size of the program and
consume more memory space. It is not appropriate to use inlining in the following situations:
(1) If the code in the function body is relatively long, using inlining will result in higher memory consumption costs.
# (2) If a loop occurs in the function body, the time to execute the code in the function body is greater than the cost of the function call.
The constructors and destructors of classes are easily misunderstood as using inline is more efficient. Beware of constructors and destructors. Functions may hide some behavior, such as "secretly" executing the constructors and destructors of base classes or member objects. So don't just put the definitions of constructors and destructors in the class declaration.
The above is the content of C++ function inlining. For more related content, please pay attention to the PHP Chinese website (www.php.cn)!

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics



The steps to implement the strategy pattern in C++ are as follows: define the strategy interface and declare the methods that need to be executed. Create specific strategy classes, implement the interface respectively and provide different algorithms. Use a context class to hold a reference to a concrete strategy class and perform operations through it.

Golang and C++ are garbage collected and manual memory management programming languages respectively, with different syntax and type systems. Golang implements concurrent programming through Goroutine, and C++ implements it through threads. Golang memory management is simple, and C++ has stronger performance. In practical cases, Golang code is simpler and C++ has obvious performance advantages.

Nested exception handling is implemented in C++ through nested try-catch blocks, allowing new exceptions to be raised within the exception handler. The nested try-catch steps are as follows: 1. The outer try-catch block handles all exceptions, including those thrown by the inner exception handler. 2. The inner try-catch block handles specific types of exceptions, and if an out-of-scope exception occurs, control is given to the external exception handler.

To iterate over an STL container, you can use the container's begin() and end() functions to get the iterator range: Vector: Use a for loop to iterate over the iterator range. Linked list: Use the next() member function to traverse the elements of the linked list. Mapping: Get the key-value iterator and use a for loop to traverse it.

C++ template inheritance allows template-derived classes to reuse the code and functionality of the base class template, which is suitable for creating classes with the same core logic but different specific behaviors. The template inheritance syntax is: templateclassDerived:publicBase{}. Example: templateclassBase{};templateclassDerived:publicBase{};. Practical case: Created the derived class Derived, inherited the counting function of the base class Base, and added the printCount method to print the current count.

Causes and solutions for errors when using PECL to install extensions in Docker environment When using Docker environment, we often encounter some headaches...

In C, the char type is used in strings: 1. Store a single character; 2. Use an array to represent a string and end with a null terminator; 3. Operate through a string operation function; 4. Read or output a string from the keyboard.

In multi-threaded C++, exception handling is implemented through the std::promise and std::future mechanisms: use the promise object to record the exception in the thread that throws the exception. Use a future object to check for exceptions in the thread that receives the exception. Practical cases show how to use promises and futures to catch and handle exceptions in different threads.
