C++ Learning Repository
Welcome to my C++ Learning Repository! This repository serves as a documentation of my journey in mastering the concepts of C++ programming through practical coding exercises and projects.
The primary goal of this repository is to provide a structured approach towards learning C++ programming language. By actively coding and implementing various concepts, I aim to solidify my understanding and proficiency in C++.
-
Code Samples: This directory contains a collection of code snippets covering fundamental to advanced topics in C++, including variables, loops, functions, classes, inheritance, polymorphism, templates, and more.
-
Projects: Here, you'll find projects that I've worked on to apply and integrate multiple C++ concepts. These projects range from simple console applications to more complex programs, demonstrating practical usage of C++.
-
Exercises: This section comprises sets of exercises designed to reinforce specific C++ concepts. Each exercise may include a problem statement, my solution, and sometimes alternative solutions for comparison.
-
Tutorials: I occasionally write tutorials covering specific topics or concepts in C++. These tutorials are aimed at providing clear explanations and examples to aid in understanding complex concepts.
C++ supports various programming paradigms, each suited to different types of problems and domains. In the context of embedded systems, certain paradigms are particularly relevant due to the specific constraints and requirements of these systems. Here are the key programming paradigms in C++, with a focus on those most used in embedded systems:
Concept: Procedural programming is based on the concept of procedure calls, where the program is structured into procedures, also known as functions or routines. These functions operate on data and are called to perform specific tasks.
Usage in Embedded Systems:
- Deterministic Behavior: Embedded systems often require deterministic behavior, which procedural programming can provide through straightforward function calls and loops.
- Efficiency: Procedural programming can be more efficient in terms of memory and processing power, which is crucial in resource-constrained embedded environments.
- Example: Writing low-level drivers and interrupt service routines (ISRs) are often done using procedural programming.
Concept: OOP organizes software design around data, or objects, rather than functions and logic. Objects are instances of classes, which can contain both data (attributes) and functions (methods).
Usage in Embedded Systems:
- Encapsulation: Encapsulation helps in managing complexity by bundling data and methods that operate on the data within classes.
- Inheritance: Inheritance allows code reuse, which is beneficial in developing scalable and maintainable embedded systems.
- Polymorphism: Polymorphism enables designing flexible and reusable code, which is essential for handling different hardware configurations and devices.
- Example: Using classes to represent hardware peripherals and their operations, such as a class for a UART interface.
Concept: Generic programming uses templates to write functions and classes that work with any data type. This allows for code that is type-safe and reusable.
Usage in Embedded Systems:
- Code Reusability: Templates facilitate writing reusable code that can handle different data types without duplication.
- Efficiency: Template code is resolved at compile time, leading to efficient executable code, which is important for performance-critical embedded applications.
- Example: Implementing a generic buffer or queue that can store data of any type, useful in communication systems.
Concept: Metaprogramming involves writing code that generates or manipulates other code at compile time. In C++, this is often achieved using templates and the preprocessor.
Usage in Embedded Systems:
- Compile-Time Optimization: Metaprogramming can be used to perform computations and generate code at compile time, reducing runtime overhead.
- Static Assertions: It helps in verifying certain conditions at compile time, ensuring safer and more robust code.
- Example: Compile-time checks for configuration parameters or static generation of lookup tables for efficient runtime access.
Concept: Concurrency involves multiple threads or processes running simultaneously, which can be managed using threads and synchronization primitives in C++.
Usage in Embedded Systems:
- Real-Time Systems: Concurrency is essential in real-time embedded systems where multiple tasks must run concurrently, such as handling sensor data, communication protocols, and user interfaces.
- Multicore Processors: Utilizing multicore processors in embedded systems requires efficient concurrency management.
- Example: Using threads to handle different tasks, like reading sensor data while simultaneously updating a display.
- Procedural Programming: For its simplicity and efficiency, making it ideal for low-level hardware interaction.
- Object-Oriented Programming: For managing complexity and providing modular and reusable code, crucial in larger and more complex embedded systems.
- Generic Programming: For creating reusable and type-safe code components, enhancing code flexibility and maintainability.
- Concurrency and Multithreading: For real-time and multitasking capabilities, essential in modern embedded systems that require handling multiple simultaneous operations.
Each of these paradigms contributes to addressing the unique challenges of embedded systems development, from resource constraints and real-time requirements to code maintainability and scalability.
C++ Interesting concepts : https://vishalchovatiya.com/pages/start-here/
May be One or two Articles from : https://nimrod.blog/
May be not specific to C++ but : https://devblogs.microsoft.com/oldnewthing/
Something I need to delve into : https://cpppatterns.com/
C++ Tips from : https://abseil.io/tips/
ModernCpp Github resource : https://github.com/federico-busato/Modern-CPP-Programming
Few other interesting and related resources:
https://blog.andreiavram.ro/cpp-custom-allocators-first-step/
https://www.rkaiser.de/wp-content/uploads/2021/03/embo2021-pmr-STL-for-Embedded-Applications-en.pdf
https://github.com/mtrebi/memory-allocators
https://maxnilz.com/
https://www.jonathanmueller.dev/talk/cache-friendly-cpp/
https://cpp-optimizations.netlify.app/
https://yyc.solvcon.net/en/latest/index.html
https://learn.microsoft.com/en-us/cpp/cpp/object-lifetime-and-resource-management-modern-cpp?view=msvc-170
https://isocpp.org/wiki/faq/strange-inheritance
https://github.com/methylDragon/coding-notes/tree/master/C%2B%2B
https://www.rkaiser.de/wp-content/uploads/2021/03/embo2021-pmr-STL-for-Embedded-Applications-en.pdf
-
C++11/14/17/20 Core Language Features
- Auto keyword
- Lambda expressions
- Move semantics and rvalue references
- Smart pointers
- Variadic templates
- Constexpr and compile-time computation
- Concepts (C++20)
-
Standard Template Library (STL)
- Containers (vector, map, unordered_map, etc.)
- Algorithms
- Iterators
- Function objects
-
Object-Oriented Programming in C++
- Classes and inheritance
- Polymorphism
- RAII (Resource Acquisition Is Initialization)
- Rule of 0/3/5
-
Memory Management
- Stack vs Heap
- Dynamic memory allocation
- Smart pointers (unique_ptr, shared_ptr, weak_ptr)
-
Generic Programming
- Templates (function and class templates)
- Template specialization
- SFINAE (Substitution Failure Is Not An Error)
-
Concurrency
- Threads and thread management
- Mutexes and locks
- Condition variables
- Futures and promises
- Atomic operations
-
Modern C++ Best Practices
- SOLID principles
- Exception handling
- Type safety
- const-correctness
-
C++20 Features
- Ranges
- Coroutines
- Modules
- Concepts
-
Build Systems and Tools
- CMake
- Compiler options and flags
- Static analyzers (e.g., Clang-Tidy)
- Debuggers (e.g., GDB, LLDB)
-
Performance Optimization
- Profiling
- Cache-friendly code
- Compiler optimizations