Skip to content

mkh-welsch/crossfilter

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

60 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

crossfilter

cross::filter adds OLAP functionality on top of an underlying concurrent std::vector.

At first crossfilter is a theadsafe drop in replacement for std::vector.

On top of the basic functionality of std::vector you can can define some kind of filtered views, for example the top 10 based on a multi-dimensional filter setup, which are than automatically updated (!) during usual data operations on the data. So you could build a lightweight and header only (M)OLAP in memory query database for ultra fast analysis of big data sets and use it for complex real time filtering tasks.

quick instructions

At first we have some common/arbitrary structured data called Record.

struct Record
{
    std::string date;
    int quantity;
    int total;
    int tip;
    std::string type;
    std::vector<std::string> productIDS;
};

Lets store some of theses records in cross::vector and investigate the new possibilities…

cross::filter<Record> payments = {
    {"2011-11-14T16:17:54Z", 2, 190, 100, "tab", {"001", "002"}},
    {"2011-11-14T16:20:19Z", 2, 190, 100, "tab", {"001", "005"}},
    {"2011-11-14T16:28:54Z", 1, 300, 200, "visa", {"004", "005"}},
    {"2011-11-14T16:30:43Z", 2, 90, 1, "tab", {"001", "002"}},
    {"2011-11-14T16:48:46Z", 2, 90, 2, "tab", {"005"}},
    {"2011-11-14T16:53:41Z", 2, 90, 3, "tab", {"001", "004", "005"}},
    {"2011-11-14T16:54:06Z", 1, 100, 4, "cash", {"001", "002", "003", "004", "005"}},
    {"2011-11-14T16:58:03Z", 2, 90, 5, "visa", {"001"}},
    {"2011-11-14T17:07:21Z", 2, 90, 6, "tab", {"004", "005"}},
    {"2011-11-14T17:22:59Z", 2, 90, 7, "tab", {"001", "002", "004", "005"}}
};

you can use all constructors and functionalties of std::vector<> with cross::filter<>, but operations are thread safe in comparison to std::vector

To use the special crossfilters functionality, you have to prepare „dimensions“ on the record sets. A dimension is just a sortable column by regarding the record set as table with single entries.

You build a dimension by giving an accressor function as generic lambda.

auto totals = payments.dimension([](auto r) { return r.total;});

Now, you can filter the record set in regarding to this dimension ...

totals.filter(90); 	// filter out every record, where total is 90

... and give back the filtered results as copys

auto filtered_results = payments.all_filtered();
    // "2011-11-14T16:30:43Z", 2, 90, 1, "tab", {"001", "002"}
    // "2011-11-14T16:48:46Z", 2, 90, 2, "tab", {"005"}
    // "2011-11-14T16:53:41Z", 2, 90, 3, "tab", {"001", "004", "005"}
    // "2011-11-14T16:58:03Z", 2, 90, 5, "visa", {"001"}
    // "2011-11-14T17:07:21Z", 2, 90, 6, "tab", {"004", "005"}
    // "2011-11-14T17:22:59Z", 2, 90, 7, "tab", {"001", "002", "004", "005"}

Now, lets add another dimension and filter and get the combined results of both filters

auto tips = payments.dimension([](auto r) { return r.tip; });
tips.filter(2, 7);                              // filter by range (2 ...... 6.999)
filtered_results = payments.all_filtered();
    // "2011-11-14T16:48:46Z", 2, 90, 2, "tab", {"005"}
    // "2011-11-14T16:53:41Z", 2, 90, 3, "tab", {"001", "004", "005"}
    // "2011-11-14T16:58:03Z", 2, 90, 5, "tab", {"001"}
    // "2011-11-14T17:07:21Z", 2, 90, 6, "visa", {"004", "005"}

A dimension can be a combination of different entries as long the accessor function returs a sortable type. However, you can make type conversion on the fly.

