青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

loop_in_codes

低調做技術__歡迎移步我的獨立博客 codemaro.com 微博 kevinlynx

基于內存查看STL常用容器內容

有時候在線上使用gdb調試程序core問題時,可能沒有符號文件,拿到的僅是一個內存地址,如果這個指向的是一個STL對象,那么如何查看這個對象的內容呢?

只需要知道STL各個容器的數據結構實現,就可以查看其內容。本文描述了SGI STL實現中常用容器的數據結構,以及如何在gdb中查看其內容。

string

string,即basic_string bits/basic_string.h

mutable _Alloc_hider  _M_dataplus;
    ... 
      const _CharT*
      c_str() const
      { return _M_data(); }
    ...    
      _CharT*
      _M_data() const 
      { return  _M_dataplus._M_p; }

    ...
      struct _Alloc_hider : _Alloc
      {
    _Alloc_hider(_CharT* __dat, const _Alloc& __a)
    : _Alloc(__a), _M_p(__dat) { }

    _CharT* _M_p; // The actual data.
      };
   
      size_type
      length() const
      { return _M_rep()->_M_length; }

      _Rep*
      _M_rep() const
      { return &((reinterpret_cast<_Rep*> (_M_data()))[-1]); }

      ...
       struct _Rep_base
      {
    size_type       _M_length;
    size_type       _M_capacity;
    _Atomic_word        _M_refcount;
      };

      struct _Rep : _Rep_base

即,string內有一個指針,指向實際的字符串位置,這個位置前面有一個_Rep結構,其內保存了字符串的長度、可用內存以及引用計數。當我們拿到一個string對象的地址時,可以通過以下代碼獲取相關值:

void ds_str_i(void *p) {
        char **raw = (char**)p;
        char *s = *raw;
        size_t len = *(size_t*)(s - sizeof(size_t) * 3);
        printf("str: %s (%zd)\n", s, len);
    }

    size_t ds_str() {
        std::string s = "hello";
        ds_str_i(&s);
        return s.size();
    }

在gdb中拿到一個string的地址時,可以以下打印出該字符串及長度:

(gdb) x/1a p
0x7fffffffe3a0: 0x606028
(gdb) p (char*)0x606028
$2 = 0x606028 "hello"
(gdb) x/1dg 0x606028-24
0x606010:       5

vector

眾所周知vector實現就是一塊連續的內存,bits/stl_vector.h

