2

または、2 つのファイルがあり、それぞれに 1 つのグローバル初期化があります。一方は他方に依存します。

簡単な例:

file1.h:

#include <string>
extern const std::string PREFIX;

file1.cpp:

#include "file1.h"
const std::string PREFIX  = "prefix,";

file2.cpp:

#include "file1.h"
std::string MSG = PREFIX + "body";
int main(){}

私はそれらを次のようにコンパイルします:

/usr/local/bin/g++-4.6.2 -c -Wall -g -o file1.o file1.cpp
/usr/local/bin/g++-4.6.2 -c -Wall -g -o file2.o file2.cpp
/usr/local/bin/g++-4.6.2 -Wall -g  -o example file1.o file2.o

これを実行すると、セグメンテーション違反が発生します。gdb トレース:

Starting program: example

Program received signal SIGSEGV, Segmentation fault.
0x00007ffff7b7ae0b in std::basic_string<char, std::char_traits<char>, std::allocator<char> >::basic_string(std::string const&) ()
  from /usr/local/lib/gcc/x86_64-unknown-linux-gnu/4.6.2/libstdc++.so.6 
(gdb) bt
#0  0x00007ffff7b7ae0b in std::basic_string<char, std::char_traits<char>, std::allocator<char> >::basic_string(std::string const&) ()
   from /usr/local/lib/gcc/x86_64-unknown-linux-gnu/4.6.2/libstdc++.so.6
#1  0x00000000004009b5 in std::operator+<char, std::char_traits<char>, std::allocator<char> > (__lhs=Traceback (most recent call last):
   File "/usr/local/lib/gcc/x86_64-unknown-linux-gnu/4.6.2/../../../../share/gcc-4.6.2/python/libstdcxx/v6/printers.py", line 587, in to_string
     return ptr.lazy_string (length = len)
RuntimeError: Cannot access memory at address 0xffffffffffffffe8
, __rhs=0x400af0 "body") at /usr/local/lib/gcc/x86_64-unknown-linux-gnu/4.6.2/include/c++/bits/basic_string.h:2346
#2  0x000000000040095f in __static_initialization_and_destruction_0 (__initialize_p=1, __priority=65535) at file2.cpp:3
#3  0x000000000040098b in _GLOBAL__sub_I_MSG () at file2.cpp:6
#4  0x0000000000400ab6 in __do_global_ctors_aux ()
#5  0x00000000004006e3 in _init ()
#6  0x00007ffff7ffa5a0 in ?? ()
#7  0x0000000000400a45 in __libc_csu_init ()
#8  0x00007ffff72dcbe0 in __libc_start_main () from /lib/libc.so.6
#9  0x00000000004007c9 in _start ()

とにかく私がやろうとしていることをすることはありますか(これは非常に単純化された例であることを覚えておいてください)? それとも、コンパイラ/リンカーによって選択された初期化順序に翻弄されていますか?

4

3 に答える 3

5

プレフィックス文字列を他の文字列の前に配置する必要があります。1つの方法は、C文字列に変更することです

// header
#include <string>
extern const char * const PREFIX;

// .cpp file
const char * const PREFIX = "prefix,";

もう 1 つの方法は、関数からプレフィックスを返し、PREFIX()以前に使用した場所を使用することですPREFIX

// header
inline const string& PREFIX() { 
  static const string value = "prefix,";
  return value;
}

最後に、ヒントだけです。すべて大文字の名前は、ほとんどのコーディング規約でのみマクロに使用されるため、変数や関数にはそのような名前を付けないようにします。

于 2012-07-12T20:56:01.000 に答える
0

これは明らかに移植可能なソリューションではありませんが、gccの場合は関数属性__attribute__構文で使用できます。つまり、__init_priority__ (priority)属性。初期化の優先順位は、翻訳単位にまたがります。リンク

于 2012-07-13T00:38:31.697 に答える
0

コードがグローバルを間違った順序で初期化しようとするため、これが表示されると思います。SO の初期化順序に関する質問があります。その回答は役立つはずです。

于 2012-07-12T20:57:58.707 に答える