1

我目前正在为我的路径跟踪器开发基本框架。我正在尝试创建一个简单的二进制模型加载器(读取和写入),但我在读回文件时遇到了问题。提供更多背景信息:

编写我的数据似乎很顺利,即没有抛出异常,文件大小似乎很好。当我开始读取数据时,一切似乎都很好,直到我的数据被损坏时(在读取所有顶点位置和一些法线之后)。当我调试时,所有数字都是正确的,但很明显在某些时候我得到 #DEN 而不是正确的浮点值。在某些时候我的所有网格都会发生同样的情况,因此在读取 UV 或三角形等时可能会发生这种情况。

(图片#DEN 在第 54 次正常(第 3 次浮动)......没有足够的声誉)

我一口气阅读了所有法线(1个子网格)。当我查看 ifstream 时,它会重置回文件的开头。当我开始阅读我的三角形索引(一个新的stream.read())时,坏位被设置了。

我使用 Visual Studio 2010 Ultimate,并且在 Windows 7 Professional x64 上运行。我正在使用 C 和 C++。

我将尝试仅显示关键代码:

if(meshHeader.numberOfNormals > 0)
{           
    Normal* _normals = Read<Normal>(meshHeader.numberOfNormals);

    if(_normals)
    {
        // Copy from array to vector
        _normalsVec = new vector<Normal>(_normals, _normals + meshHeader.numberOfNormals);

        // Cleanup              
        delete[] _normals;
    }
    else
    {
        // Cleanup
        delete[] _normals;
        throw runtime_error("Failed reading vertex normals");
    }
}

阅读方法:

template <typename T> 
T* BinaryModelIO::Read(unsigned int _size)
{
    T* _data = new T[_size];
    unsigned int _numberOfBytes = Read(sizeof(T) * _size, static_cast<void*>(_data));

    if(_numberOfBytes != (sizeof(T) * _size))
    {
        delete[] _data;
        _data = 0;
    }

    return _data;
}

(static_cast(_data) 不是原因)更多阅读:

unsigned int BinaryModelIO::Read(unsigned int _numberOfBytes, void *_buffer)
{
    if(isWriter)
        return 0;
    // no fail bit set?
    if(!readStream->fail())
    {
        try
        {
            readStream->read((char*)_buffer, _numberOfBytes);
        }
        catch(exception &e)
        {
            cerr << e.what() << endl;
            //return 0;
        }
        return _numberOfBytes;
    }
    else
    {
        if(readStream->badbit)
            cerr << "badbit" << endl;
        else if(readStream->eofbit)
            cerr << "eofbit" << endl;
        return 0;
    }
}

读取法线按预期进行,但经过一定数量的浮点数(法线)后会出错。我尝试使用 for 循环读取整个 Normals 数组,然后将所有元素复制到向量中,但这不是问题。读取三角形数据时(以类似方式完成),设置坏位(或在读取三角形之前的某个位置)。在下图中,我在读取三角形时正在调试,并且打印了 badbit。

(图片显示打印...没有足够的声誉)

我正在尝试读/写的数据结构不包含虚函数,并且全部由浮点数组成(具有 3 个无符号整数的三角形除外)。点(3 个浮点数)、UV(2 个浮点数)、正常(3 个浮点数)、三角形(3 个无符号整数)。

编辑:这是法线的定义

class Normal 
{
public:
    // Vector Public Methods
    Normal(float _x = 0.0f, float _y = 0.0f, float _z = 0.0f) : x(_x), y(_y), z(_z) {   }
        explicit Normal(const Vector3& _vector);
        explicit Normal(const Point& _point);

    // Overloaded Operators
    Normal operator+(const Normal& _norm) const 
    {
        return Normal(x + _norm.x, y + _norm.y, z + _norm.z);
    }   
    Normal& operator+=(const Normal& _norm) 
    {
        x += _norm.x; 
        y += _norm.y; 
        z += _norm.z;
        return *this;
    }
    Normal operator-(const Normal& _norm) const 
    {
        return Normal(x - _norm.x, y - _norm.y, z - _norm.z);
    }   
    Normal& operator-=(const Normal& _norm) 
    {
        x -= _norm.x;
        y -= _norm.y; 
        z -= _norm.z;
        return *this;
    }
    bool operator==(const Normal& _norm) const 
    {
        return x == _norm.x && y == _norm.y && z == _norm.z;
    }
    bool operator!=(const Normal& _norm) const 
    {
        return x != _norm.x || y != _norm.y || z == _norm.z;
    }
    Normal operator*(float _f) const 
    {
        return Normal(_f * x, _f * y, _f * z);
    }   
    Normal &operator*=(float _f) 
    {
        x *= _f; 
        y *= _f; 
        z *= _f;
        return *this;
    }
    Normal operator/(float _f) const 
    {
        // One division, 3 multiplications ^^
        float _inverse = 1.f / _f;
        return Normal(x * _inverse, y * _inverse, z * _inverse);
    }
    Normal &operator/=(float _f) 
    {
        float _inverse = 1.f / _f;
        x *= _inverse; 
        y *= _inverse; 
        z *= _inverse;
        return *this;
    }
    Normal operator-() const 
    {
        return Normal(-x, -y, -z);
    }
    float operator[](int _i) const 
    {
        // Return x, y or z
        return (&x)[_i];
    }   
    float &operator[](int _i) 
    {
        return (&x)[_i];
    }

    float LengthSquared() const 
    { 
        return x * x + y * y + z * z; 
    }
    float Length() const 
    { 
        return sqrtf(LengthSquared()); 
    }

    // Public Variables
    float x, y, z;
}; // 12 Byte

edit2:文件打开(在创建和销毁 Loader 对象时完成)

bool AbstractStream::OpenFile()
{
    bool _return = false;

    // We should open file to write or to read
    if(isWriter)
    {
        // Check if the file is already open
        if(!writeStream->is_open())
            writeStream->open(fileName, ios::in | ios::binary);

        if(!writeStream->fail())
            _return = true; // File could be opened
    }
    else
    {
        // Check if the file is already open
        if(!readStream->is_open())
            readStream->open(fileName, ios::in | ios::binary);

        if(!readStream->fail())
            _return = true; // File could be opened
    }

    return _return; // File could be opened
}

如果我忘记了什么,请告诉我,因为这是我关于 stackoverflow 的第一个问题。

我真的希望有人能帮助我,因为我暂时解决了这个问题(在渲染器的其他部分工作)。

提前致谢!

4

1 回答 1

0

除了你对新/删除(避免指针)的古董我们我猜你的问题是写数据。我的建议:尝试编写一个非(!)二进制文件,在编辑器中查看它。

于 2013-08-10T18:20:45.057 回答