auto products = payments.dimension([](auto r) { return r.productIDS.size(); });
products.filter([](auto d) { return d >= 2; });     // filter by custom function
filtered_results = payments.all_filtered();
    // "2011-11-14T16:53:41Z", 2, 90, 3, "tab", {"001", "004", "005"}
    // "2011-11-14T17:07:21Z", 2, 90, 6, "visa", {"004", "005"}

If you have a type, which is not sortable but you are just interested on specific entries you can use a workaroud to make them formal sortable.

auto no_tabs = payments.dimension([](auto r) { return r.type != std::string("tab") ? 1 : 0; });
no_tabs.filter(1); 
filtered_results = payments.all_filtered();
    // "2011-11-14T17:07:21Z", 2, 90, 6, "visa", {"004", "005"}

Ok, that is just simple filtering, but the real power of cross::filter keeps track of the filters when you change, delete or edit payments.

payments.add({"2011-11-14T17:20:20Z", 4, 90, 2, "cash", {"001", "002"}});
filtered_results = payments.all_filtered();
    // "2011-11-14T17:07:21Z", 2, 90, 6, "visa", {"004", "005"}
    // "2011-11-14T17:20:20Z", 4, 90, 2, "cash", {"001", "002"}

define dimensions

you have to prepare a dimension for everything you want to filter with cross::filter and this dimensions must be numerical sortable

auto dim1 = payments.dimension([](auto r) { return r.quantity;});
auto dim2 = payments.dimension([](auto r) { return timestamp2unixtime(r.date);});

Inside cross::filter a new indexed hashmap is build as kind of ghost copy of the data, which keeps the totals incrementally sorted [90,90,90,90,90,90,100,190,190,300] when changing payments or add a new or delete a record. So every dimension requires additional memmory space. Keep in mind.

filter

You can get a copy of all filtered data with

filtered_results = payments.all_filtered();

however, you can always get a copy of all records by not regarding the filter settings.

auto all_records = filtered_results1 = payments.all();

If you only want the top or the bottom ones, you can use specific methods. But for this methods you can't adress payments directly, because cross::filter can't know the absolute order, if you have set multiplen filters. You have to adress a specic dimension, so cross::filter knows, how to order the filtered records for you.

auto top_filtered1 = totals.top(2);
    //                           ^
    // "2011-11-14T16:54:06Z" 1 100 4 cash {"001", "002", "003", "004", "005"}
    // "2011-11-14T17:20:20Z" 4 90 2 cash {"001", "002"}

auto top_filtered2 = tips.bottom(3);
    //
    // "2011-11-14T17:20:20Z" 4 90 2 cash {"001", "002"}
    // "2011-11-14T16:53:41Z" 2 90 3 tab {"001", "004", "005"}
    // "2011-11-14T16:54:06Z" 1 100 4 cash {"001", "002", "003", "004", "005"}

You can set filters on three different ways.

totals.filter(80) 							            // by value
totals.filter(90,100); 						            // by range (90 <= x < 100)
totals.filter([](auto r) { return (r.total <80); }) 	// with a custom function, which returns a boolean

You can reset the individual filter or change it every time

no_tabs.filter();   // reset
totals.filter(100); // change total filter to 100

 filtered_results = payments.all_filtered();
    // "2011-11-14T16:54:06Z", 1, 100, 4, "cash", {"001", "002", "003", "004", "005"}

grouping (aggregated records)

docu coming soon...

run

you need STL and C++14 support to compile

cross::filter works headonly. just include the header into your project

#include "crossfilter.hpp"

and compile for example with

$ clang++ ./demo/demo1.cpp -std=c++14
$ ./a.out

however tests & benchmark require a build system

$ mkdir build
$ cd build
$ cmake -DCMAKE_BUILD_TYPE=Release ../
$ make (currently broken)
$ make test
$ make benchmark
$ ../benchmark/benchBin/benchmark1

dependencies

crossfilter uses the awesum nod singnal/slot lib. it is included.

About

C++ port of crossfilter.js

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages