2008-11-26 3 views
15

Я хочу изучить содержимое std :: vector в gdb, но у меня нет доступа к _M_impl, потому что я использую icc, а не gcc, как это сделать? Скажем, это просто std :: vector для простоты.Как просмотреть содержимое std :: vector в gdb, используя компилятор icc?

Существует очень хороший ответ here, но это не сработает, если я использую icc, сообщение об ошибке «Нет члена или метода с именем _M_impl». Как представляется, хороший инструмент для отладки here, но он также полагается на _M_impl.

+2

+1 лишний стыд, который 7 лет спустя, gdb все еще не может отображать контейнеры разумным способом. даже строки ... – 2015-10-01 19:04:56

+0

no icc ограничение: https://stackoverflow.com/questions/253099/how-do-i-print-the-elements-of-ac-vector-in-gdb – 2017-09-24 10:28:19

ответ

10

Не уверен, что это сработает с вашим вектором, но это сработало для меня.

#include <string> 
#include <vector> 

int main() { 
    std::vector<std::string> vec; 
    vec.push_back("Hello"); 
    vec.push_back("world"); 
    vec.push_back("!"); 
    return 0; 
} 

GDB:

(gdb) break source.cpp:8 
(gdb) run 
(gdb) p vec.begin() 
$1 = { 
    _M_current = 0x300340 
} 
(gdb) p $1._M_current->c_str() 
$2 = 0x3002fc "Hello" 
(gdb) p $1._M_current +1 
$3 = (string *) 0x300344 
(gdb) p $3->c_str() 
$4 = 0x30032c "world" 
0

std::vector template guarantees the data is stored contiguously. Если вы берете адрес переднего элемента (например, &v[0], например), вы можете получить доступ к любому другому элементу вектора через массив C-стиля. Это не требует, чтобы исходный код STL был доступен вашему отладчику.


После баловаться с этим некоторые, кажется, что v.front() и v.begin(), вероятно, встраиваемыми и GDB не находит их. Я буду продолжать искать, но лично я просто добавлю строку int* i = &v[0] в исходный файл, а затем буду использовать команды GDB на i во время отладки. Обратите внимание, что компилятор может удалить этот мертвый код. Возможно, вам понадобится вывести значение i, чтобы избежать этого, или просто не проверять оптимизацию.

3

Вообще, когда я имею дело с контейнерными классами в отладчике, я создаю ссылку на элемент, в качестве локальной переменной, так что легко видеть в отладчике , не обманывая в реализации контейнера.

Вот надуманный пример.

vector<WeirdStructure> myWeird; 

/* push back a lot of stuff into the vector */ 

size_t z; 
for (z = 0; z < myWeird.size(); z++) 
{ 
    WeirdStructure& weird = myWeird[z]; 

    /* at this point weird is directly observable by the debugger */ 

    /* your code to manipulate weird goes here */ 
} 

Это идиома, которую я использую.