template<typename _Tp, typename _Alloc = std::allocator<_Tp> >
    class vector : protected _Vector_base<_Tp, _Alloc>

    ...
    template<typename _Tp, typename _Alloc>
    struct _Vector_base
    {
      typedef typename _Alloc::template rebind<_Tp>::other _Tp_alloc_type;

      struct _Vector_impl
      : public _Tp_alloc_type
      {
    _Tp*           _M_start;
    _Tp*           _M_finish;
    _Tp*           _M_end_of_storage;
    _Vector_impl(_Tp_alloc_type const& __a)
    : _Tp_alloc_type(__a), _M_start(0), _M_finish(0), _M_end_of_storage(0)
    { }
      };


      _Vector_impl _M_impl;

可以看出sizeof(vector<xxx>)=24,其內也就是3個指針,_M_start指向首元素地址,_M_finish指向最后一個節點+1,_M_end_of_storage是可用空間最后的位置。

iterator
      end()
      { return iterator (this->_M_impl._M_finish); }
      const_iterator
      ...
      begin() const
      { return const_iterator (this->_M_impl._M_start); }
      ...
      size_type
      capacity() const
      { return size_type(const_iterator(this->_M_impl._M_end_of_storage)
             - begin()); }

可以通過代碼從一個vector對象地址輸出其信息:

template <typename T>
    void ds_vec_i(void *p) {
        T *start = *(T**)p;
        T *finish = *(T**)((char*)p + sizeof(void*));
        T *end_storage = *(T**)((char*)p + 2 * sizeof(void*));
        printf("vec size: %ld, avaiable size: %ld\n", finish - start, end_storage - start); 
    }

    size_t ds_vec() {
        std::vector<int> vec;
        vec.push_back(0x11);
        vec.push_back(0x22);
        vec.push_back(0x33);
        ds_vec_i<int>(&vec);
        return vec.size();
    }

使用gdb輸出一個vector中的內容:

(gdb) p p
$3 = (void *) 0x7fffffffe380
(gdb) x/1a p
0x7fffffffe380: 0x606080
(gdb) x/3xw 0x606080
0x606080:       0x00000011      0x00000022      0x00000033

list

眾所周知list被實現為一個鏈表。準確來說是一個雙向鏈表。list本身是一個特殊節點,其代表end,其指向的下一個元素才是list真正的第一個節點:

bits/stl_list.h

bool
      empty() const
      { return this->_M_impl._M_node._M_next == &this->_M_impl._M_node; }

      const_iterator
      begin() const
      { return const_iterator(this->_M_impl._M_node._M_next); }

      iterator
      end()
      { return iterator(&this->_M_impl._M_node); }

      ...

    struct _List_node_base
    {
        _List_node_base* _M_next;   ///< Self-explanatory
        _List_node_base* _M_prev;   ///< Self-explanatory
        ...
    };
         
    template<typename _Tp>
    struct _List_node : public _List_node_base
    {
      _Tp _M_data;                ///< User's data.
    };
      
    template<typename _Tp, typename _Alloc>
    class _List_base
    {
        ...
      struct _List_impl
      : public _Node_alloc_type
      {
    _List_node_base _M_node;
        ...
      };

      _List_impl _M_impl;

          
    template<typename _Tp, typename _Alloc = std::allocator<_Tp> >
    class list : protected _List_base<_Tp, _Alloc>

所以sizeof(list<xx>)=16,兩個指針。每一個真正的節點首先是包含兩個指針,然后是元素內容(_List_node)。

通過代碼輸出list的內容:

#define NEXT(ptr, T) do { \
        void *n = *(char**)ptr; \
        T val = *(T*)((char**)ptr + 2); \
        printf("list item %p val: 0x%x\n", ptr, val); \
        ptr = n; \
    } while (0)

    template <typename T>
    void ds_list_i(void *p) {
        void *ptr = *(char**)p;

        NEXT(ptr, T);
        NEXT(ptr, T);
        NEXT(ptr, T);
    }

    size_t ds_list() {
        std::list<int> lst;
        lst.push_back(0x11);
        lst.push_back(0x22);
        lst.push_back(0x33);
        ds_list_i<int>(&lst);
        return lst.size();
    }

在gdb中可以以下方式遍歷該list:

(gdb) p p
$4 = (void *) 0x7fffffffe390
(gdb) x/1a p
0x7fffffffe390: 0x606080
(gdb) x/1xw 0x606080+16         # 元素1 
0x606090:       0x00000011
(gdb) x/1a 0x606080
0x606080:       0x6060a0
(gdb) x/1xw 0x6060a0+16         # 元素2
0x6060b0:       0x00000022

map

map使用的是紅黑樹實現,實際使用的是stl_tree.h實現:

bits/stl_map.h

typedef _Rb_tree<key_type, value_type, _Select1st<value_type>,
               key_compare, _Pair_alloc_type> _Rep_type;
    ...
     _Rep_type _M_t;
    ... 

      iterator
      begin()
      { return _M_t.begin(); }

bits/stl_tree.h

struct _Rb_tree_node_base
      {
        typedef _Rb_tree_node_base* _Base_ptr;
        typedef const _Rb_tree_node_base* _Const_Base_ptr;

        _Rb_tree_color  _M_color;
        _Base_ptr       _M_parent;
        _Base_ptr       _M_left;
        _Base_ptr       _M_right;
        
        ...
      };

    template<typename _Val>
    struct _Rb_tree_node : public _Rb_tree_node_base
    {
      typedef _Rb_tree_node<_Val>* _Link_type;
      _Val _M_value_field;
    };


    template<typename _Key_compare,
           bool _Is_pod_comparator = std::__is_pod<_Key_compare>::__value>
        struct _Rb_tree_impl : public _Node_allocator
        {
      _Key_compare      _M_key_compare;
      _Rb_tree_node_base    _M_header;
      size_type         _M_node_count; // Keeps track of size of tree.
      ...
        }
    
    _Rb_tree_impl<_Compare> _M_impl;
    ...

      iterator
      begin()
      {
    return iterator(static_cast<_Link_type>
            (this->_M_impl._M_header._M_left));
      }

所以可以看出,大部分時候(取決于_M_key_compare) sizeof(map<xx>)=48,主要的元素是:

_Rb_tree_color  _M_color; // 節點顏色
        _Base_ptr       _M_parent; // 父節點
        _Base_ptr       _M_left; // 左節點
        _Base_ptr       _M_right; // 右節點
        _Val            _M_value_field // 同list中節點技巧一致,后面是實際的元素

同list中的實現一致,map本身作為一個節點,其不是一個存儲數據的節點,

_Rb_tree::end

iterator
      end()
      { return iterator(static_cast<_Link_type>(&this->_M_impl._M_header)); }

由于節點值在_Rb_tree_node_base后,所以任意時候拿到節點就可以偏移這個結構體拿到節點值,節點的值是一個pair,包含了key和value。

在gdb中打印以下map的內容:

size_t ds_map() {
        std::map<std::string, int> imap;
        imap["abc"] = 0xbbb;
        return imap.size();
    }
(gdb) p/x &imap
$7 = 0x7fffffffe370
(gdb) x/1a (char*)&imap+24       # _M_left 真正的節點
0x7fffffffe388: 0x606040          
(gdb) x/1xw 0x606040+32+8        # 偏移32字節是節點值的地址,再偏移8則是value的地址
0x606068:       0x00000bbb
(gdb) p *(char**)(0x606040+32)   # 偏移32字節是string的地址
$8 = 0x606028 "abc"

或者很多時候沒有必要這么裝逼+蛋疼:

(gdb) p *(char**)(imap._M_t._M_impl._M_header._M_left+1)
$9 = 0x606028 "abc"
(gdb) x/1xw (char*)(imap._M_t._M_impl._M_header._M_left+1)+8
0x606068:       0x00000bbb

posted on 2014-12-03 22:08 Kevin Lynx 閱讀(3870) 評論(2)  編輯 收藏 引用 所屬分類: c/c++

評論

# re: 基于內存查看STL常用容器內容 2014-12-04 18:57 xxoo

問一下,既然string的結構類型是
(size_t,size_t,size_t,char*)
為啥string地址里前4個自己是char*的地址,應該是size_t的前4個字節才對啊?  回復  更多評論   

# re: 基于內存查看STL常用容器內容 2014-12-07 18:27 Kevin Lynx

@xxoo
"即,string內有一個指針,指向實際的字符串位置,這個位置前面有一個_Rep結構,其內保存了字符串的長度、可用內存以及引用計數。"  回復  更多評論   

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲国产精品久久精品怡红院| 亚洲视频免费看| 亚洲二区精品| 黄色成人片子| 精品91免费| 亚洲风情在线资源站| 国内激情久久| 在线视频观看日韩| 亚洲黄色一区二区三区| 亚洲欧洲一区二区在线观看| 亚洲国产一区二区视频| 日韩午夜免费视频| 亚洲欧美99| 久久色在线观看| 亚洲国产综合视频在线观看 | 亚洲精品乱码久久久久久黑人| 亚洲福利精品| 亚洲校园激情| 久久免费精品视频| 亚洲福利av| 亚洲午夜在线| 牛牛影视久久网| 国产精品久久| 亚洲激情午夜| 午夜精品久久久久久久| 你懂的国产精品永久在线| 亚洲激情网站| 欧美亚洲一区| 欧美日韩国产麻豆| 国产视频亚洲精品| 一区二区三区日韩在线观看| 久久久久中文| 一区二区日韩伦理片| 亚洲精品国精品久久99热| 欧美一区二区在线观看| 久久亚洲一区二区三区四区| 亚洲精品午夜| 久久婷婷综合激情| 国产精品亚洲欧美| 一区二区三区视频观看| 六月天综合网| 午夜精彩视频在线观看不卡| 欧美精品在线免费| 18成人免费观看视频| 小黄鸭精品aⅴ导航网站入口| 欧美激情麻豆| 久久免费的精品国产v∧| 欧美视频在线一区二区三区| 亚洲国产日韩精品| 久久综合九色综合欧美就去吻 | 亚洲精品免费在线播放| 久久免费99精品久久久久久| 宅男噜噜噜66一区二区66| 欧美高清在线播放| 亚洲激情一区| 麻豆精品视频在线观看| 西西人体一区二区| 国产精品亚洲一区| 亚洲欧美日韩国产中文| 99在线观看免费视频精品观看| 麻豆精品在线播放| 1000部国产精品成人观看| 久久香蕉国产线看观看av| 性做久久久久久免费观看欧美| 欧美色综合天天久久综合精品| 亚洲精品久久久久久下一站| 欧美成人精品高清在线播放| 久久久夜精品| 在线观看亚洲视频啊啊啊啊| 看欧美日韩国产| 久久视频在线看| 亚洲国产精品一区二区www| 欧美大成色www永久网站婷| 久久久在线视频| 亚洲人成网站在线播| 最近中文字幕日韩精品| 欧美日韩国产高清| 亚洲欧美综合v| 欧美在线观看视频| 亚洲第一区中文99精品| 亚洲国产裸拍裸体视频在线观看乱了| 另类av导航| 99国产精品久久久久久久| 亚洲精选在线| 国产精品亚洲美女av网站| 久久综合九色| 欧美精品导航| 午夜久久福利| 久久免费高清| 亚洲午夜久久久久久久久电影网| av成人免费在线观看| 午夜亚洲性色福利视频| 一区二区三区我不卡| 亚洲黄色影院| 国产精品永久入口久久久| 久热精品视频在线观看| 欧美理论在线| 欧美中文字幕第一页| 另类国产ts人妖高潮视频| 宅男66日本亚洲欧美视频 | 欧美va天堂va视频va在线| 亚洲专区在线| 久久久亚洲人| 亚洲性av在线| 久久综合给合| 午夜精品久久久久久久99水蜜桃 | 欧美在线免费视频| 亚洲九九九在线观看| 午夜精品久久久99热福利| 亚洲国产黄色| 午夜精品电影| 一本色道久久综合亚洲精品高清| 香蕉尹人综合在线观看| 亚洲麻豆视频| 久久精品亚洲一区| 亚洲综合电影一区二区三区| 久久男人资源视频| 性色av一区二区三区| 欧美美女视频| 欧美成人xxx| 国产在线拍偷自揄拍精品| 亚洲视频免费观看| 一区二区三区 在线观看视| 久久在线免费| 久久久久免费视频| 国产精品影音先锋| 亚洲一区激情| 亚洲欧美日韩成人高清在线一区| 蜜桃av一区二区| 美女主播视频一区| 好男人免费精品视频| 亚洲欧美另类综合偷拍| 午夜精品短视频| 国产精品久久久久久久9999| 亚洲精品国产精品乱码不99按摩 | 18成人免费观看视频| 性久久久久久久久久久久| 亚洲一区区二区| 欧美极品aⅴ影院| 欧美激情免费观看| 亚洲第一福利社区| 六月婷婷一区| 欧美a一区二区| 亚洲三级国产| 欧美成黄导航| 日韩西西人体444www| 99热免费精品| 欧美日韩一区二区三区免费| 亚洲麻豆视频| 亚洲欧美成人一区二区在线电影| 欧美三级午夜理伦三级中视频| 日韩午夜电影| 国产久一道中文一区| 亚洲天堂网在线观看| 亚洲小说区图片区| 国产精品亚洲综合一区在线观看| 亚洲女人天堂成人av在线| 欧美在线免费观看亚洲| 国内精品久久久久影院 日本资源| 久久国产精品一区二区三区| 每日更新成人在线视频| 亚洲精品免费在线播放| 欧美日韩免费观看一区二区三区| 日韩系列在线| 久久国产精品一区二区三区四区| 国产一区99| 欧美wwwwww| 亚洲色图综合久久| 久久精品视频免费播放| 亚洲国产成人porn| 欧美日韩一区在线| 欧美一级艳片视频免费观看| 欧美成年人在线观看| 在线亚洲激情| 国产偷国产偷亚洲高清97cao| 久久天天躁狠狠躁夜夜爽蜜月| 欧美大片免费久久精品三p | 亚洲一区在线视频| 久久亚洲欧美| 9久re热视频在线精品| 国产精品一区二区久久久久| 久久精品视频导航| 日韩视频精品| 榴莲视频成人在线观看| 亚洲一区欧美一区| 在线观看亚洲视频啊啊啊啊| 欧美色大人视频| 久久久久免费视频| 中文在线一区| 亚洲国产成人91精品| 久久丁香综合五月国产三级网站| 亚洲人在线视频| 国产日韩欧美一区二区| 欧美精品少妇一区二区三区| 欧美一级视频免费在线观看| 日韩天堂在线观看| 欧美高清在线| 另类图片国产| 久久九九精品99国产精品| 亚洲一区在线免费|