The majority of code presented in this section comes from my nik library which can be described as: An independent C++ template library focused on the reimplementation of core STL data structures while maintaining a clean narrative design. The basic design decisions that will help you navigate its code content are as follows:
- additivity: Branches are split up and indexed by hardware abstractions. For example there are narrative branches for memory, the screen, the keyboard etc.—but to emphasize the idea that these are abstractions the branches are given alternative names than their respective hardware types: numeric, graphic, literic, etc.
- modularity: The modules within each branch are split up into semiotic and media modes. The difference being that media variants are meant for general users of the library with the implicit assumption that code in each mode is user-friendly and user-safe. Semiotic variants make no such assumptions, the code here is for the library architects: It is meant to be as reusable as possible.
- composability: Within semiotic modes, structure and function are kept independent and are only combined (adhering to the object oriented paradigm) within media modules. This allows for greater composability of structure with structure, and function with function.
- optimization: The above design decisions are the default but may be deviated from during implementation if a strong enough optimization is justified.
coproduct
May 3, 2018
Many if not most classes in nik are modelled as algebraic data types, meaning they are constructed from coproducts and products. Bartosz Milewski points out that although coproducts have no formal grammar in C++ they can still be interpreted within the language. Most notably, they show up as the disjoint union of overloaded constructors within a single class definition.
The intentions of my library differ slightly, and as consequence so does my model: To build an axiomatic narrative for data structure construction, I choose to interpret raw pointers as coproducts. For example, if you were to iterate over a linked list, effectively you're iterating over the pointers of its nodes. And even though these nodes exist concurrently in memory, they are accessed sequentially, meaning you can only look at them one at a time (hence the iteration). As such, you can view these nodes as the runtime disjoint union of alternatives: The interpretation then is that pointers themselves are runtime coproducts.
The source code is here.
read_type
May 4, 2018
From the experience of past projects I have included a read_type template class in the global definitions of my library. This class has no runtime consequences, it's entirely overhead for maintaining type safety as an alternative to using the const keyword in regards to pointer definitions. Intuitively, you give it a Type, and it either adds or removes the const accordingly, taking special care when that type is a pointer Type*.
The main application of such meta-programming comes when defining iterators, const_iterators. In practice such iterators tend to have nearly identical definitions with the exception of a few const placements, and as const is entirely a compile time type safety consideration, it makes sense to handle it in this way. The alternative is to define separate cases for each iterator which is unnecessarily redundant.
The source code is here.
copair
May 4, 2018
With some foresight regarding the larger narrative design of this library, I have included as part of the definitions a pair of coproducts—aptly named copair.
Coproducts are used to define iterators, a construct already common in C++. On the other hand, until recently the idea of ranges—which already show up in many other languages—was underutilized. Copairs will be used to define such ranges in this library. Conceptually they parallel iterators, and as such they will be refered to as selectors. This particular name derives from the ubiquitous digital utility of text selection.
The source code is here.
product
May 4, 2018
As mentioned with coproducts, many if not most classes in nik are modelled as algebraic data types, meaning products are also a fundamental grammatical construct within this library.
If a coproduct is interpreted as the sequential union of instances of a type, then a product is the concurrent union of such instances. Granted, these are narrower definitions of coproducts, products than the full category theoretic varieties, but it is our intention to take an interpretation which aligns well with the idea of a register machine, and which aligns well with the grammar of C++ itself.
Concurrent memory requires an array, as such our product as type is implemented as one of fixed length. We might be tempted to build into this low-level definition a more powerful dynamic length array, but the overarching narrative of this library is that all data structures are fundamentally implemented as trees, which are themselves decomposed into fixed length arrays. For a given data structure, there's no assumption its size will change, on the other hand it's easy enough to define equipment for these products to actively vary the length of such memory.
Finally, even though the memory of products is concurrent, the assumption is that access to each product component is still sequential, meaning we need ways of expressing and navigating product substructures. In this class we declare iterators and selectors to do just this.
The source code is here.