mirror of
https://github.com/RobTillaart/Arduino.git
synced 2024-10-03 18:09:02 -04:00
259 lines
4.1 KiB
C++
259 lines
4.1 KiB
C++
|
//
|
||
|
// FILE: Set.cpp
|
||
|
// AUTHOR: Rob Tillaart
|
||
|
// VERSION: 0.1.06
|
||
|
// PURPOSE: SET library for Arduino
|
||
|
// URL:
|
||
|
//
|
||
|
// HISTORY:
|
||
|
// 0.1.06 added flag to constructor to optimize +,-,*,
|
||
|
// set -> Set
|
||
|
// 0.1.05 bug fixing + performance a.o. count()
|
||
|
// 0.1.04 support for + - *, some optimizations
|
||
|
// 0.1.03 changed &= to *= to follow Pascal conventions
|
||
|
// 0.1.02 documentation
|
||
|
// 0.1.01 extending/refactor etc (09/11/2014)
|
||
|
// 0.1.00 initial version by Rob Tillaart (09/11/2014)
|
||
|
//
|
||
|
// Released to the public domain
|
||
|
//
|
||
|
|
||
|
#include "Set.h"
|
||
|
|
||
|
/////////////////////////////////////////////////////
|
||
|
//
|
||
|
// CONSTRUCTORS
|
||
|
//
|
||
|
Set::Set(bool clear)
|
||
|
{
|
||
|
if (clear)
|
||
|
{
|
||
|
clr();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Set::Set(Set &t)
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
_mem[i] = t._mem[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////
|
||
|
//
|
||
|
// METHODS
|
||
|
//
|
||
|
void Set::add(uint8_t v)
|
||
|
{
|
||
|
uint8_t mask = 1 << (v & 7);
|
||
|
uint8_t idx = v / 8;
|
||
|
_mem[idx] |= mask;
|
||
|
}
|
||
|
|
||
|
void Set::sub(uint8_t v)
|
||
|
{
|
||
|
uint8_t mask = 1 << (v & 7);
|
||
|
uint8_t idx = v / 8;
|
||
|
_mem[idx] &= ~mask;
|
||
|
}
|
||
|
|
||
|
void Set::invert(uint8_t v)
|
||
|
{
|
||
|
uint8_t mask = 1 << (v & 7);
|
||
|
uint8_t idx = v / 8;
|
||
|
_mem[idx] ^= mask;
|
||
|
}
|
||
|
|
||
|
bool Set::has(uint8_t v)
|
||
|
{
|
||
|
uint8_t mask = 1 << (v & 7);
|
||
|
uint8_t idx = v / 8;
|
||
|
return (_mem[idx] & mask) > 0;
|
||
|
}
|
||
|
|
||
|
// TODO OPTIMIZE COUNT
|
||
|
uint8_t Set::count()
|
||
|
{
|
||
|
uint8_t cnt = 0;
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
// kerningham bit count trick
|
||
|
uint8_t b = _mem[i];
|
||
|
for (; b; cnt++)
|
||
|
{
|
||
|
b &= b-1;
|
||
|
}
|
||
|
}
|
||
|
return cnt;
|
||
|
}
|
||
|
|
||
|
void Set::clr()
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
_mem[i] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Set::invert()
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
_mem[i] ^= 0xFF;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int Set::first()
|
||
|
{
|
||
|
for (int i = 0; i < 256; i++)
|
||
|
{
|
||
|
if (has(i))
|
||
|
{
|
||
|
_current = i;
|
||
|
return _current;
|
||
|
}
|
||
|
}
|
||
|
_current = -1;
|
||
|
return _current;
|
||
|
}
|
||
|
|
||
|
int Set::next()
|
||
|
{
|
||
|
if (_current != -1)
|
||
|
{
|
||
|
_current++;
|
||
|
for (int i = _current; i < 256; i++)
|
||
|
{
|
||
|
if (has(i))
|
||
|
{
|
||
|
_current = i;
|
||
|
return _current;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
_current = -1;
|
||
|
return _current;
|
||
|
}
|
||
|
|
||
|
int Set::prev()
|
||
|
{
|
||
|
if (_current != -1)
|
||
|
{
|
||
|
_current--;
|
||
|
for (int i = _current; i > -1; --i)
|
||
|
{
|
||
|
if (has(i))
|
||
|
{
|
||
|
_current = i;
|
||
|
return _current;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
_current = -1;
|
||
|
return _current;
|
||
|
}
|
||
|
|
||
|
int Set::last()
|
||
|
{
|
||
|
_current = -1;
|
||
|
for (int i = 255; i >=0; --i)
|
||
|
{
|
||
|
if (has(i))
|
||
|
{
|
||
|
_current = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return _current;
|
||
|
}
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////
|
||
|
//
|
||
|
// OPERATORS
|
||
|
//
|
||
|
|
||
|
Set Set::operator + (Set &t) // union
|
||
|
{
|
||
|
Set s(false);
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
s._mem[i] = this->_mem[i] | t._mem[i];
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
Set Set::operator - (Set &t) // diff
|
||
|
{
|
||
|
Set s(false);
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
s._mem[i] = this->_mem[i] & ~t._mem[i];
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
Set Set::operator * (Set &t) // intersection
|
||
|
{
|
||
|
Set s(false);
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
s._mem[i] = this->_mem[i] & t._mem[i];
|
||
|
}
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
void Set::operator += (Set &t) // union
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
_mem[i] |= t._mem[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Set::operator -= (Set &t) // diff
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
_mem[i] &= ~t._mem[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void Set::operator *= (Set &t) // intersection
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
_mem[i] &= t._mem[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool Set::operator == (Set &t) // equal
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
if (_mem[i] != t._mem[i]) return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool Set::operator != (Set &t) // not equal
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
if (_mem[i] != t._mem[i]) return true;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool Set::operator <= (Set &t) // subSet
|
||
|
{
|
||
|
for (uint8_t i=0; i<32; i++)
|
||
|
{
|
||
|
if ((_mem[i] & ~t._mem[i]) > 0) return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
//
|
||
|
// END OF FILE
|
||
|
//
|