Coding Practice for my Job interviews and personal learnings
- mmap()
- page allocation (multiple subsequent calls to malloc())
- mutexes
- processes (fork())
- False Sharing
- Cache hits/misses
- TLB
- Memory Orderings
- Bus Faults (unaligned accesses)
- Compression and Decompression using Huffman Coding
- Nqueens problem
- Coins problem
- Number of possible paths from top left to bottom right in a grid
- AVL tree
- Maximum SubArray sum
- Binary Tree Traversals { In-Order, Pre-Order, Post-Order }
- Graph Traversals { DFS and BFS }
- Linked List Intersection find, Cycle detection, Singly Reversal, and Doubly Reversal
- Dynamic Bit List (unlike std::bitset<> which is static in size)
- Power Set Problem (Generate subsets of a given set)
- Binary Search Tree Validation (using Inorder Traversal, OR recursively validating each subtree, etc.)
- In-Order Successor of a node in a Binary Tree
- Finding Build Order of many projects given their dependency graph using DFS
- Generating all possible arrays which would generate a given binary search tree
{ Recursive Method (Permuting each sub-tree Mental Model procuded unexhaustive solutions), Weaving the output from both the subtrees produces all solutions } - Generating all possible weaved arrays out of two given arrays such that relative order among the elements remain consistent { Recursive Method (Ladder-Climbing Mental Model) }
- Checking if a binary tree is a subtree of another binary tree using recursion and substring matching.
- Finding all possible paths, not necessarily starting from root and ending to leaves, which sum to a given target sum { using Recursion }
- Finding magic index of an array (magic index == array[magic index]) using recursion (halving the search space recursively)
- Recursively multiply two positive integers using only addition/substraction/bitshifting while minimizing number of such operations
- Tower of Hanoi (transfering all disks from one tower to another using another tower as buffer while respecting the constraints) { using Recursion }
- Generating all permutations of a string { using Recursion }
- Generating all permutations of a string such that no duplication permutation occurs { using Recursion }
- Generating all valid permutations of given pairs of paranthesises { using Recursion }
- Generating all possible frequency of given set of coins to make a given sum { using Recursion }
- Generating all possible ways of evaluating (associating operands to operators) a given boolean expression such that the result is equal to a given value { using Recursion }
- Finding an element in a sorted then rotated array using Recursion method and Iterative method
- Finding an element in a sorted array with no given size, instead it returns -1 if index is out of range { using Binary Methods }
- Finding a string in a sorted array of strings containing empty strings intersparsed { using Binary Methods, Recursion }
- Finding Stationary Points using Gradient Descend
- Finding Roots using Tangent Descend or Newton Raphson Method
- Dynamic Bit Set
- Binary Search using
std::lower_bound()
#include <type_traits>
struct Data
{
int value;
};
template<typename ValueType>
static void function(ValueType& value, auto visitor)
{
visitor(value);
}
template<typename T>
struct remove_pointer
{
typedef T type;
};
template<typename T> struct remove_pointer<T*>
{
typedef T type;
};
template<typename T>
struct is_refers_to_const_value
{
static constexpr bool value = std::is_const<typename remove_pointer<typename std::remove_reference<T>::type>::type>::value;
};
template<typename T> requires(std::is_pointer<T>::value)
static void functionWrapper(T& t)
{
// NOTE: if T is const Data*
// then decltype(t->value) would return 'int'
// That means decltype() doesn't care about the cv-qualification of t when t->value is passed into it!
// Therefore we need to add const qualification based on if t refers to a constant value
using ValueType = typename std::conditional<is_refers_to_const_value<decltype(t)>::value,
typename std::add_const<decltype(t->value)>::type,
decltype(t->value)>::type;
function(t->value, [](ValueType& value) { });
}
static void myFunction()
{
Data* data = new Data { };
functionWrapper(data);
const Data* const_data = data;
functionWrapper(const_data);
}
#include <type_traits>
struct Data
{
char __bytes[255];
};
static void function()
{
const Data value { };
const Data& dataRef = value;
static_assert(std::is_same<decltype(dataRef), const Data&>::value);
// std::is_const<> to reference types always returns false as we can't qualify a reference type as 'const'
// NOTE: std::is_const<> tells whether the type is const qualified or not, it doesn't tell the intrinsic constness!
static_assert(std::is_const<decltype(dataRef)>::value); // it fails here
// Also
// std::is_const<const int*>::value or std::is_const<int const*>::value would evaluate to false
// And std::is_const<int * const>::value or std::is_const<const int* const>::value would evaluate to true
}
gcc -Wall -Wextra -Wpedantic main.cpp -o ./main
-Wpendatic compiler flag disables all the language extensions
gcc -fsanitize=address,undefined -c main.cpp -o ./main.o
Sanitizers must also be enbaled when linking, gcc -fsanitize=address,undefined main.o -o ./main
$ clang-tidy ...
$ cppcheck ...
OR
const auto data = [](){
std::vector<int> result;
// fill result with things.
return return;
}();
- This eliminates dynamic allocations
- Compiler can better optimize the code which uses the static constexpr variables
static constexpr std::array<int, 5> angles{-90, -45, 0, 45, 90};
// instead of
// static const std::vector<int> angles{-90, -45, 0, 45, 90};
std::string_view getValue() { return "Hello World"; }
std::string str = getValue(); // std::string would be instantiated here -> dynamic memory allocation
NOTE: Also
// Returns pointer to constant data
const MyClass* getValuePtr() { return new MyClass { }; }
// Type of value1 is 'const MyClass*'
auto value1 = getValuePtr();
// Type of value2 is still 'const MyClass*'
auto* value2 = getValuePtr();
// Type of value3 is still 'const MyClass*'
const auto* value3 = getValuePtr();
// Type of value4 is 'const MyClass* const'
const auto value4 = getValuePtr();
- Avoids type conversions (int to std::size_t or vice-versa)
- Looks compact
- Use clang-tidy's modernize-loop-convert check
- NOTE: I've done experiments to demonstrates how bit-width and signness conversions in integers affect performance
for(const auto& element : container) { }
// int to std::size_t conversion (or its opposte maybe)
for(int i = 0; i < container.size(); ++i) { }
NOTE: std::map<std::string, int>::value_type equals std::pair<const std::string, int>, if you use std::pair<std::string, int>, even just reference, it would create temporary object!
// Conversion from double to int
for(const int value : container_of_double) { }
// Accidental Slicing
for(const base value : container_of_derived) { }
// No problem
for(const auto& value: container) { }
class Stack { public: Data pop() { ... return value; } };
// This would avoid extra copy which may happen otherwise from temporary to 'value'
for(auto&& value : StackPopIterator(myStack))
const auto has_value = std::any_of(begin(container), end(container), greater_than(12));
// C++20
const auto has_value = std::any_of(container, greater_than(12));
https://github.com/pmd/pmd
In arch linux, you can install it directly. Package name is 'pmd', the tool name is 'pmd-cpd'
Do the following instead:
enum class Values
{
Value1;
Value2;
Value3;
};
std::string_view get_name(Values value)
{
switch(value)
{
case Value1: return "Value1";
case Value2: return "Value2";
} // unhandled enum value warning now
return "unknown";
}
- Move things out of the template code if possible
- This reduces binary size and compile times
- One may allow unchecked input to be passed into internal API functions
- But check every input in external API functions and produce diagnostic messages
- It is a C++17 attribute that tells the compiler to warn if a return value is ignored.
- It can be used on functions or the types itself
- C++20 adds the ability to provide a description
#include <stdexcept>
#include <concepts>
[[nodiscard]] auto divide(std::integral auto numerator, std::integral auto denominator) {
// is integer division
if (denominator == 0) {
throw std::runtime_error("divide by 0!");
}
return numerator / denominator;
}
[[nodiscard]] auto divide(auto numerator, auto denominator) {
// is floating point division
return numerator / denominator;
}
double high_precision_thing(double);
double high_precision_thing(float) = delete;
- Cmake
- Meson
- Bazel
- pkg-config
- conan
- vcpkg
- Each compiler does different analyses and implements the standard slightly different way.
struct SomeType { };
template<typename T>
concept SomeTypeLike = std::is_same_v<T, SomeType>;
template<SomeTypeLike T>
void myFunction(T* myPtr) { }
//...
myFunction(nullptr) // no matching call because of substitution failure
myFunction(static_cast<SomeType*>(nullptr)) // OK
I should always try to make the code simple to understand for others so that its reliability could be judged.
I should design a test first, which will also help me clear any interface ambiguities and reach to a solid interface specification. Once interface has been derived, the implementation usually becomes easier (my own experience).
"If you have even a slightest doubt about an Algorithm then Write (Theoretical) Proofs for its correctness"
Writing (mathematical, theoretical) proofs guarantee that the algorithm or the code would work under all inputs.
Any new data structure or algorithm I add must be accompanied its test suite consisting of normal inputs and inputs leading to errors.
Almost all popular computer scienists I came across on the web or textbooks are all have strong background in Mathematics or Physics.