Skip to content

Coding Practice for my Job interviews as well as experiments.

License

Notifications You must be signed in to change notification settings

ravi688/CrackingTheCodingInterview

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

CrackingTheCodingInterview

Coding Practice for my Job interviews and personal learnings

OS concepts

  • mmap()
  • page allocation (multiple subsequent calls to malloc())
  • mutexes
  • processes (fork())

Hardware concepts

  • False Sharing
  • Cache hits/misses
  • TLB
  • Memory Orderings
  • Bus Faults (unaligned accesses)

Algorithms, and Data Structures

  • 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 }

Numerical Algorithms

  • Finding Stationary Points using Gradient Descend
  • Finding Roots using Tangent Descend or Newton Raphson Method

Reusable Code Snippets

  1. Dynamic Bit Set
  2. Binary Search using std::lower_bound()

C++ Template Argument Deduction

Constness of data member variables of a const qualified class object

#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);
}

What does std::is_const<>::value evalutes to for reference types?

#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
}

C++ Software Engineering

Use Warnings

gcc -Wall -Wextra -Wpedantic main.cpp -o ./main
-Wpendatic compiler flag disables all the language extensions

Use Sanitisers

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

Use Static Analysis

$ clang-tidy ...
$ cppcheck ...

Always try to qualify variables as const

OR

const auto data = [](){
 std::vector<int> result;
 // fill result with things.
 return return;
}();

Use static constexpr for values known at compile time

  • 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};

Always use auto to avoid conversions and memory allocations

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();

Always use range based for loop over Old loops

  • 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) { }

Use auto in ranged for loops

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))

Prefer declaractive sytle () algorithms

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));

Use Copy-Paste Detector (CPD)

https://github.com/pmd/pmd
In arch linux, you can install it directly. Package name is 'pmd', the tool name is 'pmd-cpd'

Do not use default in switch case statements

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";
}

De-template-ize your generic code

  • Move things out of the template code if possible
  • This reduces binary size and compile times

Internal and External APIs [ Optional ]

  • One may allow unchecked input to be passed into internal API functions
  • But check every input in external API functions and produce diagnostic messages

Use [[nodiscard]] liberally

  • 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;
}

Consider deleting problematic conversions

double high_precision_thing(double);
double high_precision_thing(float) = delete;

Use Build Generators (for portability)

  • Cmake
  • Meson
  • Bazel

Use Package Managers (for dep management)

  • pkg-config
  • conan
  • vcpkg

Use Multiple Compilers

  • Each compiler does different analyses and implements the standard slightly different way.

Miscelleneous

Type constraint on Pointer types leads to substituion failure with nullptr

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

Insightful lessons from very experienced Software Engineers and Scientists

"Simplicity is the prerequisite to reliability"

I should always try to make the code simple to understand for others so that its reliability could be judged.

"If the code can't easily be tested then it is not well designed"

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.

"Do not leave unit testing for later, Write everytime you add new code"

Any new data structure or algorithm I add must be accompanied its test suite consisting of normal inputs and inputs leading to errors.

"Study Mathematics to become a real Computer Scientist"

Almost all popular computer scienists I came across on the web or textbooks are all have strong background in Mathematics or Physics.

Releases

No releases published

Packages

No packages published