STC ccommon: Generic algorithms and macros
The following macros are recommended to use, and they safe/have no side-effects.
General defer mechanics for resource acquisition. These macros allows you to specify the
freeing of the resources at the point where the acquisition takes place.
The checkauto utility described below, ensures that the c_auto*
macros are used correctly.
Usage | Description |
---|---|
c_with (Type var=init, drop) |
Declare var . Defer drop... to end of scope |
c_with (Type var=init, pred, drop) |
Adds a predicate in order to exit early if init failed |
c_auto (Type, var1,...,var4) |
c_with (Type var1=Type_init(), Type_drop(&var1)) ... |
c_scope (init, drop) |
Execute init and defer drop to end of scope |
c_defer (drop...) |
Defer drop... to end of scope |
continue |
Exit a block above without memory leaks |
For multiple variables, use either multiple c_with in sequence, or declare variable outside scope and use c_scope. For convenience, c_auto support up to 4 variables.
// `c_with` is similar to python `with`: it declares and can drop a variable after going out of scope.
bool ok = false;
c_with (uint8_t* buf = malloc(BUF_SIZE), buf != NULL, free(buf))
c_with (FILE* fp = fopen(fname, "rb"), fp != NULL, fclose(fp))
{
int n = fread(buf, 1, BUF_SIZE, fp);
if (n <= 0) continue; // auto cleanup! NB do not break or return here.
...
ok = true;
}
return ok;
// `c_auto` automatically initialize and destruct up to 4 variables, like c_with.
c_auto (cstr, s1, s2)
{
cstr_append(&s1, "Hello");
cstr_append(&s1, " world");
cstr_append(&s2, "Cool");
cstr_append(&s2, " stuff");
printf("%s %s\n", cstr_str(&s1), cstr_str(&s2));
}
c_with (cstr str = cstr_lit("Hello"), cstr_drop(&str))
{
cstr_append(&str, " world");
printf("%s\n", cstr_str(&str));
}
// `c_scope` is like `c_with` but works with an already declared variable.
static pthread_mutex_t mut;
c_scope (pthread_mutex_lock(&mut), pthread_mutex_unlock(&mut))
{
/* Do syncronized work. */
}
// `c_defer` executes the expressions when leaving scope. Prefer c_with or c_scope.
cstr s1 = cstr_lit("Hello"), s2 = cstr_lit("world");
c_defer (cstr_drop(&s1), cstr_drop(&s2))
{
printf("%s %s\n", cstr_str(&s1), cstr_str(&s2));
}
Example: Load each line of a text file into a vector of strings:
#include <errno.h>
#include <stc/cstr.h>
#define i_val_str
#include <stc/cvec.h>
// receiver should check errno variable
cvec_str readFile(const char* name)
{
cvec_str vec = cvec_str_init(); // returned
c_with (FILE* fp = fopen(name, "r"), fp != NULL, fclose(fp))
c_with (cstr line = cstr_NULL, cstr_drop(&line))
while (cstr_getline(&line, fp))
cvec_str_emplace_back(&vec, cstr_str(&line));
return vec;
}
int main()
{
c_with (cvec_str x = readFile(__FILE__), cvec_str_drop(&x))
c_foreach (i, cvec_str, x)
printf("%s\n", cstr_str(i.ref));
}
The checkauto program will check the source code for any misuses of the c_auto*
macros which
may lead to resource leakages. The c_auto*
- macros are implemented as one-time executed for-loops,
so any return
or break
appearing within such a block will lead to resource leaks, as it will disable
the cleanup/drop method to be called. A break
may originally be intended to break a loop or switch
outside the c_auto
scope.
NOTE: One must always make sure to unwind temporary allocated resources before a return
in C. However, by using c_auto*
-macros,
- it is much easier to automatically detect misplaced return/break between resource acquisition and destruction.
- it prevents forgetting to call the destructor at the end.
The checkauto utility will report any misusages. The following example shows how to correctly break/return
from a c_auto
scope:
int flag = 0;
for (int i = 0; i<n; ++i) {
c_auto (cstr, text)
c_auto (List, list)
{
for (int j = 0; j<m; ++j) {
List_push_back(&list, i*j);
if (cond1())
break; // OK: breaks current for-loop only
}
// WRONG:
if (cond2())
break; // checkauto ERROR! break inside c_auto.
if (cond3())
return -1; // checkauto ERROR! return inside c_auto
// CORRECT:
if (cond2()) {
flag = 1; // flag to break outer for-loop
continue; // cleanup and leave c_auto block
}
if (cond3()) {
flag = -1; // return -1
continue; // cleanup and leave c_auto block
}
...
}
// do the return/break outside of c_auto
if (flag < 0) return flag;
else if (flag > 0) break;
...
} // for
Iterate compound literal array elements. Additional to i.ref
, you can access i.data
, i.size
, and i.index
of the input list/element.
// apply multiple push_backs
c_forlist (i, int, {1, 2, 3})
cvec_i_push_back(&vec, *i.ref);
// insert in existing map
c_forlist (i, cmap_ii_raw, { {4, 5}, {6, 7} })
cmap_ii_insert(&map, i.ref->first, i.ref->second);
// string literals pushed to a stack of cstr:
c_forlist (i, const char*, {"Hello", "crazy", "world"})
cstack_str_emplace(&stk, *i.ref);
// reverse the list:
c_forlist (i, int, {1, 2, 3})
cvec_i_push_back(&vec, i.data[i.size - 1 - i.index]);
Usage | Description |
---|---|
c_foreach (it, ctype, container) |
Iteratate all elements |
c_foreach (it, ctype, it1, it2) |
Iterate the range [it1, it2) |
c_forpair (key, val, ctype, container) |
Iterate with structured binding |
#define i_key int
#define i_val int
#define i_tag ii
#include <stc/csmap.h>
...
csmap_ii map = c_make(csmap_ii, { {23,1}, {3,2}, {7,3}, {5,4}, {12,5} });
c_foreach (i, csmap_ii, map)
printf(" %d", i.ref->first);
// 3 5 7 12 23
// same without using c_foreach:
for (csmap_ii_iter i = csmap_ii_begin(&map); i.ref; csmap_ii_next(&i))
printf(" %d", i.ref->first);
csmap_ii_iter it = csmap_ii_find(&map, 7);
// iterate from it to end
c_foreach (i, csmap_ii, it, csmap_ii_end(&map))
printf(" %d", i.ref->first);
// 7 12 23
// structured binding:
c_forpair (id, count, csmap_ii, map)
printf(" (%d %d)", *_.id, *_.count);
// (3 2) (5 4) (7 3) (12 5) (23 1)
Abstraction for iterating sequence of numbers. Like python's for i in range() loop.
Usage | Python equivalent |
---|---|
c_forrange (stop) |
for _ in range(stop): |
c_forrange (i, stop) // i type = long long |
for i in range(stop): |
c_forrange (i, start, stop) |
for i in range(start, stop): |
c_forrange (i, start, stop, step) |
for i in range(start, stop, step): |
c_forrange (5) printf("x");
// xxxxx
c_forrange (i, 5) printf(" %lld", i);
// 0 1 2 3 4
c_forrange (i, -3, 3) printf(" %lld", i);
// -3 -2 -1 0 1 2
c_forrange (i, 30, 0, -5) printf(" %lld", i);
// 30 25 20 15 10 5
Iterate containers with stop-criteria and chained range filtering.
Usage | Description |
---|---|
c_forfilter (it, ctype, container, filter) |
Filter out items in chain with && |
c_forfilter (it, ctype, container, filter, pred) |
Filter and iterate until pred is false |
c_forwhile (it, ctype, start, pred) |
Iterate until pred is false |
Built-in filter | Description |
---|---|
c_flt_skip(it, numItems) |
Skip numItems |
c_flt_take(it, numItems) |
Take numItems |
c_flt_skipwhile(it, predicate) |
Skip items until predicate is false |
c_flt_takewhile(it, predicate) |
Take items until predicate is false |
c_flt_last(it) |
Get count of last filter successes |
c_flt_lastwhile(it) |
Get value of last while-filter |
it.index
holds the index of the source item.
// Example:
#include <stc/algo/crange.h>
#include <stc/algo/filter.h>
#include <stdio.h>
bool isPrime(int i) {
for (int j=2; j*j <= i; ++j) if (i % j == 0) return false;
return true;
}
// Get 10 prime numbers after 1 million, but only every 25th of them.
int main() {
crange R = crange_make(1000001, INT32_MAX, 2);
c_forfilter (i, crange, R,
isPrime(*i.ref)
&& (c_flt_skip(i, INT32_MAX) ||
c_flt_last(i) % 25 == 0)
, c_flt_take(i, 10)) // breaks loop on false.
{
printf(" %d", *i.ref);
}
}
// Out: 1000303 1000639 1000999 1001311 1001593 1001981 1002299 1002583 1002887 1003241
Note that c_flt_take()
is given as an optional argument, which breaks the loop on false
(for efficiency). Without the comma, it will give same result, but the full input is processed first.
- c_make: Make any container from an initializer list. Example:
#define i_val_str // cstr value type
#include <stc/cset.h>
#define i_key int
#define i_val int
#include <stc/cmap.h>
...
// Initializes with const char*, internally converted to cstr!
cset_str myset = c_make(cset_str, {"This", "is", "the", "story"});
int x = 7, y = 8;
cmap_int mymap = c_make(cmap_int, { {1, 2}, {3, 4}, {5, 6}, {x, y} });
- c_new(Type): Allocate and init a new object on the heap
- c_delete(Type, ptr): Drop and free an object allocated on the heap
#include <stc/cstr.h>
cstr *stringptr = c_new(cstr, cstr_from("Hello"));
printf("%s\n", cstr_str(stringptr));
c_delete(cstr, stringptr);
A number sequence generator type, similar to boost::irange. The crange_value type is long long
. Below start, stop, and step are of type crange_value:
crange& crange_obj(...) // create a compound literal crange object
crange crange_make(stop); // will generate 0, 1, ..., stop-1
crange crange_make(start, stop); // will generate start, start+1, ... stop-1
crange crange_make(start, stop, step); // will generate start, start+step, ... upto-not-including stop
// note that step may be negative.
crange_iter crange_begin(crange* self);
crange_iter crange_end(crange* self);
void crange_next(crange_iter* it);
// 1. All primes less than 32:
crange r1 = crange_make(3, 32, 2);
printf("2"); // first prime
c_forfilter (i, crange, r1
, isPrime(*i.ref))
printf(" %lld", *i.ref);
// 2 3 5 7 11 13 17 19 23 29 31
// 2. The 11 first primes:
printf("2");
c_forfilter (i, crange, crange_obj(3, INT64_MAX, 2)
, isPrime(*i.ref)
, c_flt_take(10))
printf(" %lld", *i.ref);
// 2 3 5 7 11 13 17 19 23 29 31
Find or erase linearily in containers using a predicate
// Search vec for first value > 2:
cvec_i_iter i;
c_find_if(i, cvec_i, vec, *i.ref > 2);
if (i.ref) printf("%d\n", *i.ref);
// Search map for a string containing "hello" and erase it:
cmap_str_iter it, it1 = ..., it2 = ...;
c_find_if(it, csmap_str, it1, it2, cstr_contains(it.ref, "hello"));
if (it.ref) cmap_str_erase_at(&map, it);
// Erase all strings containing "hello":
// Note 1: iter i need not be declared.
// Note 2: variables index and count can be accessed in predicate.
c_erase_if(i, csmap_str, map, cstr_contains(i.ref, "hello"));
// Safe macro for swapping internals of two objects of same type:
c_swap(cmap_int, &map1, &map2);
// Drop multiple containers of same type:
c_drop(cvec_i, &vec1, &vec2, &vec3);
int c_default_cmp(const Type*, const Type*);
Type c_default_clone(Type val); // simple copy
Type c_default_toraw(const Type* val); // dereference val
void c_default_drop(Type* val); // does nothing
typedef const char* crawstr;
int crawstr_cmp(const crawstr* x, const crawstr* y);
bool crawstr_eq(const crawstr* x, const crawstr* y);
uint64_t crawstr_hash(const crawstr* x);
Memory allocator wrappers that uses signed sizes.
Return number of elements in an array. array must not be a pointer!
int array[] = {1, 2, 3, 4};
intptr_t n = c_arraylen(array);