From 87274c560e278110337a2cd530099a8e8c2bf5c0 Mon Sep 17 00:00:00 2001
From: Brett <brettmaster1@gmail.com>
Date: Thu, 13 Mar 2025 21:35:58 -0400
Subject: [PATCH] more examples

---
 CMakeLists.txt           |  2 +-
 src/michael_examples.cpp | 78 +++++++++++++++++++++++++++++++++++++++-
 2 files changed, 78 insertions(+), 2 deletions(-)

diff --git a/CMakeLists.txt b/CMakeLists.txt
index 9c2357b..a57387a 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -51,7 +51,7 @@ macro(blt_add_project name source type)
     project(tower-defense)
 endmacro()
 
-project(tower-defense VERSION 0.0.5)
+project(tower-defense VERSION 0.0.6)
 
 option(ENABLE_ADDRSAN "Enable the address sanitizer" OFF)
 option(ENABLE_UBSAN "Enable the ub sanitizer" OFF)
diff --git a/src/michael_examples.cpp b/src/michael_examples.cpp
index 88b466c..1aaf38b 100644
--- a/src/michael_examples.cpp
+++ b/src/michael_examples.cpp
@@ -36,6 +36,14 @@
 
 #include <blt/std/assert.h>
 
+/*
+ * Naming Conventions
+ */
+// most classes, functions, variable names, etc should use lower_snake_case
+// enum / enum class should use lower_snake_case for the type name, then UPPER_SNAKE_CASE for elements
+// class member (private) variables should be prefixed with m_
+// prefer enum class defs over enum defs, they are namespaced.
+
 void michael_examples()
 {
 	/*
@@ -79,7 +87,7 @@ void michael_examples()
 		// the initializer is run before any other code, thus doing it inline guarantees the class is in a valid state.
 		explicit rule_of_5_t(const blt::i32 id): m_id(id)
 		{
-			BLT_ASSERT(m_id > 0 && "id must be greater than 0 (zero is used to represent empty state)");
+			// BLT_ASSERT(m_id > 0 && "id must be greater than 0 (zero is used to represent empty state)");
 			BLT_TRACE("I ({}) was constructed", m_id);
 		}
 
@@ -131,7 +139,11 @@ void michael_examples()
 		blt::i32 m_id = 0;
 	};
 
+	BLT_TRACE("--{}--", "{Rule of 5}");
+
 	// look at the output from this, you will understand what is going on.
+	// we should talk about R and L value references right?
+	// soon! (keep this concept in mind, basically R value references are temporary objects, L values point to real objects)
 	{
 		// scoped block means lifetimes of objects only exist for the lifetime of the block
 		rule_of_5_t test1(1); // constructed
@@ -152,6 +164,9 @@ void michael_examples()
 		// you'll notice that destruction is in reverse order of declaration
 	}
 
+	BLT_TRACE("");
+	BLT_TRACE("--{}--", "{Rule of 0}");
+
 	/*
 	 * You know from 2P95 that we can use new/delete to allocate memory.
 	 * Don't do that. Fun fact: you basically never need to use new/delete anymore
@@ -161,4 +176,65 @@ void michael_examples()
 	 * What this means is, using standard headers, you never need to worry about manual memory management.
 	 * This is called the rule of 0. You don't need to define copy/move/destructors when the underlying types handle it for you.
 	 */
+
+	struct rule_of_0_t
+	{
+		// this is a vector of rule_of_5_t, use it to show that copy and move are already made for us
+		std::vector<rule_of_5_t> m_rules;
+	};
+
+	{
+		rule_of_0_t test3;
+		// emplace_back is used to construct an object in place, you pass this function arguments. If you want to pass an already existing object,
+		// you can use .push_back(object) instead.
+		test3.m_rules.emplace_back(1);
+		test3.m_rules.emplace_back(2);
+		test3.m_rules.emplace_back(3);
+
+		// this will make a copy
+		rule_of_0_t copy_of_3 = test3;
+		// this will make a move
+		const rule_of_0_t move_of_3 = std::move(test3);
+		// all of which is implicitly defined, because std::vector has respective move and copy constructors defined.
+
+		// this function acts as a black box to the compiler. Can prevent optimizations in cases like this.
+		blt::black_box(move_of_3);
+	}
+
+	BLT_TRACE("");
+	BLT_TRACE("--{}--", "{std::vector reserve()}");
+
+	// now remember the problem of vectors allocating, then moving objects around internally as a result?
+	// well you can manually allocate a number of elements in a vector.
+	{
+		std::vector<rule_of_5_t> vector_of_rules;
+		vector_of_rules.reserve(100); // vector is empty, but can store 100 elements before reallocating
+
+		BLT_ASSERT(vector_of_rules.empty());
+		BLT_ASSERT(vector_of_rules.capacity() == 100);
+
+		for (int i = 0; i < 10; ++i)
+			vector_of_rules.emplace_back(i + 1);
+		// you'll notice in the console there's no sign of reallocation - objects are only constructed - in place
+	}
+
+	// Ok. Cool. If you know the number of elements you can reserve and not have to reallocate. But you said std::vector can replace basically
+	// any dynamic array. How about when I know the number of elements, do I always have to use .emplace_back()/.push_back()????
+	// the answer is of course not!
+	BLT_TRACE("");
+	BLT_TRACE("--{}--", "{std::vector resize()}");
+	{
+		// you can directly ask a size in the constructor (as long as you are not using an integer type as T)
+		// (I don't tend to use this and prefer calling the function directly)
+		// If the type is not default constructible you must provide the type to fill the requested elements with.
+		// in which case the type must be copy constructible
+		std::vector<rule_of_5_t> vector_of_rules{10, rule_of_5_t{0}};
+		// or manually call resize
+		vector_of_rules.resize(100, rule_of_5_t{0}); // vector has 100 elements of rule_of_5_t{0}
+
+		// welcome to our first use of a BLT iterator
+		// this is called a structured binding. You can use it to unpack structs. enumerate returns a tuple of <size_t, T>
+		for (const auto [i, v] : blt::enumerate(vector_of_rules))
+			v = rule_of_5_t{static_cast<int>(i)};
+	}
 }