我的编译器不支持C++11。
事实证明,没有简单的方法来初始化标准库容器的元素。例如,如果要初始化字符串向量,通常会使用如下所示的push_back()
调用序列:
// file.h
void init_input_tbl(vector<string>& tbl){
tbl.push_back("jan");
tbl.push_back("feb");
tbl.push_back("mar");
tbl.push_back("apr");
tbl.push_back("may");
tbl.push_back("jun");
tbl.push_back("jul");
tbl.push_back("aug");
tbl.push_back("sep");
tbl.push_back("oct");
tbl.push_back("nov");
tbl.push_back("dec");}
vector<string>month_input_tb
// file.cpp
main(){
init_input_tbl(month_input_tb);
//...
问题 1.有没有其他方法可以在 C++03 中初始化容器?
除此之外,当我尝试初始化时,在头文件内或全局环境中使用函数init_input_tbl()
它会显示以下错误:this declaration has no storage class or type specifier
.
问题 2.在file.h
内部或全局环境中使用init_input_tbl()
初始化矢量month_input_tb
的方法或技术是什么?
全局的替代方法是创建一个函数,该函数返回要在初始化中使用的值:
std::vector<std::string> month_inputs();
std::vector<std::string> month_input_tb = month_inputs();
然后在 cpp 中(或者如果你这样做是内联的标题),你可以定义函数:
std::vector<std::string> month_inputs() {
std::vector<std::string> result;
result.push_back(...);
...
return result;
}
如果可以拉入一些其他依赖项,则可以使用已经提供此类功能的 boost 分配:
std::vector<std::string> month_inputs_tb = list_of<std::string>()("One")("Two");
首先,在没有static
或类的情况下,在标头中声明这样的全局变量是一个坏主意。如果标头包含在多个.cpp
文件中,则会遇到编译器问题。
问题的一种解决方案是使用单例模式。具有静态变量的类,可由函数获取,该函数在第一次调用函数时初始化。
David Rodriguez的解决方案也有效,但是您将回到只能包含在一个源文件中的头文件的问题。但如果你这样做:
//utils.h
struct utils {
static std::vector months_inputs_tb;
};
//utils.cpp
#include "utils.h"
std::vector month_inputs() {
...
}
utils::months_inputs_tb = month_inputs();
//code.cpp
#include "utils.h"
//use utils::months_input as you want
然后,您可以在多个.cpp
文件中自由包含包含months_inputs_tb
的标头。
如果你想避免在你的主cpp文件中进行任何类型的初始化,你最好的选择是使用编译器特定的库init/constructor,如MSVC中的DllMain
或g++中的__attribute__((constructor))
。我相信在 c++ 程序范围内无法保证静态类成员的初始值设定项的顺序。
例如 g++:
#include <vector>
#include <string>
#include <iostream>
using namespace std;
__attribute__((init_priority(1000)))
vector<string> myvect;
__attribute__((constructor(1001)))
void myvectinit() {
myvect.push_back("abc");
myvect.push_back("def");
myvect.push_back("ghi");
}
int main() {
cout << myvect.size() << endl; // output: 3
return 0;
}
我假设您的目标只是一种初始化复杂对象的便捷方法,例如 vector
s.(我不明白你为什么关心代码的位置,标头与 cpp 文件)。
使用此帮助程序:
template<size_t N, typename T>
vector<T> construct_vector(const T (&init)[N]) {
vector<T> tbl;
for(size_t i=0; i<N; ++i) {
tbl.push_back( init[i] );
}
return tbl;
}
你可以像这样构造你的向量:
vector<string> month_input_tb = construct_vector((string[]) {"jan","feb"
,"mar","apr","may","jun","jul","aug","sep","oct","nov","dec"});
这是有效的,因为您只是在使用 string
.更复杂对象的向量可能更困难。