繁体   English   中英

指向文件的C ++向量

[英]C++ Vector of pointers to files

我目前正在尝试从矢量访问指向文件的指针。 我面临的问题,我不能独自解决的问题是,我正在用上一次创建的向量(或者至少看起来是这样)重写已经存储在vector中的指针。 因此,如果我尝试访问它们,则只有最后一个可用。 也许有一个我看不到的简单解决方案,但是我已经花了三天时间试图找出自己做错了什么(我已经完全从头开始重写了两次代码)。

int cnt = 0;
vector<AbstractInput*> abstractFiles;
while (cnt < 6) {
    string path = "D:/tempSort_" + to_string(cnt) + ".txt";
    ofstream fileOutput(path);
    if (!fileOutput)
    {
        // todo error 
        exit(1);
    }
    AbstractOutput* tmp = new FileOutput(fileOutput, kernel);
    tmp->WriteLine(to_string(cnt) + " zkouska");
    tmp->WriteLine(to_string(cnt) + " zkouska2");
    tmp->WriteLine(to_string(cnt) + " zkouska3");
    tmp->Close();
    delete(tmp);

    ifstream fileInput(path);
    if (!fileInput)
    {
        exit(1);
    }       

    abstractFiles.push_back(&(FileInput(fileInput, kernel)));

    output->WriteLine("CNT = " + to_string(cnt));
    for (AbstractInput* it : abstractFiles)
    {
        bool succes;
        output->WriteLine((it)->ReadLine(succes));
    }       


    cnt++;

} 

FileInput:

using namespace std;

class FileInput : public AbstractInput
{
    using AbstractInput::AbstractInput;


private:
    ifstream& inputFile;
    bool closed;

public:
    FileInput::FileInput(ifstream& inputFile, Kernel* kernel) : 

    AbstractInput(kernel), inputFile{ inputFile }, closed(false)
    {   

    }

    int FileInput::Close()
    {
        inputFile.close();
        closed = true;
        return 0;
    }

    bool FileInput::HasNext()
    {
        return !closed;
    }
    string FileInput::Read()
    {
        return GetKernel()->ReadFromKeyboard();
    }
    string FileInput::ReadLine(bool& success)
    {
        string line = GetKernel()->ReadLineFromFile(inputFile, success);

        closed = !success;
        return line;    
    }

};

内核功能:

string Kernel::ReadLine(istream& stream, bool& success)
{
    string line;
    if (getline(stream, line))
    {
        success = true;
        return line;
    }
    success = false;
    return "";
}

string Kernel::ReadLineFromFile(ifstream& stream, bool& success)
{   
    return ReadLine(stream, success);
}

编辑:添加了整个代码+ FileInput类

EDIT2:添加了内核功能

EDIT3:我尝试使用&(FileInput(inputFile, kernel))new FileInput(inputFile, kernel)创建新的指针。 两者都以相同的结果结尾->仅最后添加的指针有效,其他指针指向同一文件。

EDIT4:当前输出+预期输出

当前版本的输出:

CNT = 0
0 zkouska
CNT = 1
1 zkouska
1 zkouska2
CNT = 2
2 zkouska
2 zkouska2
2 zkouska3
CNT = 3
3 zkouska
3 zkouska2
3 zkouska3

CNT = 4
4 zkouska
4 zkouska2
4 zkouska3


CNT = 5
5 zkouska
5 zkouska2
5 zkouska3



.

预期结果将是:

CNT = 0
0 zkouska
CNT = 1
0 zkouska
1 zkouska
CNT = 2
0 zkouska
1 zkouska
2 zkouska

还有更多...由于我仅读取每个文件的第一行,因此zkouska1,zkouska2,...应该永远不会写入输出。

&(FileInput(fileInput))是保存对变量的引用的临时地址。

  1. 您用悬空的指针填充向量。
  2. 您的类包含对变量的引用,该变量在循环结束时超出范围。

您要在堆上创建非临时对象并存储其地址。

auto fileInput = new std::ifstream(path);
abstractFiles.push_back(new FileInput(*fileInput));

请记住,之后要正确删除这些指针(ifstream和FileInput)。

注意:for循环应该做什么? 您可以在每次while迭代中从abstractFiles每个有效条目中读取一行。

我期望的输出是:

CNT = 0
0 zkouska
CNT = 1
0 zkouska2
1 zkouska
CNT = 2
0 zkouska3
1 zkouska2
2 zkouska
CNT = 3

1 zkouska3
2 zkouska2
3 zkouska
CNT = 4


2 zkouska3
3 zkouska2
4 zkouska

更换

ifstream fileInput(path);
if (!fileInput)
{
    exit(1);
}       

std::shared_ptr<ifstream> fileInput(new ifstream(path) );
if (*fileInput)
{
    exit(1);
}       

然后用std::shared_ptr<ifstream>替换所有其他ifstream&实例(当实际使用流从文件中读取时,您还必须将stream称为*stream )。 这将防止文件流对象超出范围,并在程序结束时负责销毁和重新分配对象。 (只要您负责破坏AbstractFiles向量中的对象;实际上,您应该考虑在那里使用unique_ptr )。

关于您的预期输出...好吧,如果您打开一个文件,读取一行,然后在下一个循环中使用相同的文件描述符读取另一行,则必须读取下一行,除非您使用类似的方法重置流fileStream->Seek( 0, SeekOrigin::Begin ); 这里解释。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM