我有一个 std::map,我在其中使用 shared_ptrs 作为键:
std::map<shared_ptr<object>, shared_ptr<object>>
但是,在调试时,我无法使用macOS上的lldb或Linux上的gdb打开这些映射。使用 gdb 在 Ubuntu 上,通过修改文件 libstdcxx/v6/printers.py并将迭代器添加到 SharedPointerPrinter 类(代码如下所示(修复了此问题。但是,我不知道如何为 lldb 解决此问题。如果有人能帮助我或为我指出正确的方向,那就太好了。
如果您需要其他信息来帮助我解决此问题,请告诉我,我很乐意提供。
class SharedPointerPrinter:
"Print a shared_ptr or weak_ptr"
class _iterator:
def __init__(self, sharedPointer):
self.sharedPointer = sharedPointer
self.managedValue = sharedPointer.val['_M_ptr']
self.count = 0
def __iter__(self):
return self
def next(self):
if self.managedValue == 0:
raise StopIteration
self.count = self.count + 1
if (self.count == 1):
return ('Use count', self.sharedPointer.val['_M_refcount']['_M_pi']['_M_use_count'])
elif (self.count == 2):
return ('Weak count', self.sharedPointer.val['_M_refcount']['_M_pi']['_M_weak_count'] - 1)
elif (self.count == 3):
return ('Managed value', self.managedValue)
else:
raise StopIteration
def __init__ (self, typename, val):
self.typename = typename
self.val = val
def children (self):
return self._iterator(self)
def to_string (self):
state = 'empty'
refcounts = self.val['_M_refcount']['_M_pi']
if refcounts != 0:
usecount = refcounts['_M_use_count']
weakcount = refcounts['_M_weak_count']
if usecount == 0:
state = 'expired, weakcount %d' % weakcount
else:
state = 'usecount %d, weakcount %d' % (usecount, weakcount - 1)
return '%s (%s) to %s' % (self.typename, state, self.val['_M_ptr'])
您可以使用lldb的Python数据格式化程序执行几乎相同的操作,如本页所述:
http://lldb.llvm.org/varformats.html
LLDB有std::map的数据格式化程序和clang的STL版本的std::shared_ptr。 例如,我看到:
(lldb) fr v a_map
(std::__1::map<std::__1::shared_ptr<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >, std::__1::shared_ptr<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >, std::__1::less<std::__1::shared_ptr<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > > >, std::__1::allocator<std::__1::pair<const std::__1::shared_ptr<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >, std::__1::shared_ptr<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > > > > >) a_map = size=1 {
[0] = {
first = "some text" strong=2 weak=1 {
__ptr_ = "some text"
}
second = "other text" strong=2 weak=1 {
__ptr_ = "other text"
}
}
}
伙计,那些C++类型很啰嗦!
GCC 的 STL 也有一些格式化程序,但我对这些格式化程序的状态知之甚少。
IIUC 数据格式化程序可以在 gdb 中的表达式中使用。 在 lldb 中,frame variable
命令将接受简单的结构访问表达式,因此您可以执行以下操作:
(lldb) frame var a_map[0].first
(const std::__1::shared_ptr<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >) a_map[0].first = "some text" strong=2 weak=1 {
__ptr_ = "some text"
}
但是,不能在完整表达式分析器中使用格式化程序生成的子级数据。