Подход к проектированию, данные таблицы строк, переменные, использование памяти stl - PullRequest
0 голосов
/ 09 марта 2010

У меня есть старый класс структуры, подобный этому: typedef vector<vector<string>> VARTYPE_T;, который работает как одна переменная. Эта переменная может содержать от одного значения в списке до данных, подобных таблице. Большинство значений long, double, string или double [3] для координат (x, y, z). Я просто конвертирую их по мере необходимости. Переменные управляются на карте следующим образом: map<string,VARTYPE_T *> где строка содержит имя переменной. Конечно, они завернуты в классы. Также у меня есть дерево узлов, где каждый узел может содержать одну из этих карт переменных.

Используя VS 2008 SP1 для этого, я обнаружил много фрагментации памяти. Сравнивая со stlport, stlport казался быстрее (20%) и использует меньше памяти (30%, для моих тестовых случаев).

Итак, вопрос: Какова лучшая реализация, чтобы быстро решить эту проблему с правильно используемой памятью? Должен ли я написать собственный распределитель, как распределитель пула. Как бы вы это сделали?

Заранее спасибо,

Хови

1 Ответ

0 голосов
/ 09 марта 2010

Измените typedef vector<vector<string>> VARTYPE_T; на typedef deque<deque<string>> VARTYPE_T; и проверьте, существует ли еще фрагментация памяти.

Кстати, как вы оцениваете это с VS 2008 SP1?

Обновление: Я знаю, как HP справляется с фрагментацией памяти HP-UX. После поиска я нашел интересную ссылку Куча с низкой фрагментацией . Это цитата: The low-fragmentation heap (LFH) helps to reduce heap fragmentation. Мне это интересно, так как оно напоминает подход HP к фрагментации памяти, судя по его описанию. Я думаю (кроме использования deque) это еще одна хорошая идея, чтобы попробовать и проверить. `

Обновление 2: 1) Скорость. Раньше ты ничего не говорил о скорости, поэтому у меня нет никаких советов, чтобы дать тебе хороший совет. Я также не знаю, где ваша программа проводит большую часть времени, выполняя.

Если вы считаете, что он в основном тратит время на преобразование строк в длинные и двойные, вам придется выполнять это преобразование только один раз и использовать его при доступе к значениям переменных. Вероятно, это хорошая идея хранить не строки, а реальные значения. Например, в союзе или Boost.Variant. Если вы считаете, что в переменной отсутствуют индексы, которые обеспечивают быстрый доступ к ее значениям, вам следует добавить эти индексы.

2) Использование памяти. Вы на самом деле пишете серверное приложение, работающее в течение нескольких дней, и вас должно беспокоить потребление памяти? Тогда я уже говорил вам, используйте Low-fragmentation Heap в Windows и попробуйте использовать размеры блоков меньше 16K. Также избавьтесь от утечек памяти. Конечно, проверьте это, но, насколько я понимаю, Low-fragmentation Heap должен обрабатывать ваши новости.

3) Если вы пишете серверное приложение, тогда std :: vector иногда не является хорошим выбором. Он выделяет память в одном блоке. Если оно больше 16 КБ, это может привести к фрагментации памяти.

4) Наконец, посмотрите, как выглядит хороший отчет от HP-UX о распределении памяти (чтобы найти фрагментацию памяти):

Actual Heap Usage:
    Heap Start  =   0x60000000000fea38
    Heap End    =   0x6000000026580000
    Heap Size   =   642258376 bytes

Outstanding Allocations:
    251948524 bytes allocated in 4239425 blocks
                              Detailed Report

-------------------------------------------------------------------------
65343264 bytes in 1361318 blocks (25.94% of all bytes allocated)
These range in size from 48 to 48 bytes and are allocated
#0  stlp_std::__malloc_alloc::allocate(unsigned long&)   from ./libstlport.so.5.1
#1  boost::multi_index::detail::ordered_index<boost::multi_index::identity<csubs::clnt_tax_hist_t>, stlp_std::less<csubs::clnt_tax_hist_t>, boost::multi_index::detail::nth_layer<1, csubs::clnt_tax_hist_t, boost::multi_index::indexed_by<boost::multi_index::ordered_non_unique<boost::multi_index::identity<csubs::clnt_tax_hist_t>, mpl_::na, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, hrs_allocator::counting_allocator<csubs::clnt_tax_hist_t, (hrs_allocator::allocator_enums)9> >, boost::mpl::vector0<mpl_::na>, boost::multi_index::detail::ordered_non_unique_tag>::insert(csubs::clnt_tax_hist_t const&)   at _alloc.h:381



-------------------------------------------------------------------------
47510512 bytes in 848402 blocks (18.86% of all bytes allocated)
These range in size from 56 to 56 bytes and are allocated
#0  stlp_std::__malloc_alloc::allocate(unsigned long&)   from ./libstlport.so.5.1
#1  csubs::cache_impl<csubs::subs_data_t, csubs::search_policy_range<false>, csubs::erase_policy_range, hrs_allocator::counting_allocator<csubs::subs_data_t, (hrs_allocator::allocator_enums)5>, boost::multi_index::multi_index_container<csubs::subs_data_t, boost::multi_index::indexed_by<boost::multi_index::ordered_non_unique<boost::multi_index::identity<csubs::subs_data_t>, mpl_::na, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, hrs_allocator::counting_allocator<csubs::subs_data_t, (hrs_allocator::allocator_enums)5> > >::put(csubs::subs_data_t const&)   at _alloc.h:381
#2  csubs::db_cache_loader<csubs::cache_impl<csubs::subs_data_t, csubs::search_policy_range<false>, csubs::erase_policy_range, hrs_allocator::counting_allocator<csubs::subs_data_t, (hrs_allocator::allocator_enums)5>, boost::multi_index::multi_index_container<csubs::subs_data_t, boost::multi_index::indexed_by<boost::multi_index::ordered_non_unique<boost::multi_index::identity<csubs::subs_data_t>, mpl_::na, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, hrs_allocator::counting_allocator<csubs::subs_data_t, (hrs_allocator::allocator_enums)5> > >, csubs::ctrl_loader_nocheck>::operation()   at cache_subs_loaders.h:121
#3  csubs::group_subs_cache::load_caches_from_db(db_date const&, csubs::cache_options const&, otl_connect&, csubs::cache_stat*)   at ./caches/cache_subs_caches.cpp:1452

и т. Д.

...