Мне нужно реализовать очень простой кэш LRU, в котором хранятся адреса памяти.
- Число этих адресов фиксировано (во время выполнения).
- Меня интересует только последний использованный адрес (меня не волнует порядок других элементов).
- Каждый адрес имеет соответствующий порядковый номер (простое целое число), который не является уникальным и может изменяться.
Реализация должна выполняться с как можно меньшими издержками. В дополнение к каждому адресу есть также структура связанной информации (которая содержит индекс).
Мой текущий подход использует std::list
для хранения пары адрес / информация и boost::unordered_multimap
, который является отображением между индексом и связанным итератором списка.
Следующий пример не имеет ничего общего с моим рабочим кодом. Обратите внимание, что это только для лучшего понимания.
struct address_info
{
address_info() : i(-1) {}
int i;
// more ...
};
int main()
{
int const MAX_ADDR_COUNT = 10,
MAX_ADDR_SIZE = 64;
char** s = new char*[MAX_ADDR_COUNT];
address_info* info = new address_info[MAX_ADDR_COUNT]();
for (int i = 0; i < MAX_ADDR_COUNT; ++i)
s[i] = new char[MAX_ADDR_SIZE]();
typedef boost::unordered_multimap<int, std::list<std::pair<address_info, char*>>::const_iterator> index_address_map;
std::list<std::pair<address_info, char*>> list(MAX_ADDR_COUNT);
index_address_map map;
{
int i = 0;
for (std::list<std::pair<address_info, char*>>::iterator iter = list.begin(); i != MAX_ADDR_COUNT; ++i, ++iter)
*iter = std::make_pair(info[i], s[i]);
}
// usage example:
// try to find address_info 4
index_address_map::const_iterator iter = map.find(4);
if (iter == map.end())
{
std::pair<address_info, char*>& lru = list.back();
if (lru.first.i != -1)
map.erase(lru.first.i);
lru.first.i = 4;
list.splice(list.begin(), list, boost::prior(list.end()));
map.insert(std::make_pair(4, list.begin()));
}
else
list.splice(list.begin(), list, iter->second);
for (int i = 0; i < MAX_ADDR_COUNT; ++i)
delete[] s[i];
delete[] info;
delete[] s;
return 0;
}