Мне тяжело с multi_index_container.
Это в основном то, что мне нужно:
- Управление виртуальным списком.
- Сортировка элементов внесколькими способами (поэтому я хотел использовать multi_index_container).
- Произвольный доступ к элементам в зависимости от того, как они сортируются и отображаются в элементе управления списком.
- Сохранение исходного порядка вставки / записи элементов.
- Перемещение элементов, вставка и удаление элементов в элементе управления списком (и соответствующим образом обновлять контейнер).
До сих пор я использовал несколько векторов / карт в синхронизации (отображениепозиции списка, сопоставленные с реальными данными и т. д.), поэтому было бы очень хорошо перейти к multi_index_container.
Я попытался написать минимальную версию своего исходного кода (или пытается написать ее).
Большое спасибо за любую помощь в этом!:)
#include <boost/multi_index_container.hpp>
#include <boost/multi_index/member.hpp>
#include <boost/multi_index/ordered_index.hpp>
#include <boost/multi_index/random_access_index.hpp>
#include <boost/multi_index/sequenced_index.hpp>
#include <cstring>
namespace bmi = boost::multi_index;
typedef unsigned int uint;
struct Item_t
{
public:
Item_t(std::wstring name, uint quantity) : Name(name), Quantity(Quantity) {}
std::wstring Name;
uint Quantity;
// more members here, such as item type, category, etc
};
struct record_order;
struct name_order;
struct quantity_order;
typedef bmi::multi_index_container<
Item_t,
bmi::indexed_by<
bmi::random_access<>,
bmi::sequenced<bmi::tag<record_order>>,
bmi::ordered_non_unique<
bmi::tag<name_order>,
bmi::member<Item_t, std::wstring, &Item_t::Name>
>,
bmi::ordered_non_unique<
bmi::tag<quantity_order>,
bmi::member<Item_t, uint, &Item_t::Quantity>
>
>
> ItemContainer;
void populateContainer(ItemContainer& container)
{
container.push_back(Item_t(L"Salmon roll", 3));
container.push_back(Item_t(L"Chinese cola", 0));
container.push_back(Item_t(L"Norwegian cap", 1));
container.push_back(Item_t(L"Like-new socks", 3));
container.push_back(Item_t(L"Empty bottle", 4));
container.push_back(Item_t(L"Nice tie", 1));
/* sorted:
Chinese cola
Empty bottle
Like-new socks
Nice tie
Norwegian cap
Salmon roll
*/
}
int wmain(int argc, wchar_t* argv[])
{
ItemContainer container;
populateContainer(container);
// sort items to be displayed to the user, by Item_t::Name
// Please see my comment regarding this
container.rearrange(container.get<name_order>().begin());
for (ItemContainer::iterator it = container.begin(), end_ = container.end(); it != end_; ++it)
{
std::wcout << it->Name << std::endl;
}
std::wcout << std::endl;
{
// get Item_t from container where position in displayed list (after sort) is 5
const Item_t& item = *(container.begin() + 5);
// "Salmon roll"
std::wcout << item.Name << std::endl;
}
{
// TODO: insert and sort automatically: Item_t(L"Another useless thing", 1)
const Item_t& item = *(container.begin() + 5);
// Need this to be "Norwegian cap"
std::wcout << item.Name << std::endl;
}
return 0;
}