OpenGL VBOs:绘制球体

3

我在绘制球体时遇到了一些问题。

std::vector<GLfloat> ballVerts;

for(int i = 0; i <= 40; i++)
{
    double lat0 = M_PI * (-0.5 + (double) (i - 1) / 40);
    double z0  = sin(lat0);
    double zr0 =  cos(lat0);

    double lat1 = M_PI * (-0.5 + (double) i / 40);
    double z1 = sin(lat1);
    double zr1 = cos(lat1);

    for(int j = 0; j <= 40; j++)
    {
        double lng = 2 * M_PI * (double) (j - 1) / 40;
        double x = cos(lng);
        double y = sin(lng);

        ballVerts.push_back(x * zr0); //X
        ballVerts.push_back(y * zr0); //Y
        ballVerts.push_back(z0);      //Z

        ballVerts.push_back(0.0f); 
        ballVerts.push_back(1.0f); 
        ballVerts.push_back(0.0f); 
        ballVerts.push_back(1.0f); //R,G,B,A

        ballVerts.push_back(x * zr1); //X
        ballVerts.push_back(y * zr1); //Y
        ballVerts.push_back(z1);      //Z

        ballVerts.push_back(0.0f); 
        ballVerts.push_back(1.0f); 
        ballVerts.push_back(0.0f); 
        ballVerts.push_back(1.0f); //R,G,B,A
    }
}

glGenBuffers(1, &ballVbo);
glBindBuffer(GL_VERTEX_ARRAY, ballVbo);

GLuint sphereSize = 3200*7*4; //3200 vertixes * 7 floats
glBufferData(GL_VERTEX_ARRAY,sphereSize, &ballVerts, GL_STATIC_DRAW);



/*
    Draw a ball
*/
glBindBuffer(GL_VERTEX_ARRAY, ballVbo);
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, 7*4, 0);
glEnableClientState(GL_COLOR_ARRAY);
glColorPointer(4, GL_FLOAT, 7*4, (void*)(3*4));

glDrawArrays(GL_TRIANGLE_STRIP, 0, 3200);

glBindBuffer(GL_ARRAY_BUFFER, 0);

生成球体的代码在立即模式下运行良好,但当我将其放入VBO中时,glDrawArrays会不断抛出访问冲突异常。有什么建议吗?
2个回答

7
glBufferData(GL_VERTEX_ARRAY,sphereSize, &ballVerts, GL_STATIC_DRAW);

ballVerts 不是一个数组,而是一个 std::vector。取一个 std::vector 的地址并不能得到它所包含的数组的地址。你需要使用 &ballVerts[0]


2
如果你有C++11的访问权限,你也可以使用std::vector的.data()方法。我相信这可以避免在空数组上使用地址运算符的问题。 - luke

0

更好的预调整向量并使用迭代器进行遍历,就像这样:

 std::vector <GLfloat> vertices;
 vertices.resize(count * 3);
 ...
 std::vector <GLfloat>::iterator v = vertices.begin();

 for( U32 r = 0; r < m_prec/2; ++r ) //широта
    {
        float const theta1 = -F_PI_BY_TWO + pi2p * r;
        float const theta2 = theta1 +  pi2p;

        for( U32 s = 0; s < m_prec; ++s ) //долгота
        {
            float const theta3 =  s * pi2p;     

            float ex = cosf(theta3) *cosf(theta2);
            float ey = sinf(theta2);
            float ez = sinf(theta3) * cosf(theta2);
            *v++ = m_x + m_radius * ex;
            *v++ = m_y + m_radius * ey;
            *v++ = m_z + m_radius * ez;

::glVertexPointer(3, GL_FLOAT, 0 , &vertices[0]);

可以工作


它真的有效吗?为什么我在中间模式下尝试代码时,它只给了我一个圆圈?而当我尝试使用glBegin(GL_Points)时,它只给了我构成圆圈的点。 - Trung Bún

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接