Definitely the first! You use vector for its automatic memory management. Using a raw pointer to a vector means you don't get automatic memory mana Interesting thing is when I run the same binary on the same hardware, To have a useful example for the object class I selected the Particle class which can simulate some physical interactions and implements a basic Euler method: The Particle class holds 72 bytes, and theres also some extra array for our further tests (commented out for now). It shows how much more expensive it is to sort a vector of large objects that are stored by value, than it is when they're stored by pointer [3]. For 1000 particles we need on the average 2000 cache line reads! So, why it is so important to care about iterating over continuous block of memory? slightly different data: For all our tests the variance is severely affected, its clearly It might be easier to visualize if you decompose that statement to the equivalent 2 lines: To actually remove the pointer from the vector, you need to say so: This would remove the pointer from the array (also shifting all things past that index). Copyright 2023 www.appsloveworld.com. You can read more in a separate blog post: Custom Deleters for C++ Smart Pointers. 0}. In general you may want to look into iterators when using containers. With the Celero Yes, you created a memory leak by that. There are more ways to create a std::span. Idea 4. Vector of 20,000 small objects vs vector of 20,000 object pointers to 20,000 heap objects. When I run Celero binary in I've prepared a valuable bonus if you're interested in Modern C++! Not consenting or withdrawing consent, may adversely affect certain features and functions. In C++, a variable is the variable that it is representing. Does it need to stay sorted? All Rights Reserved. C++, Member function returning const reference to vector containing pointers to const objects, Vector of pointers to member functions with multiple objects c++, Vector of objects containing references or pointers. 10k. << Notes on C++ SFINAE, Modern C++ and C++20 Concepts, Revisiting An Old Benchmark - Vector of objects or pointers. Accessing the objects takes a performance hit. Any other important details? You must also ask yourself if the Objects or the Object* are unique. the measurement happens: Additionally I got the test where the randomization part is skipped. So it might make sense that entities and projectiles store pointers, so they actually point at the same objects. Containers of pointers let you avoid the slicing problem. The vector wouldn't have the right values for the objects. Usually solution 1 is what you want since its the simplest in C++: you dont have to take care of managing the memory, C++ does all that for you ( The C-array (1), std::vector(2), and the std::array (3) have int's. 2011-2022, Bartlomiej Filipek my tests using 10k particles, 1k updates I got the following output: The great thing about Nonius is that you dont have to specify number of we might create a bit more advanced scenarios for our benchmarks. You just need to Heres the code for a vector of unique_ptr, the code is almost the same for a vector of shared_ptr. * Z Score. data for benchmarks. Springbrooks Cirrus is a true cloud financial platform built for local government agency needs. Note about C++11: In C++11 shared_ptr became part of the standard as std::shared_ptr, so Boost is no longer required for this approach. Heres the corresponding graph (this time I am using mean value of of 2023 ITCodar.com. Create a variable and insert a value in it. But in a general case, the control block might lay in a different place, thats why the shared pointer holds two pointers: one to the object and the other one to the control block. To provide the best experiences, we and our partners use technologies like cookies to store and/or access device information. Thank you! For a Plain Old Data (POD) type, a vector of that type is always more efficient than a vector of pointers to that type at least until sizeof(POD) > sizeof(POD*). github/fenbf/benchmarkLibsTest. Why is this? Your vector still contains an old pointer, which has became invalid by the time the object was deleted. A view does not own data, and it's time to copy, move, assignment it's constant. Larger objects will take more time to copy, as well as complex or compound objects. A vector of pointers takes performance hits because of the double dereferencing, but doesn't incur extra performance hits when copying because pointers are a consistent size. However, you can choose to make such a Capitalize First letter of each word in a String in Java | Camel Case, C++11 Multithreading Part 1 : Three Different ways to Create Threads, C++11 Move Contsructor & rvalue References, Different ways to iterate over a set in C++, How to trim strings in C++ using Boost String Algorithm Library, How to add an element in Vector using vector::push_back, Using std::find & std::find_if with User Defined Classes, Pandas Dataframe: Get minimum values in rows or columns & their index position. That is, the elements the vector manages are the pointers, not the pointed objects. C++, Source code available on githib: Obviously there is very good locality of access to both arrays. This can lead to a huge problem in long-running applications or resource-constrained hardware environments. It seems that you have already subscribed to this list. affected by outliers. The benchmarks was solely done from scratch and theyve used only - default constructor, copy constructors, assignment, etc.) Dynamic Polymorphism and Dynamic Memory Allocation. There are two global variables that you probably have used, but let them be the only ones: std::cin & std::cout. "Does the call to delete affect the pointer in the vector?". Are there any valid use cases to use new and delete, raw pointers or c-style arrays with modern C++? All rights reserved. It can be done using 2 steps: Square brackets are used to declare fixed size. The size of std::vector is fixed, because it essentially just contains a pointer to the real data that is dynamically allocated. A typical implementation consists of a pointer to its first element and a size. write a benchmark that is repeatable. Further, thanks to the functions std::erase and std::erase_if, the deletion of the elements of a container works like a charm. How can I point to a member of a std::set in such a way that I can tell if the element has been removed? Contracts did not make it into C++20. Safety and Robustness are also more important. For example, if the difference between the worst performing data structure and the best is 10 nanoseconds, that means that you will need to perform at least 1E+6 times in order for the savings to be significant. C++ - Performance of vector of pointer to objects, vs performance of objects, Leaked Mock Objects when using GoogleMock together with Boost::Shared Pointers, C++: Operator overloading of < for pointers to objects. Larger objects will take more time to copy, as well as complex or compound objects. Consenting to these technologies will allow us to process data such as browsing behavior or unique IDs on this site. This contiguous memory can be a plain array, a pointer with a size, a std::array, a std::vector, or a std::string. So, to replace a thread object in vector, we first need to join the existing object and then replace it with new one i.e. With Nonius I have to write 10 benchmarks separately. If you really need to store resources that have to be allocated by new, then you should use boost::shared_ptr. The technical storage or access that is used exclusively for statistical purposes. It all depends on what exactly you're trying to do. 3. 1. In the generated CSV there are more data than you could see in the I've recently released a new book on Modern C++: runs generate method - so that we have some random numbers assigned. You still need to do the delete yourself as, again, the vector is only managing the pointer, not the YourType. But CPUs are quite smart and will additionally use a thing called Hardware Prefetcher. There are many convenience functions to refer to the elements of the span. How to delete objects from vector of pointers to object? The technical storage or access is necessary for the legitimate purpose of storing preferences that are not requested by the subscriber or user. Also, you probably don't need a pointer to a vector in the first place, but I won't judge you since I don't know your situation. The algorithmstd::iota fills myVec with thesequentially increasing values, starting with 0. vArray is nullptr (represented as X), while vCapacity and vSize are 0. If you need to store objects of multiple polymorphic types in the same vector, you must store pointers in order to avoid slicing. Full repository can be found here: github/fenbf/PointerAccessTest but the code is also tested with Quick Bench: Theres also experimental code at https://github.com/fenbf/benchmarkLibsTest where I wrote the same benchmark with a different library: Celero, Google Benchmark, Nonius or Hayai (and see the corresponding blog post: Revisiting An Old Benchmark - Vector of objects or pointers). * Experiment, WebYou use a vector of pointers when you need a heterogeneous container of polymorphic objects, or your objects need to persist against operations performed on the vector, for Notice that only the first 8 bytes from the second load are used for the first particle. looks at gender info then creates vector of objects, also sets the name and age for each match with the help of pointer. WebVector of Objects A vector of Objects has first, initial performance hit. The code will suffer from a memory leak if the programmer does not free up the memory before exiting. I try to write complete and accurate articles, but the web-site will not be liable for any errors, omissions, or delays in this information or any losses, injuries, or damages arising from its display or use. 2. std::vector obs1; char * * obs2; Effectively, obs1 Your success with Springbrook software is my first priority., 1000 SW Broadway, Suite 1900, Portland, OR 97205 United States, Cloud financial platform for local government, Payment Solutions: Integrated with Utility Billing, Payment Solutions agency savings calculator, Springbrook Survey Shows Many Government Employees Still Teleworking, Springbrook Software Announces Strongest Third Quarter in Companys 35-year History Powered by New Cirrus Cloud Platform, Springbrook Debuts New Mobile App for Field Work Orders, Springbrook Software Releases New Government Budgeting Tool, GovTech: Springbrook Software Buys Property Tax Firm Publiq for ERP, Less training for new hires through an intuitive design, Ease of adoption for existing Springbrook users, Streamlined navigationwithjust a few simple clicks. How to Switch Between Blas Libraries Without Recompiling Program, Weird Behavior of Right Shift Operator (1 >> 32), How to Compile Qt 5 Under Windows or Linux, 32 or 64 Bit, Static or Dynamic on Visual Studio or G++, What Is Shared_Ptr's Aliasing Constructor For, Why Istream Object Can Be Used as a Bool Expression, Reading from Ifstream Won't Read Whitespace, Using Qsocketnotifier to Select on a Char Device, What Is the Easiest Way to Parse an Ini File in C++, Does Vector::Erase() on a Vector of Object Pointers Destroy the Object Itself, Is Adding to a "Char *" Pointer Ub, When It Doesn't Actually Point to a Char Array, What Is the Purpose of Using -Pedantic in the Gcc/G++ Compiler, How Can My C/C++ Application Determine If the Root User Is Executing the Command, Returning Temporary Object and Binding to Const Reference, Is 'Long' Guaranteed to Be at Least 32 Bits, Does "Const" Just Mean Read-Only or Something More, How to Force a Static Member to Be Initialized, What Does the "Lock" Instruction Mean in X86 Assembly, Why Isn't 'Int Pow(Int Base, Int Exponent)' in the Standard C++ Libraries, About Us | Contact Us | Privacy Policy | Free Tutorials. when working with a vector of pointers versus a vector of value types. Press J to jump to the feed. Now lets create a std::function<> object that we will pass to thread object as thread function i.e. The safest version is to have copies in the vector, but has performance hits depending on the size of the object and the frequency of reallocating the reserved memory area. We can also push std::thread without specifically specifying std::move(), if we pass them as rvalue i.e. Back in main the data type receives this vector pointer by a necessary data type. When I run Sometimes you want a vector of objects, sometimes you want a vector of pointers to objects, and sometimes you want something else entirely. Make your choice! WebStore pointers to your objects in a vectorinstead But if you do, dont forget to deletethe objects that are pointed to, because the vectorwont do it for you. 2k 10k without writing code separately. You can modify the entire span or only a subspan. So they not only read the data but also perform a copy (when the algorithm decides to swap items or move to a correct place according to the order). particles example I just wanted to test with 1k particles, 2k. C++ Core Guidelines: Type Erasure with Templates, C++ Core Guidelines: Rules for Templates and Generic Programming, C++ Core Guidelines: Rules for Constants and Immutability, The new pdf bundle is ready: C++ Core Guidelines - Concurrency and Parallelism, I'm Proud to Present: Modern C++ Concurrency is available as interactive course, C++ Core Guidelines: Rules about Exception Handling, C++ Core Guidelines: The noexcept Specifier and Operator, C++ Core Guidelines: A Short Detour to Contracts in C++20, C++ Core Guidelines: Rules for Error Handling, C++ Core Guidelines: The Remaining Rules about Lock-Free Programming, C++ Core Guidelines: The Resolution of the Riddle, C++ Core Guidelines: Concurrency and lock-free Programming, The Update of my Book "Concurreny with Modern C++", C++ Core Guidelines: Be Aware of the Traps of Condition Variables, C++ Core Guidelines: More Traps in the Concurrency, C++ Core Guidelines: Taking Care of your Child Thread, C++ Core Guidelines: Sharing Data between Threads, C++ Core Guidelines: Use Tools to Validate your Concurrent Code, C++ Core Guidelines: More Rules about Concurrency and Parallelism, C++ Core Guidelines: Rules for Concurrency and Parallelism, The new pdf bundle is ready: Functional Features in C++, C++ Core Guidelines: The Remaining Rules about Performance, C++ Core Guidelines: More Rules about Performance, The Truth about "Raw Pointers Removed from C++", No New New: Raw Pointers Removed from C++, C++ Core Guidelines: Rules about Performance, C++ Core Guidelines: Rules about Statements and Arithmetic, C++ Core Guidelines: More about Control Structures, C++ Core Guidelines: To Switch or not to Switch, that is the Question, C++ Core Guidelines: Rules for Statements, C++ Core Guidelines: Rules for Conversions and Casts, C++ Core Guidelines: More Rules for Expressions, C++ Core Guidelines: Rules for Expressions, C++ Core Guidelines: More Rules for Declarations, C++ Core Guidelines: Declarations and Initialisations, C++ Core Guidelines: Rules for Expressions and Statements, C++ Core Guidelines: Passing Smart Pointers, C++ Core Guidelines: Rules for Smart Pointers, The new pdf bundle is available: Embedded - Performance Matters, C++ Core Guidelines: Rules for Allocating and Deallocating, C++ Core Guidelines: Rules about Resource Management, C++ Core Guidelines: Rules for Enumerations, C++ Core Guidelines: More Rules for Overloading, C++ Core Guidelines: Rules for Overloading and Overload Operators, The C++ Standard Library: The Second Edition includes C++17, C++ Core Guidelines: Accessing Objects in a Hierarchy, C++ Core Guidelines: The Remaining Rules about Class Hierarchies, The new pdf bundle is available: Functional Programming with C++17 and C++20, C++ Core Guidelines: More Rules about Class Hierarchies, C++ Core Guidelines: Function Objects and Lambdas, C++ Core Guidelines: Comparison, Swap, and Hash, C++ Core Guidelines: Rules for Copy and Move, My open C++ Seminars in the First Half of 2018, I Proudly present my Book is Ready "Concurrency with Modern C++", C++ Core Guidelines: The Rule of Zero, Five, or Six, C++ Core Guidelines: Semantic of Function Parameters and Return Values, C++ Core Guidelines: The Rules for in, out, in-out, consume, and forward Function Parameter, "Concurrency with Modern C++" is 95% complete; Including all Source Files, C++ Core Guidelines: Function Definitions, C++ Core Guideline: The Guideline Support Library, My Book "Concurrency with Modern C++" is 75% complete, My Book "Concurrency with Modern C++" is 50% complete, Get the Current Pdf Bundle: "Multithreading: The High-Level Interface", My Book "Concurrency with Modern C++" is 30% complete. This time each element is a pointer to a memory block allocated in a possibly different place in RAM. Persistent Mapped Buffers, Benchmark Results. Why inbuilt sort is not able to sort map of vectors? In the declaration: vector v; the word vector represents the object's base type. * Standard Deviation Insert the address of the variable inside the vector. Boost MultiIndex - objects or pointers (and how to use them?)? So, as usual, its best to measure and measure. Can I be sure a vector contains objects and not pointers to objects? Will you spend more time looping through it than adding elements to it? C++: Vector of objects vs. vector of pointers to new objects? I remember during an assignment for a class I took during fall semester that we had to use vectors of pointers instead of just the objects. These are all my posts to then ranges library: category ranges library. First, let's create a synthetic "large" object that has well defined ordering properties by some numeric ID: struct SomeLargeData { SomeLargeData ( int id_) : id (id_) {} int id; int arr [ 100 ]; }; The real truth can be found by profiling the code. This is 78% more cache line reads than the first case! we can not copy them, only move them. KVS and SoftRight customers now have the ability to upgrade to Springbrooks new Cirrus cloud platform: Since you are explicitly stating you want to improve your C++, I am going to recommend you start using Boost. span1 references the std::vector vec(1). Why is RTTI needed for non-polymorphic typeid? Course: Modern C++ Concurrency in Practice, Course: C++ Standard Library including C++14 & C++17, Course: Embedded Programming with Modern C++, Course: C++ Fundamentals for Professionals, Interactive Course: The All-in-One Guide to C++20, Subscribe to the newsletter (+ pdf bundle), std::span in C++20: Bounds-Safe Views for Sequences of Objects, Automatically deduces the size of a contiguous sequence of objects, Create a std::span from a pointer and a size, Design Patterns and Architectural Patterns with C++, Clean Code: Best Practices fr modernes C++. Standard containers, like std::vector, containing raw pointers DO NOT automatically delete the things that the pointers are pointing at, when removing the pointers from the containers. However, to pass a vector there are two ways to do so: Pass By value. All data and information provided on this site is for informational purposes only. Just to recall we try to compare the following cases: Additionally, we need to take into account address randomization. WebIn that case, when you push_back(something), a copy is made of the object. Operations with the data structures may need to be performed a huge amount of times in order for the savings to be significant. C++ : Is it bad practice to use a static container in a class to contain pointers to all its objects for ease of access? The raw pointers must be deleted before the vector can be destructed; or a memory leak is created. You may remember that a std::span is sometimes called a view.Don't confuse a std::span with a view from the ranges library (C++20) or a std::string_view (C++17). Such benchmark code will be executed twice: once during the You haven't provided nearly enough information. Two cache line reads. Is there any advantage to putting headers in an "include" subdir of the project? If you don't use pointers, then it is a copy of the object you pass in that gets put on the vector. A pointer to a vector is very rarely useful - a vector is cheap to construct and destruct. For elements in the vector , there's no correct ans In C++ we can declare vector pointers using 3 methods: Using std::vector container Using [ ] notations Using the new keyword (Dynamic Memory) 1. Now, as std::thread objects are move only i.e. An unsafe program will consume more of your time fixing issues than a safe and robust version. Uups this time we cannot use data loaded in the second cache line read (from the first step), because the second particle data is located somewhere else in the memory! Class members that are objects - Pointers or not? Yes, it is possible - benchmark it. Dynamic dispatch (virtual method calls) work only on pointers and references (and you can't store references in a std::vector). When an object is added to the vector, it makes a copy. In other words, for each particle, we will need 1.125 cache line reads. https://en.cppreference.com/w/cpp/container/span/operator_at states that operator[] is undefined behaviour on out of bounds access. For example, a std::string and std::vector can be created at modified at compile-time. This method will be memory-bound as all operations inside are too simple. If the objects are in dynamic memory, the memory must be initialized first (allocated). When a vector is passed to a function, a copy of the vector is created. I think it has something to do with push_back and the capacity of the vector and if the capacity is reached a new vector that uses new contiguous addresses that don't contain the right objects is created. Free the pointer (Remove address from variable). As vector contains various thread objects, so when this vector object is destructed it will call destructor of all the thread objects in the vector. Most of the time its better to have objects in a single memory block. In In Re Man. Which pdf bundle should I provide? Each benchmark will be executed 20 times (20 A-143, 9th Floor, Sovereign Corporate Tower, We use cookies to ensure you have the best browsing experience on our website. Container of references / non-nullable pointers, Avoiding preprocessor for mutual exclusive function call in C++20, How Iostream file is located in computer by c++ code during execution, Get text from a button in an application using win32 C++ and hooks. Copying a pointer into a vector is not dependent on the object size. Consenting to these technologies will allow us and our partners to process personal data such as browsing behavior or unique IDs on this site. Not consenting or withdrawing consent, may adversely affect certain features and functions. A Computer Science portal for geeks. C++: Defined my own assignment operator for my type, now .sort() wont work on vectors of my type? get even more flexibility and benchmarks can be executed over different How to approach copying objects with smart pointers as class attributes? c++ How to find the minimum number of elements from a vector that sum to a given number, Passing a 2d dynamic array to a function in C++. * Min (us) However its also good to remember that when the object inside a container is heavy it might be better to leave them in the same place, but use some kind of indexing when you sort or perform other algorithms that move elements around. C++ has several container types defined for you in the standard library: Yes, I've read it, but as far as I understand, the only data structures that are appropriate for this is. It contains well written, well thought and well explained computer science and programming articles, quizzes and practice/competitive programming/company interview Questions. There are: In the second step, we have already 56 bytes of the second particle, so we need another load - 64 bytes - to get the rest. Vector of pointers are vectors that can hold multiple pointers. Memory leaks; Shallow copies; Memory Leaks It's not unusual to put a pointer into a standard library container. How to use find algorithm with a vector of pointers to objects in c++? The The Winner is: Multithreading: The high-level Interface. I think it would be interesting the discussion and I would like , Jake, GS, Lawton Shoemake, Animus24, Jozo Leko, John Breland. As you can see we can even use it for algorithms that uses two As you can see this time, we can see the opposite effect. To make polymorphism work You have to use some kind of pointers. In my seminar, I often hear the question: How can I safely pass a plain array to a function? Parameters (none) Return value Pointer to the underlying element storage. If all you care about is the objects, then they are more or less equivalent; you just have an extra level of indirection. This time we also get some data of the third particle. A view (std::span) and a std::string_view are non-owning views and can deal with strings. How to use boost lambda to populate a vector of pointers with new objects, C++ vector of objects vs. vector of pointers to objects. comparator for sorting a vector contatining pointers to objects of custom class, GDB & C++: Printing vector of pointers to objects. In C++, should different game entities have different classes? Why can't `auto&` bind to a volatile rvalue expression? :) Check out the Boost documentation. Some objects are cheaper to construct/copy contruct/move construct/copy/move/destruct than others, regardless of size. Particles vector of pointers: mean is 121ms and variance is not I've read it, but I didn't find an answer as to which one is faster. The test code will take each element of the problem Does vector::erase() on a vector of object pointers destroy the object itself? https://www.youtube.com/watch?v=YQs6IC-vgmo, https://www.youtube.com/watch?v=WDIkqP4JbkE, Performance of container of objects vs performance of container of pointers. Pass By Reference. It doesn't affect the pointer. This can help you with your problem in three different ways: Using a shared_ptr could declare your vector like this: This would give you polymorphism and would be used just like it was a normal vector of pointers, but the shared_ptr would do the memory-management for you, destroying the object when the last shared_ptr referencing it is destroyed. Overloading, variadic functions and bool type, Unable to discriminate template specialization with enable_if and is_base_of. What's special about R and L in the C++ preprocessor? Same as #2, but first sort Using Looking for Proofreaders for my new Book: Concurrency with Modern C++, C++17: Improved Associative Containers and Uniform Container Access, C++17: New Parallel Algorithms of the Standard Template Library, Get the Current Pdf Bundle: Concurrency with C++17 and C++20, C++17 - Avoid Copying with std::string_view, C++17- More Details about the Core Language, And the Winners are: The C++ Memory Model/Das C++ Speichermodell, I'm Done - Geschafft: Words about the Future of my Blogs, Parallel Algorithms of the Standard Template Library, Recursion, List Manipulation, and Lazy Evaluation, Functional in C++11 and C++14: Dispatch Table and Generic Lambdas, Object-Oriented, Generic, and Functional Programming, Memory Pool Allocators by Jonathan Mller, Pros and Cons of the various Memory Allocation Strategies, Copy versus Move Semantics: A few Numbers, Automatic Memory Management of the STL Containers, Memory and Performance Overhead of Smart Pointers, Associative Containers - A simple Performance Comparison, Published at Leanpub: The C++ Standard Library, I'm proud to present: The C++ Standard Library, My Conclusion: Summation of a Vector in three Variants, Multithreaded: Summation with Minimal Synchronization, Thread-Safe Initialization of a Singleton, Ongoing Optimization: Relaxed Semantic with CppMem, Ongoing Optimization: A Data Race with CppMem, Ongoing Optimization: Acquire-Release Semantic with CppMem, Ongoing Optimization: Sequential Consistency with CppMem, Ongoing Optimization: Locks and Volatile with CppMem, Ongoing Optimization: Unsynchronized Access with CppMem, Looking for Proofreaders for my New C++ Book, Acquire-Release Semantic - The typical Misunderstanding. http://info.prelert.com/blog/stl-container-memory-usage, http://en.cppreference.com/w/cpp/container. So both vectors will manage their pointers, but you have to think of how the lifecycle of those two pointers (the one from entities and the one from projectiles) interact with the object itself. can be as inexpensive as a POD's or arbitrarily more expensive. To fully understand why we have such performance discrepancies, we need to talk about memory latency. If any of the destructed thread object is joinable and not joined then std::terminate() will be called from its destructor.Therefore its necessary to join all the joinable threads in vector before vector is destructed i.e. For example, we can try std::variant against regular runtime polymorphism. Using vectors of pointers #include #include using namespace std; static const int NUM_OBJECTS = 10; These seminars are only meant to give you a first orientation. Nonius are easy to use and can pick strange artefacts in the results C++ Core Guidelines: More Non-Rules and Myths, More Rules about the Regular Expression Library, C++ Core Guidelines: Improved Performance with Iostreams, Stuff you should know about In- and Output with Streams, More special Friends with std::map and std::unordered_map, C++ Core Guidelines: std::array and std::vector are your Friends, C++ Core Guidelines: The Standard Library, C++ Core Guidelines: The Remaining Rules about Source Files, The new pdf bundle is available: C++ Core Guidlines - Templates and Generic Programming, Types-, Non-Types, and Templates as Template Parameters, C++ Core Guidelines: Surprise included with the Specialisation of Function Templates, C++ Core Guidelines: Other Template Rules, C++ Core Guidelines: Programming at Compile Time with constexpr, C++ Core Guidelines: Programming at Compile Time with Type-Traits (The Second), C++ Core Guidelines: Programming at Compile Time with the Type-Traits, C++ Core Guidelines: Programming at Compile Time, C++ Core Guidelines: Rules for Template Metaprogramming, C++ Core Guidelines: Rules for Variadic Templates, C++ Core Guidelines: Rules for Templates and Hierarchies, C++ Core Guidelines: Ordering of User-Defined Types, C++ Core Guidelines: Template Definitions, C++ Core Guidelines: Surprises with Argument-Dependent Lookup, C++ Core Guidelines: Regular and SemiRegular Types, C++ Core Guidelines: Pass Function Objects as Operations, I'm Proud to Present: The C++ Standard Library including C++14 & C++17, C++ Core Guidelines: Definition of Concepts, the Second, C++ Core Guidelines: Rules for the Definition of Concepts, C++ Core Guidelines: Rules for the Usage of Concepts.