Обработчик двоичного файла не записывает последнее значение - PullRequest
1 голос
/ 27 апреля 2020

Я написал небольшой класс обработчика файлов на C ++

Он имеет базовый класс, который управляет закрытием файла RAII, и два производных класса: один для записи и один для чтения. Производные классы имеют 2 метода write или read value и write или read vector.

#include <vector>
#include <fstream>
#include <iostream>
#include <algorithm>


class File{
protected:
    std::streampos offset;
    std::fstream file;

public:
    File(){
        offset = 0;
    }

    virtual ~File(){
        file.close();
    }
};

class oFile : public File{
public:
    oFile(std::string const& filename) {
        file.open(filename, std::ios::out|std::ios::binary);
    }

    template <typename T>
    void write_value(T value){
        file.seekp(offset);
        file.write(reinterpret_cast<char*> (&value), sizeof(T));
        offset += sizeof(T);
        std::cout<< "wrote a value: " << value << "  current offset: " << offset << std::endl;
    }

    template <typename T>
    void write_vector(std::vector<T> v){

        write_value(v.size());

        std::for_each(v.begin(), v.end(), [this](T& i){write_value(i);});
    }
};

class iFile : public File{
public:

    iFile(std::string const& filename){
        file.open(filename, std::ios::in|std::ios::binary);
    }

    template <typename T>
    void read_value(T& v){
        file.seekg(offset);
        file.read(reinterpret_cast<char*> (&v), sizeof(T));
        offset += sizeof(T);
        std::cout << "value read: " << v << " current offset: " << offset << std::endl;
    }

    template <typename T>
    void read_vector(std::vector<T>& v){
        typename std::vector<T>::size_type vsize;
        read_value(vsize);

        std::cout << "size of vector: " << vsize << std::endl;

        for(typename std::vector<T>::size_type i = 0; i < vsize; ++i){
            T value;
            read_value(value);

            v.push_back(value);
        }
    }
};

Чтобы проверить классы, я написал эту функцию:

    oFile file("test.ndf");

    double b = 12.;
    file.write_value(b);

    std::vector<double> v{1., 2., 3., 4., 6., 10., 15.};

    file.write_vector(v);

    iFile ifile("test.ndf");

    double ib;
    ifile.read_value(ib);

    std::cout << ib << std::endl;

    std::vector<double> iv;
    ifile.read_vector(iv);

    std::cout << "vector size: " << iv.size() << std::endl;

    for(auto val : iv){
        std::cout << val << " ";
    }
    std::cout << "last val: " << iv[iv.size() - 1] << std::endl;
}

Но вывод имеет проблему

wrote a value: 7 offset: 12
wrote a value: 1 offset: 20
wrote a value: 2 offset: 28
wrote a value: 3 offset: 36
wrote a value: 4 offset: 44
wrote a value: 6 offset: 52
wrote a value: 10 offset: 60
wrote a value: 15 offset: 68 <--- tells me he wrote 15
value read: 12 offset: 8
12
value read: 7 offset: 12
size of vector: 7
value read: 1 offset: 20
value read: 2 offset: 28
value read: 3 offset: 36
value read: 4 offset: 44
value read: 6 offset: 52
value read: 10 offset: 60
value read: 10 offset: 68 <--- problem
vector size: 7
1 2 3 4 6 10 10 last val: 10 <--- last value should be 15
Hello world!

Последняя строка вывода представляет собой векторное содержимое, и последнее значение повторяется, хотя в методе записи мне сообщают, что записано правильное значение (15).

Я не знаю не понимаю, почему он пишет или читает последнее значение 2 раза? Хорош ли мой обработчик файлов? Есть ли способ лучше?

скомпилировано с Code :: Blocks (v20.03) с использованием MinGW (v9.2.0)

1 Ответ

1 голос
/ 27 апреля 2020

Хорошо ли мое решение для работы с файлами?

RAII для управления ресурсами - это хороший и рекомендуемый способ.

Однако с RAII область действия (и, следовательно, время жизни) переменных должна рассматриваться еще более тщательно.

В открытом коде OP:

int main()
{
    oFile file("test.ndf");

    double b = 12.;
    file.write_value(b);

    std::vector<double> v{1., 2., 3., 4., 6., 10., 15.};

    file.write_vector(v);

    iFile ifile("test.ndf");

    double ib;
    ifile.read_value(ib);

    std::cout << ib << std::endl;

    std::vector<double> iv;
    ifile.read_vector(iv);

    std::cout << "vector size: " << iv.size() << std::endl;

    for(auto val : iv){
        std::cout << val << " ";
    }
    std::cout << "last val: " << iv[iv.size() - 1] << std::endl;
} // <-- all local variables incl. file will be destroyed here

