全局变量相互依赖和初始化顺序的解决办法

简介: 如果是定义一个全局的map,会出现如下core: Program received signal SIGSEGV, Segmentation fault. 0x00007ffff7b449ea in std::_Rb_tree_decreme...
如果是定义一个全局的map,会出现如下core:

Program received signal SIGSEGV, Segmentation fault.
0x00007ffff7b449ea in std::_Rb_tree_decrement(std::_Rb_tree_node_base*) () from /usr/lib/x86_64-linux-gnu/libstdc++.so.6
(gdb) bt
#0  0x00007ffff7b449ea in std::_Rb_tree_decrement(std::_Rb_tree_node_base*) () from /usr/lib/x86_64-linux-gnu/libstdc++.so.6
#1  0x0000000000404071 in std::_Rb_tree_iterator<:pair const int> >::operator-- (this=0x7fffffffe330) at /usr/include/c++/4.6/bits/stl_tree.h:203
#2  0x00000000004037b2 in std::_Rb_tree, std::_Select1st<:pair const int> >, std::less, std::allocator<:pair const int> > >::_M_insert_unique (this=0x607280, __v=...) at /usr/include/c++/4.6/bits/stl_tree.h:1287
#3  0x000000000040331b in std::map, std::allocator<:pair const int> > >::insert (this=0x607280, __x=...)
    at /usr/include/c++/4.6/bits/stl_map.h:518
#4  0x0000000000402818 in ArgsParser::register_arg (param_name=..., arg_info=0x6080a0) at ./args_parser.cpp:130
#5  0x00000000004017d6 in util::CArgInfo<:basic_string :char_traits>, std::allocator > >::CArgInfo (this=0x6080a0, param_name=...)
    at /data1/mooon/run/include/util/args_parser.h:307
#6  0x00000000004016b3 in util::CStringArgInfo<:basic_string :char_traits>, std::allocator > >::CStringArgInfo (this=0x6080a0, 
    optional=false, param_name=..., default_value=..., help_string=...) at /data1/mooon/run/include/util/args_parser.h:392
#7  0x000000000040143d in __static_initialization_and_destruction_0 (__initialize_p=1, __priority=65535) at x.cpp:7
#8  0x0000000000401523 in _GLOBAL__sub_I__ZN10ArgsParser2ipE () at x.cpp:12
#9  0x0000000000404a8d in __libc_csu_init ()
#10 0x00007ffff7528700 in __libc_start_main (main=0x401374 , argc=1, ubp_av=0x7fffffffe6a8, init=0x404a30 <__libc_csu_init>, fini=, 
    rtld_fini=, stack_end=0x7fffffffe698) at libc-start.c:185
#11 0x00000000004012b9 in _start ()

问题弄清楚了,是因为全局变量相互依赖的问题,被依赖的晚于初始化造成的(如全局对象A依赖于全局对象B,而B晚于A被构造出来)这和编译器版本有关,因为之前一直未发现这个问题。解决办法是,将被依赖的改成局部静态变量,通过函数方式引用,如:

原来为:static string g_info;
改成:
string& info()
{
    static string s_info; // 局部静态对象,只有在使用到时才会被构造出来,由于是在全局对象构造中被调用到,所以不存在多线程问题,因为这发生在main函数之前,其它线程还没有创建出来,如果这个时候就有线程了,那就考虑是否设计合理。
    return s_info;
}

问题就可以解决了。
相关文章
|
7月前
|
C++
全局变量初始化顺序探究
全局变量初始化顺序探究
|
5月前
|
编译器 C语言 C++
一分钟搞懂什么是this指针(未涉及静态成员和函数)
一分钟搞懂什么是this指针(未涉及静态成员和函数)
|
8月前
|
存储 C++
C++基础语言之(一)static关键字的作用
C++基础语言之(一)static关键字的作用
110 0
|
存储 程序员 编译器
容易混淆的基本概念 成员变量 局部变量 全局变量
在实际开发与学习中,特别容易混淆几个基本概念:成员变量、局部变量、全局变量。了解这些概念的属性,存储在实际编码中非常有用。
110 0
容易混淆的基本概念 成员变量 局部变量 全局变量
类对象与静态,局部,全局变量的关系
自行编写代码,检验以下三种对象的构造与析构函数的执行顺序 (1)全局对象的构造函数在文件中的所有函数(包括main函数)执行之前调用。 (2) 局部自动对象 (3) 静态(static)局部对象
161 0
类对象与静态,局部,全局变量的关系
|
数据库
局部变量,全局变量怎么用合适?
局部变量,全局变量怎么用合适?
232 0
局部变量,全局变量怎么用合适?
|
Java 编译器 C++
【C++ 语言】面向对象 ( 类定义 | 限制头文件引用次数 | 构造方法 | 析构方法 )
【C++ 语言】面向对象 ( 类定义 | 限制头文件引用次数 | 构造方法 | 析构方法 )
165 0
【C++ 语言】面向对象 ( 类定义 | 限制头文件引用次数 | 构造方法 | 析构方法 )
类、变量、块、构造器、继承初始化顺序,终极解答。
最近发现微信群里面有些群友在讨论类的初始化顺序,如类的静态变量、成员变量、静态代码块、非静态代码块、构造器,及继承父类时,它们的初始化顺序都是怎样的,下面我通过例子来说明这个情况,以免被人误导。
类、变量、块、构造器、继承初始化顺序,终极解答。
有继承关系的对象执行顺序,包括静态变量,静态代码块,普通变量,普通代码块,继承方法.
static最先执行,如果生成的是子类对象,则先会去父类中寻找,如果也有static变量或static代码块,则先执行父类中的. 其次再执行了父类的普通变量和普通代码块+父类的构造函数. --- >  再其次才是子类的普通变量和普通代码块+子类的构造函数. 而如果调用方法,在子类中有覆盖父类的同方法时,只执行子类的方法.而不会再使用父类被覆盖掉的.将编程看作是一门艺术,而不单单是个技术。
789 0