Think Parallel - Bryce Adelstein Lelbach - Meeting C++ online
Meeting C++ online did kick off the year with a sponsored event by NVIDIA with Bryce Adelstein Lelbach presenting his talk Think Parallel:
Think Parallel - Bryce Adelstein Lelbach
March 19-21, Madrid, Spain
April 1-4, Bristol, UK
June 16-21, Sofia, Bulgaria
By Meeting C++ | Jan 19, 2025 09:18 AM | Tags: meetingcpp community c++20 basics advanced
Meeting C++ online did kick off the year with a sponsored event by NVIDIA with Bryce Adelstein Lelbach presenting his talk Think Parallel:
Think Parallel - Bryce Adelstein Lelbach
By Blog Staff | Jan 15, 2025 11:15 AM | Tags: None
This article explores how C++20 concepts can replace CRTP when implementing static interfaces for a family of classes. By leveraging concepts, we achieve cleaner, more readable, and less error-prone code—provided you have access to C++20.
Replace CRTP with Concepts
by Sandor Dargo
From the article:
In my Meeting C++ 2024 trip report, among my favourite ideas I mentioned Klaus Iglberger’s talk where he mentioned the possibility of replacing the curiously returning template pattern with the help of class tagging and concepts.
Class tagging might mean different things in different contexts, or at least might be implemented in different ways. The end goal is to mark, in other words, tag classes or functions to be used in certain contexts, with certain algorithms. As you’ll see, in our case it’ll also be a tool to prevent duck typing.
We are going to see an example implementation of a static interface with CRTP with a couple of different derived classes, then we’ll see the implementation without CRTP.
The CRTP solution
With the static interface, we are creating a static family of types. There is no need for dynamic polymorphism to share the same interface. It’s still granted through a base class, which is a template taking the deriving class as a parameter.
By Administrator | Jan 13, 2025 04:52 PM | Tags: None
The 2025-01 mailing of new standards papers is now available.
WG21 Number | Title | Author | Document Date | Mailing Date | Previous Version | Subgroup |
---|---|---|---|---|---|---|
P0149R1 | Generalised member pointers | Jeff Snyder | 2025-01-13 | 2025-01 | P0149R0 | EWG Evolution |
P0260R14 | C++ Concurrent Queues | Detlef Vollmann | 2025-01-12 | 2025-01 | P0260R13 | SG1 Concurrency and Parallelism,LEWG Library Evolution |
P0876R19 | fiber_context - fibers without scheduler | Oliver Kowalke | 2025-01-13 | 2025-01 | P0876R18 | EWG Evolution,CWG Core,LWG Library |
P1030R8 | std::filesystem::path_view | Niall Douglas | 2024-12-21 | 2025-01 | P1030R7 | LEWG Library Evolution |
P1839R7 | Accessing object representations | Brian Bi | 2025-01-11 | 2025-01 | P1839R6 | CWG Core |
P2079R6 | System execution context | Lucian Radu Teodorescu | 2025-01-13 | 2025-01 | P2079R5 | SG1 Concurrency and Parallelism,LEWG Library Evolution |
P2414R5 | Pointer lifetime-end zap proposed solutions | Paul E. McKenney | 2025-01-11 | 2025-01 | P2414R4 | SG1 Concurrency and Parallelism,LEWG Library Evolution,LWG Library |
P2434R3 | Nondeterministic pointer provenance | S. Davis Herring | 2025-01-13 | 2025-01 | P2434R2 | SG22 Compatibility,EWG Evolution,CWG Core |
P2654R1 | Modules and Macros | Alisdair Meredith | 2025-01-13 | 2025-01 | P2654R0 | EWG Evolution,LEWG Library Evolution |
P2663R6 | Proposal to support interleaved complex values in std::simd | Daniel Towner | 2025-01-13 | 2025-01 | P2663R5 | LEWG Library Evolution |
P2664R9 | Proposal to extend std::simd with permutation API | Daniel Towner | 2025-01-13 | 2025-01 | P2664R8 | SG1 Concurrency and Parallelism,LEWG Library Evolution |
P2688R5 | Pattern Matching: `match` Expression | Michael Park | 2025-01-13 | 2025-01 | P2688R4 | EWG Evolution |
P2719R2 | Type-aware allocation and deallocation functions | Louis Dionne | 2025-01-06 | 2025-01 | P2719R1 | EWG Evolution,CWG Core |
P2719R3 | Type-aware allocation and deallocation functions | Louis Dionne | 2025-01-08 | 2025-01 | P2719R2 | EWG Evolution,CWG Core |
P2746R7 | Deprecate and Replace Fenv Rounding Modes | Hans Boehm | 2025-01-12 | 2025-01 | P2746R6 | SG6 Numerics,LEWG Library Evolution |
P2758R4 | Emitting messages at compile time | Barry Revzin | 2025-01-07 | 2025-01 | P2758R3 | LEWG Library Evolution,CWG Core |
P2806R3 | do expressions | Barry Revzin | 2025-01-12 | 2025-01 | P2806R2 | EWG Evolution |
P2825R4 | Overload resolution hook: declcall( unevaluated-call-expression ) | Gašper Ažman | 2025-01-13 | 2025-01 | P2825R3 | EWG Evolution,CWG Core |
P2830R8 | Standardized Constexpr Type Ordering | Gašper Ažman | 2025-01-09 | 2025-01 | P2830R7 | LWG Library |
P2830R9 | Standardized Constexpr Type Ordering | Gašper Ažman | 2025-01-10 | 2025-01 | P2830R8 | EWG Evolution,LEWG Library Evolution,CWG Core,LWG Library |
P2841R6 | Concept and variable-template template-parameters | Corentin Jabot | 2025-01-13 | 2025-01 | P2841R5 | CWG Core |
P2843R1 | Preprocessing is never undefined | Alisdair Meredith | 2025-01-13 | 2025-01 | P2843R0 | SG22 Compatibility,EWG Evolution |
P2883R1 | `offsetof` Should Be A Keyword In C++26 | Alisdair Meredith | 2025-01-10 | 2025-01 | P2883R0 | EWGI SG17: EWG Incubator,EWG Evolution |
P2899R0 | Contracts for C++ - Rationale | Timur Doumler | 2025-01-13 | 2025-01 | SG21 Contracts,EWG Evolution,LEWG Library Evolution | |
P2900R13 | Contracts for C++ | Joshua Berne | 2025-01-13 | 2025-01 | P2900R12 | CWG Core,LWG Library |
P2933R3 | std::simd overloads for <bit> header | Daniel Towner | 2025-01-13 | 2025-01 | P2933R2 | LWG Library |
P2952R2 | auto& operator=(X&&) = default | Arthur O'Dwyer | 2025-01-08 | 2025-01 | P2952R1 | CWG Core |
P2953R1 | Forbid defaulting operator=(X&&) && | Arthur O'Dwyer | 2025-01-08 | 2025-01 | P2953R0 | EWGI SG17: EWG Incubator |
P2971R3 | Implication for C++ | Walter E Brown | 2025-01-12 | 2025-01 | P2971R2 | EWG Evolution,LEWG Library Evolution |
P2988R9 | std::optional<T&> | Steve Downey | 2025-01-13 | 2025-01 | P2988R8 | LEWG Library Evolution,LWG Library |
P2996R9 | Reflection for C++26 | Barry Revzin | 2025-01-13 | 2025-01 | P2996R8 | EWG Evolution |
P3019R12 | Vocabulary Types for Composite Class Design | Jonathan Coe | 2025-01-12 | 2025-01 | P3019R11 | LEWG Library Evolution,LWG Library |
P3045R5 | Quantities and units library | Mateusz Pusz | 2025-01-13 | 2025-01 | P3045R4 | SG6 Numerics,SG16 Unicode,LEWGI SG18: LEWG Incubator,LEWG Library Evolution |
P3070R2 | Formatting enums | Victor Zverovich | 2025-01-13 | 2025-01 | P3070R1 | LEWG Library Evolution |
P3081R1 | Core safety profiles for C++26 | Herb Sutter | 2025-01-06 | 2025-01 | P3081R0 | EWG Evolution,LEWG Library Evolution |
P3086R3 | Proxy: A Pointer-Semantics-Based Polymorphism Library | Mingxin Wang | 2025-01-13 | 2025-01 | P3086R2 | LEWGI SG18: LEWG Incubator,LEWG Library Evolution |
P3094R6 | std::basic_fixed_string | Mateusz Pusz | 2025-01-10 | 2025-01 | P3094R5 | LEWG Library Evolution |
P3111R3 | Atomic Reduction Operations | Gonzalo Brito Gadeschi | 2025-01-13 | 2025-01 | P3111R2 | EWG Evolution,LEWG Library Evolution |
P3125R3 | constexpr pointer tagging | Hana Dusíková | 2025-01-13 | 2025-01 | P3125R2 | LEWG Library Evolution |
P3139R1 | Pointer cast for unique_ptr | Zhihao Yuan | 2024-12-27 | 2025-01 | P3139R0 | LEWG Library Evolution |
P3148R1 | Formatting of chrono Time Values | Alan Talbot | 2025-01-13 | 2025-01 | P3148R0 | LEWG Library Evolution |
P3164R3 | Early Diagnostics for Sender Expressions | Eric Niebler | 2025-01-09 | 2025-01 | P3164R2 | LEWG Library Evolution |
P3176R1 | The Oxford variadic comma | Jan Schultke | 2024-12-18 | 2025-01 | P3176R0 | CWG Core |
P3179R5 | C++ parallel range algorithms | Ruslan Arutyunyan | 2025-01-13 | 2025-01 | P3179R4 | LEWG Library Evolution |
P3206R0 | A sender query for completion behaviour | Maikel Nadolski | 2025-01-13 | 2025-01 | SG1 Concurrency and Parallelism | |
P3229R0 | Making erroneous behaviour compatible with Contracts | Timur Doumler | 2025-01-13 | 2025-01 | SG21 Contracts,EWG Evolution,LEWG Library Evolution | |
P3289R1 | Consteval blocks | Daveed Vandevoorde | 2025-01-12 | 2025-01 | P3289R0 | EWG Evolution,CWG Core |
P3347R1 | Invalid/Prospective Pointer Operations | Paul E. McKenney | 2025-01-11 | 2025-01 | P3347R0 | EWG Evolution |
P3348R2 | C++26 should refer to C23 not C17 | Jonathan Wakely | 2025-01-13 | 2025-01 | P3348R1 | SG6 Numerics,LEWG Library Evolution |
P3351R2 | views::scan | Yihe Li | 2025-01-12 | 2025-01 | P3351R1 | SG9 Ranges |
P3367R3 | constexpr coroutines | Hana Dusíková | 2025-01-06 | 2025-01 | P3367R2 | EWG Evolution,LWG Library |
P3373R1 | Of Operation States and Their Lifetimes | Robert Leahy | 2025-01-11 | 2025-01 | P3373R0 | LEWG Library Evolution |
P3374R1 | Adding formatter for fpos | Liang Jiaming | 2025-01-07 | 2025-01 | P3374R0 | LEWGI SG18: LEWG Incubator,LEWG Library Evolution |
P3375R2 | Reproducible floating-point results | Guy Davidson | 2025-01-13 | 2025-01 | P3375R1 | SG6 Numerics,EWGI SG17: EWG Incubator |
P3385R3 | Attributes reflection | Aurelien Cassagnes | 2025-01-06 | 2025-01 | P3385R2 | SG7 Reflection |
P3388R1 | When Do You Know connect Doesn't Throw? | Robert Leahy | 2025-01-11 | 2025-01 | P3388R0 | LEWG Library Evolution |
P3394R1 | Annotations for Reflection | Daveed Vandevoorde | 2025-01-12 | 2025-01 | P3394R0 | SG7 Reflection,LEWG Library Evolution,CWG Core |
P3395R0 | Formatting of std::error_code | Victor Zverovich | 2024-12-22 | 2025-01 | SG16 Unicode | |
P3400R0 | Specifying Contract Assertion Properties with Labels | Joshua Berne | 2025-01-09 | 2025-01 | SG21 Contracts,All of WG21 | |
P3402R2 | A Safety Profile Verifying Initialization | Marc-André Laverdière | 2025-01-13 | 2025-01 | P3402R1 | SG23 Safety and Security |
P3407R1 | Make idiomatic usage of `offsetof` well-defined | Brian Bi | 2025-01-11 | 2025-01 | P3407R0 | EWG Evolution |
P3411R1 | `any_view` | Hui Xie | 2025-01-11 | 2025-01 | P3411R0 | SG9 Ranges,LEWG Library Evolution |
P3412R1 | String interpolation | Bengt Gustafsson | 2025-01-12 | 2025-01 | P3412R0 | EWGI SG17: EWG Incubator,EWG Evolution |
P3420R1 | Reflection of Templates | Andrei Alexandrescu | 2025-01-13 | 2025-01 | P3420R0 | SG7 Reflection |
P3423R1 | Extending User-Generated Diagnostic Messages | Yihe Li | 2025-01-12 | 2025-01 | P3423R0 | EWG Evolution |
P3425R1 | Reducing operation-state sizes for subobject child operations | Lewis Baker | 2025-01-13 | 2025-01 | P3425R0 | LEWG Library Evolution |
P3430R2 | simd issues: explicit, unsequenced, identity-element position, and members of disabled simd | Matthias Kretz | 2025-01-13 | 2025-01 | P3430R1 | LEWG Library Evolution |
P3431R0 | Deprecate const-qualifier on begin/end of views | Jonathan Müller | 2025-01-13 | 2025-01 | SG9 Ranges | |
P3439R1 | Chained comparisons: Safe, correct, efficient | Herb Sutter | 2025-01-06 | 2025-01 | P3439R0 | EWG Evolution |
P3475R1 | Defang and deprecate memory_order::consume | Hans Boehm | 2025-01-11 | 2025-01 | P3475R0 | SG1 Concurrency and Parallelism,EWG Evolution,LEWG Library Evolution,CWG Core,LWG Library |
P3477R2 | There are exactly 8 bits in a byte | JF Bastien | 2025-01-11 | 2025-01 | P3477R1 | LEWG Library Evolution,CWG Core |
P3480R3 | std::simd is a range | Matthias Kretz | 2025-01-13 | 2025-01 | P3480R2 | LEWG Library Evolution |
P3481R1 | std::execution::bulk() issues | Lucian Radu Teodorescu | 2025-01-13 | 2025-01 | P3481R0 | SG1 Concurrency and Parallelism,LEWG Library Evolution |
P3491R1 | define_static_{string,object,array} | Barry Revzin | 2025-01-12 | 2025-01 | P3491R0 | EWG Evolution,LEWG Library Evolution |
P3496R0 | Immediate-Escalating Expressions | Barry Revzin | 2025-01-05 | 2025-01 | EWG Evolution,LEWG Library Evolution | |
P3499R0 | Exploring strict contract predicates | Lisa Lippincott | 2025-01-13 | 2025-01 | SG21 Contracts,EWG Evolution | |
P3500R0 | Are Contracts "safe"? | Timur Doumler | 2025-01-13 | 2025-01 | EWG Evolution | |
P3501R0 | The ad-dressing of cats | S. Davis Herring | 2025-01-10 | 2025-01 | EWG Evolution,CWG Core,LWG Library | |
P3506R0 | P2900 Is Still not Ready for C++26 | Gabriel Dos Reis | 2025-01-13 | 2025-01 | EWG Evolution | |
P3516R0 | Uninitialized algorithms for relocation | Louis Dionne | 2025-01-13 | 2025-01 | LEWG Library Evolution | |
P3527R1 | Pattern Matching: *variant-like* and `std::expected` | Michael Park | 2025-01-13 | 2025-01 | P3527R0 | EWG Evolution,LEWG Library Evolution |
P3533R1 | constexpr virtual inheritance | Hana Dusíková | 2025-01-10 | 2025-01 | P3533R0 | EWG Evolution |
P3534R0 | Avoid UB When Compiling Code That Violates Library Specification | Alisdair Meredith | 2025-01-13 | 2025-01 | LWG Library | |
P3541R1 | Violation handlers vs `noexcept` | Andrzej Krzemieński | 2025-01-06 | 2025-01 | P3541R0 | SG21 Contracts,SG23 Safety and Security,EWG Evolution |
P3546R0 | Explicit return type deduction for std::numeric_limits and numbers | Thomas Mejstrik | 2024-12-19 | 2025-01 | LEWGI SG18: LEWG Incubator | |
P3547R0 | Modeling Access Control With Reflection | Dan Katz | 2025-01-09 | 2025-01 | SG7 Reflection,LEWG Library Evolution | |
P3548R0 | P1030 `std::filesystem::path_view` forward progress options | Niall Douglas | 2025-01-13 | 2025-01 | LEWG Library Evolution | |
P3549R0 | Diverging expressions | Barry Revzin | 2025-01-12 | 2025-01 | EWG Evolution | |
P3550R0 | Imports cannot ... | Alisdair Meredith | 2025-01-13 | 2025-01 | EWGI SG17: EWG Incubator,LEWGI SG18: LEWG Incubator | |
P3552R0 | Add a Coroutine Lazy Type | Dietmar Kühl | 2025-01-13 | 2025-01 | SG1 Concurrency and Parallelism,LEWG Library Evolution | |
P3554R0 | Non-transient allocation with vector and basic_string | Barry Revzin | 2025-01-05 | 2025-01 | EWG Evolution | |
P3555R0 | An infinite range concept | Jonathan Müller | 2025-01-13 | 2025-01 | SG9 Ranges | |
P3557R0 | High-Quality Sender Diagnostics with Constexpr Exceptions | Eric Niebler | 2025-01-12 | 2025-01 | LEWG Library Evolution | |
P3558R0 | Core Language Contracts By Default | Joshua Berne | 2025-01-12 | 2025-01 | SG21 Contracts,SG23 Safety and Security | |
P3559R0 | Trivial relocation: One trait or two? | Arthur O'Dwyer | 2025-01-08 | 2025-01 | LEWG Library Evolution | |
P3560R0 | Error Handling in Reflection | Barry Revzin | 2025-01-12 | 2025-01 | EWG Evolution,LEWG Library Evolution | |
P3561R0 | Index based coproduct operations on variant, and library wording | Esa Pulkkinen | 2025-01-13 | 2025-01 | LEWGI SG18: LEWG Incubator | |
P3564R0 | Make the concurrent forward progress guarantee usable in `bulk` | Mark Hoemmen | 2025-01-12 | 2025-01 | SG1 Concurrency and Parallelism | |
P3565R0 | Virtual floating-point values | S. Davis Herring | 2025-01-10 | 2025-01 | SG6 Numerics | |
P3566R0 | You shall not pass `char*` - Safety concerns working with unbounded null-terminated strings | Marco Foco | 2025-01-13 | 2025-01 | SG23 Safety and Security,LEWG Library Evolution,LWG Library,ARG ABI Review Group | |
P3567R0 | `flat_meow` Fixes | Hui Xie | 2025-01-11 | 2025-01 | LEWG Library Evolution,LWG Library | |
P3568R0 | break label; and continue label; | Jan Schultke | 2025-01-12 | 2025-01 | EWGI SG17: EWG Incubator,SG22 Compatibility | |
P3569R0 | Split define_aggregate from Reflection | Shafik Yaghmour | 2025-01-11 | 2025-01 | EWG Evolution | |
P3572R0 | Pattern matching | Bjarne Stroustrup | 2025-01-12 | 2025-01 | EWG Evolution | |
P3573R0 | Contract concerns | Bjarne Stroustrup | 2025-01-12 | 2025-01 | SG21 Contracts,EWG Evolution,LEWG Library Evolution | |
P3574R0 | Constexpr Callables | Steve Downey | 2025-01-13 | 2025-01 | LEWG Library Evolution,LWG Library | |
P3575R0 | SG14: Low Latency/Games/Embedded/Financial Trading virtual Meeting Minutes 2024/11/13 | Michael Wong | 2025-01-12 | 2025-01 | SG14 Low Latency | |
P3576R0 | SG19: Machine Learning virtual Meeting Minutes to 2024/11/14-2024/12/12 | Michael Wong | 2025-01-12 | 2025-01 | SG19 Machine Learning | |
P3577R0 | Requiring a non-throwing system-provided (default) contract-violation handler | John Lakos | 2025-01-12 | 2025-01 | SG21 Contracts,EWG Evolution | |
P3578R0 | Language Safety and Grandma Safety | Ryan McDougall | 2025-01-13 | 2025-01 | SG21 Contracts,SG23 Safety and Security,EWG Evolution | |
P3579R0 | Fix matching of non-type template parameters when matching template template parameters | Matheus Izvekov | 2025-01-13 | 2025-01 | CWG Core | |
P3580R0 | The Naming of Things | Alan Talbot | 2025-01-13 | 2025-01 | EWG Evolution,LEWG Library Evolution | |
P3581R0 | No, inplace_vector shouldn't have an Allocator | Nevin Liber | 2025-01-13 | 2025-01 | LEWG Library Evolution | |
P3582R0 | Observed a contract violation? Skip subsequent assertions! | Andrzej Krzemieński | 2025-01-13 | 2025-01 | SG21 Contracts,EWG Evolution | |
P3583R0 | Contracts, Types & Functions | Jonas Persson | 2025-01-13 | 2025-01 | SG21 Contracts | |
P3584R0 | Enrich Facade Creation Facilities for the Pointer-Semantics-Based Polymorphism Library - Proxy | Mingxin Wang | 2025-01-13 | 2025-01 | LEWGI SG18: LEWG Incubator,LEWG Library Evolution | |
P3585R0 | allocator_traits::is_internally_relocatable | Pablo Halpern | 2025-01-13 | 2025-01 | LEWGI SG18: LEWG Incubator,LEWG Library Evolution | |
P3586R0 | The Plethora of Problems With Profiles | Corentin Jabot | 2025-01-13 | 2025-01 | EWG Evolution | |
P3587R0 | Reconsider reflection access for C++26 | Lauri Vasama | 2025-01-13 | 2025-01 | SG7 Reflection,EWG Evolution | |
P3588R0 | Allow static data members in local and unnamed classes | Brian Bi | 2025-01-13 | 2025-01 | EWG Evolution | |
P3589R0 | C++ Profiles: The Framework | Gabriel Dos Reis | 2025-01-13 | 2025-01 | EWG Evolution | |
P3590R0 | Constexpr Coroutines Burdens | Daveed Vandevoorde | 2025-01-13 | 2025-01 | EWG Evolution,CWG Core |
By Blog Staff | Jan 10, 2025 11:06 AM | Tags: None
Last time, we investigated the puzzle of why the compiler wouldn’t let us put an object into a
std::optional
. It came down to the fact that the object is not copy-constructible, move-constructible, copy-assignable, or move-assignable, so there’s no way to put the temporary object into the std::optional
.
Solving the Puzzle of Trying to Put an Object into a std::optional
by Raymond Chen
From the article:
What we have to do is construct the object in place inside the
std::optional
. And the C++ standard library term for “construct an object inside a container” is “emplace”.struct Doodad { Doodad(); ~Doodad(); std::unique_ptr<DoodadStuff> m_stuff; }; struct Widget { std::optional<Doodad> m_doodad; Widget() { if (doodads_enabled()) { m_doodad.emplace(); } } };The parameters to
emplace
are whatever parameters you would have passed to theDoodad
constructor. In our case, we wanted the default constructor, so that means that we pass nothing toemplace()
.
By Andrey Karpov | Jan 8, 2025 07:28 AM | Tags: top 10 pvs-studio errors bugs
Every year, we witness the same drama: bugs wreak havoc on our code as if asserting their dominance. But today, the tide turns—it's time for judgment. Let's dive into the most intriguing bugs we've uncovered this year.
Court is in session: Top 10 most notorious C and C++ errors in 2024
by Aleksandra Uvarova
From the article:
The seventh defendant, what are you draggin' behind your back? Show us quickly! Ah! A little thief! Decided to pocket a juicy chunk of code, huh? No chance, we're bringing this to the light. Honestly, we really wanted to paste the full code of the function here—which has almost 400 code lines—to make searching for errors more interesting. However, we don't want to abuse your mouse wheel, so we've included just the most intriguing part.
By Blog Staff | Jan 6, 2025 11:15 AM | Tags: None
Since its introduction,
constexpr
in C++ has evolved significantly, offering powerful ways to optimize code at compile-time. This article shares a real-world story of using constexpr
to dramatically reduce memory usage and code size in an embedded system, showcasing its potential to improve both performance and efficiency.
Write More C++ Code Thanks to constexpr
by Andreas Fertig
From the article:
I'm a big fan of
constexpr
and am not alone. Jason Turner is also very vocal, having coined the term "constexpr all the things".Well, demonstrating the powers of
constexpr
is nonetheless something difficult. I know that from my training classes and various consulting contracts. Today, I'd like to share a story from back in time when a customer hired me to consult. They did develop an embedded system and ran out of memory. Not during run-time, but before. The features they wanted to put in the chip were too big in code size and somewhat RAM.Initial
constexpr
-free exampleThey used a class I've seen a couple of times in embedded systems with some variations. A string brings its memory picky-back.
By Blog Staff | Jan 3, 2025 11:07 AM | Tags: None
We all know that every ‘,’ matters in this language, so I decided to talk directly about that character today. So, how much impact can be for such a small little character?
It’s just ‘,’ – The Comma Operator
by Coral Kashri
From the article:
This operator comes from C, where it tells the compiler to evaluate all the expressions (left to right) and to return the result of the latest evaluated expression. For example:
int
a, b;
a = 5, b = 4, b += a, ++a, std::cout << b <<
" "
<< a;
// Prints 9 6
Another example of that operator usage is as follows:
for
(
size_t
i = 0, k = 500; i < 10; ++i, ++k) {
/*...*/
}
We can see this operator in action in the third section of thefor
statement. It evaluates the++i
and then evaluates++k
.
By Blog Staff | Dec 31, 2024 10:53 AM | Tags: None
The topic of this post is to show different ways to ensure that a class is either non-moveable or non-copyable.
How to Ensure a Class is not Copyable or Movable
by Sandor Dargo
From the article:
If we follow the classification proposed by Sebastian Theophil, we can talk about 4 different class types:
- value classes
- container classes
- resource classes
- singleton classes
While the first two should be regular classes offering both copy and move semantics, the latter two are different. One shouldn’t be able to copy resources and singletons probably shouldn’t be moveable.
It’s up to us to ensure that a class we create implements the right special member functions (SMFs from now on). And the Hinnant table is here to guide us.
By Blog Staff | Dec 27, 2024 10:47 AM | Tags: None
In C++, associating member objects like properties or events with their containing class often requires passing this redundantly. This article explores a generalized, flexible solution using templates, variadic arguments, and deducing this to streamline ownership initialization without boilerplate.
In C++, How Can I Make a Default Parameter be the This Pointer of the Caller? Revisited
by Raymond Chen
From the article:
Some time ago, we looked at making the default parameter of a method be the
this
pointer of the caller. The scenario was something like this:struct Property { Property(char const* name, int initial, Object* owner) : m_name(name), m_value(initial), m_owner(owner) {} ⟦ other methods elided - use your imagination ⟧ char const* m_name; Object* m_owner; int m_value; }; struct Widget : Object { Property Height{ "Height", 10, this }; Property Width{ "Width", 10, this }; };and we didn’t want to have to type
this
as the last parameter to all theProperty
constructors. We came up with this:template<typename D> struct PropertyHelper { Property Prop(char const* name, int initial) { return Property(name, initial, static_cast<D*>(this)); } }; struct Widget : Object, PropertyHelper<Widget> { Property Height = Prop("Height", 10); Property Width = Prop("Width", 10); };
By Blog Staff | Dec 23, 2024 01:33 PM | Tags: None
Some time ago, I noted that the
std::map
subscript operator is an attractive nuisance. It is the most convenient syntax, but is not often what you actually want.
The Operations for Reading and Writing Single Elements for C++ Standard Library Maps
by Raymond Chen
From the article:
I’ve broken down the various std::map
lookup and update operations into a table so you can choose the best one for your situation.
In the table above,
key
is the map key,value
is the mapped type, andparams
are parameters to the mapped type constructor.Note that
insert
and the firstemplace
¹ take a value which is discarded if it turns out that the key already exists. This is undesirable if creating the value is expensive.One frustrating scenario is the case where the mapped type’s default constructor is not the constructor you want to use for
operator[]
, or if you want the initial mapped value to be the result of a function call rather than a constructor. Here’s something I sort of threw together.