экземпляр ofile file живет до конца main.

wrote a value: 12  current offset: 8
wrote a value: 7  current offset: 16
wrote a value: 1  current offset: 24
wrote a value: 2  current offset: 32
wrote a value: 3  current offset: 40
wrote a value: 4  current offset: 48
wrote a value: 6  current offset: 56
wrote a value: 10  current offset: 64
wrote a value: 15  current offset: 72
value read: 12 current offset: 8
12
value read: 7 current offset: 16
size of vector: 7
value read: 1 current offset: 24
value read: 2 current offset: 32
value read: 3 current offset: 40
value read: 4 current offset: 48
value read: 6 current offset: 56
value read: 10 current offset: 64
value read: 10 current offset: 72
vector size: 7
1 2 3 4 6 10 10 last val: 10

Live Demo на coliru (где я воспроизвел проблему OP).

как внутренний поток закрыто в деструкторе ofile, до этой точки может быть внутренне буферизованное (т.е. не очищенное) содержимое, хотя поток сообщит о них как записанный.

Решение простое: Область действия ofile file должно быть ограничено.

Фиксированный код:

int main()
{
    std::vector<double> v{1., 2., 3., 4., 6., 10., 15.};

    { // start new scope
        oFile file("test.ndf");

        double b = 12.;
        file.write_value(b);

        file.write_vector(v);
    } // close scope -> destroy file (and b)

    iFile ifile("test.ndf");

    double ib;
    ifile.read_value(ib);

    std::cout << ib << std::endl;

    std::vector<double> iv;
    ifile.read_vector(iv);

    std::cout << "vector size: " << iv.size() << std::endl;

    for(auto val : iv){
        std::cout << val << " ";
    }
    std::cout << "last val: " << iv[iv.size() - 1] << std::endl;
}

Выход:

wrote a value: 12  current offset: 8
wrote a value: 7  current offset: 16
wrote a value: 1  current offset: 24
wrote a value: 2  current offset: 32
wrote a value: 3  current offset: 40
wrote a value: 4  current offset: 48
wrote a value: 6  current offset: 56
wrote a value: 10  current offset: 64
wrote a value: 15  current offset: 72
value read: 12 current offset: 8
12
value read: 7 current offset: 16
size of vector: 7
value read: 1 current offset: 24
value read: 2 current offset: 32
value read: 3 current offset: 40
value read: 4 current offset: 48
value read: 6 current offset: 56
value read: 10 current offset: 64
value read: 15 current offset: 72
vector size: 7
1 2 3 4 6 10 15 last val: 15

Live Demo on coliru


Я не понимаю, почему он записывает или читает последнее значение 2 раза?

Это не было ясно Мне тоже, пока я не понял, что слишком сосредоточился на , прочитал последнее значение 2 раза .

На самом деле это не так - он просто не может прочитать последнее значение и снова ошибочно сообщает предыдущее значение .

Чтобы проверить это, я добавил минимальную «обработку ошибок» в iFile::read_value:

    template <typename T>
    void read_value(T& v){
        file.seekg(offset);
        file.read(reinterpret_cast<char*> (&v), sizeof(T));
        if (!file) std::cerr << "AARG! Input failed. :-(\n";
        offset += sizeof(T);
        std::cout << "value read: " << v << " current offset: " << offset << std::endl;
    }

Вывод:

wrote a value: 12  current offset: 8
wrote a value: 7  current offset: 16
wrote a value: 1  current offset: 24
wrote a value: 2  current offset: 32
wrote a value: 3  current offset: 40
wrote a value: 4  current offset: 48
wrote a value: 6  current offset: 56
wrote a value: 10  current offset: 64
wrote a value: 15  current offset: 72
value read: 12 current offset: 8
12
value read: 7 current offset: 16
size of vector: 7
value read: 1 current offset: 24
value read: 2 current offset: 32
value read: 3 current offset: 40
value read: 4 current offset: 48
value read: 6 current offset: 56
value read: 10 current offset: 64
AARG! Input failed. :-(
value read: 10 current offset: 72
vector size: 7
1 2 3 4 6 10 10 last val: 10

Демонстрация в реальном времени на coliru

Конечно, файловый ввод-вывод может быть неудачным по разным причинам. Итак, успешность файловых операций должна проверяться всегда.

...