如何从函数中读取文件中的数据

How to read data from a file from within a function

我想让我的代码更有效率,特别是从文本文件中读取数据。这是它现在的样子的快照:

values V(name); 
    V.population = read_value(find_line_number(name, find_in_map(pop, mapping)));
    V.net_growth = read_value(find_line_number(name, find_in_map(ngr, mapping)));
... // and so on

基本上,read_value函数创建一个ifstream对象,打开文件,读取一行数据,然后关闭文件连接。这种情况发生了很多次。我想要做的是打开文件一次,将需要的每一行读入结构,然后关闭文件连接。

这是创建 values 结构函数的参数:

static values create_struct(std::string name, std::map<std::string, int> mapping) {
    values V(name); 
    V.population = read_value(find_line_number(name, find_in_map(pop, mapping)), file);
    V.net_growth = read_value(find_line_number(name, find_in_map(ngr, mapping)), file);
    // more values here
    return V;
}

调用create_struct的函数如下所示:

void initialize_data(string name) {
    // read the appropriate data from file into a struct
    value_container = Utility::create_struct(name, this->mapping);
}

我正在考虑在函数 initialize_data 中定义 ifstream 对象。根据显示的有关我的程序的内容,这是创建文件对象、打开连接、读取值然后关闭连接的最佳位置吗?另外,我是否需要将 ifstream 对象传递到 create_values 结构中,如果需要,按值、引用或指针传递?

简短的回答是首先创建您的 ifstream 对象并将其作为对解析器的引用传递。记得在你离开你的功能之前,或者当你开始阅读的时候,把流找回开头。

RAII 要做的是创建一个包装器对象,当它超出范围时自动执行此操作。

class ifStreamRef{
    ifStreamRef(std::ifstream& _in) : mStream(_in){}
    ~ifStreamRef(){mStream.seekg(0);}
    std::ifstream& mStream;
}

然后在输入将读取 fstream 的方法时创建一个包装器实例。

void read_value(std::ifstream& input, ...){
    ifStreamRef autoRewind(input);
}

或者,因为 Ctor 可以进行转换...

void read_value(ifStreamRef streamRef, ...) {
    streamRef.mStream.getLine(...);
}

std::ifstream 本身遵循 RAII,因此当您的流超出范围时它会为您关闭()流。


长答案是你应该阅读依赖注入。不要在 objects/functions 中创建可以共享的依赖项。有很多关于依赖注入和依赖倒置的视频和文档。

基本上,构造您的对象所依赖的对象并将它们作为参数传入。

注入现在依赖于您传入的对象的接口。因此,如果您更改 ifStreamRef class 以充当接口:

class ifStreamRef{
    ifStreamRef(std::ifstream& _in) : mStream(_in){}
    ~ifStreamRef(){mStream.seekg(0);}

    std::string getLine(){
        // todo : mStream.getLine() + return "" on error;
    } 
    bool eof() { return mStream.eof(); }
    std::ifstream& mStream;
}

然后您可以稍后更改将引用 vector<string>& 而不是 ifstream...

的内部实现
class ifStreamRef{
    ifStreamRef(std::vector<string>& _in) : mStream(_in), mCursor(0){}
    ~ifStreamRef(){}

    std::string getLine(){
        // todo : mStream[mCursor++] + return "" on error;
    } 
    bool eof() { return mCursor >= mStream.size(); }
    std::vector<string>& mStream;
    size_t mCursor;
}

我把一些事情简单化了。