- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我发现了一些我不理解的 del
行为,希望您能给我一些见解。我正在尝试使用 PyOpenGL 和 glfw 实现 OpenGL 的 hello_triangle。关闭 OpenGL 窗口后,我的程序应该进行清理,但是 glDeleteBuffers
会引发 TypeError,但仅限于在 __del__
函数内部调用它时:
class Scene:
def __init__ (self):
# ...
self.buffer = glGenBuffers(1)
# ...
def __del__ (self):
# ...
glDeleteBuffers(1, [self.buffer]) # TypeError: ('No array-type handler for type builtins.type (value: [1]) registered', <OpenGL.converters.CallFuncPyConverter object at ...>)
# ...
# ...
scene = Scene()
while not glfwWindowShouldClose(window):
scene.render()
glfwSwapBuffers(window)
glfwPollEvents()
del scene
如果我像这样实现它
class Scene:
def __init__ (self):
# ...
self.buffer = glGenBuffers(1)
# ...
def delete (self): # Renamed __del__ to delete
# ...
glDeleteBuffers(1, [self.buffer]) # No error
# ...
# ...
scene = Scene()
while not glfwWindowShouldClose(window):
scene.render()
glfwSwapBuffers(window)
glfwPollEvents()
scene.delete() # Swapped del scene for scene.delete()
glDeleteBuffers
突然工作并且没有抛出任何错误。 这是为什么?如果您想自己尝试一下,这里是完整的代码:
import ctypes
import sys
# OpenGL + GLFW
import glfw
from glfw.GLFW import *
from OpenGL.GL import *
glfw.ERROR_REPORTING = False # Catch errors by return values
class obj: pass # Object to assign arbitrary properties to
def main (args):
# Initialize GLFW + create window
if glfwInit() == GL_TRUE:
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 4)
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 5)
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE)
window = glfwCreateWindow(800, 600, "Title", None, None)
if window:
glfwMakeContextCurrent(window)
window_size_callback(window, 800, 600)
glfwSetWindowSizeCallback(window, window_size_callback)
# Render stuff
scene = Scene()
while not glfwWindowShouldClose(window):
scene.render()
glfwSwapBuffers(window)
glfwPollEvents()
# Clean up
del scene # TypeError
# scene.delete() # no TypeError
else:
print("Failed to create GLFW window!")
glfwTerminate()
else:
print("Failed to initialize GLFW!")
def window_size_callback (window, width, height):
glViewport(0, 0, width, height)
class Scene:
_instances = []
class vertex (ctypes.Structure):
_fields_ = [
("x", GLfloat),
("y", GLfloat)
]
def __static_init__ ():
# Create rendering pipeline program
vertex_shader = glCreateShader(GL_VERTEX_SHADER)
glShaderSource(vertex_shader, """#version 450 core
in vec4 pos;
void main () {
gl_Position = vec4(pos.xy, 0.0, 1.0);
}""")
glCompileShader(vertex_shader)
fragment_shader = glCreateShader(GL_FRAGMENT_SHADER)
glShaderSource(fragment_shader, """#version 450 core
out vec4 frag_color;
void main () {
frag_color = vec4(1.0, 1.0, 1.0, 1.0);
}""")
glCompileShader(fragment_shader)
Scene._program = glCreateProgram()
glAttachShader(Scene._program, vertex_shader)
glAttachShader(Scene._program, fragment_shader)
glLinkProgram(Scene._program)
glDeleteShader(vertex_shader)
glDeleteShader(fragment_shader)
# Create VAO
Scene._vertex_array = glGenVertexArrays(1)
glBindVertexArray(Scene._vertex_array)
Scene._attrib_pos = glGetAttribLocation(Scene._program, "pos")
glVertexAttribFormat(Scene._attrib_pos, 2, GL_FLOAT, GL_FALSE, 0)
glEnableVertexAttribArray(Scene._attrib_pos)
glVertexAttribBinding(Scene._attrib_pos, Scene._attrib_pos)
def __static_del__ ():
glDeleteVertexArrays(1, [Scene._vertex_array]) # Alsa raises a TypeError, if glDeleteBuffers' error is catched before
glDeleteProgram(Scene._program)
def __init__ (self):
if len(Scene._instances) == 0:
Scene.__static_init__()
Scene._instances.append(self)
# Create VBO
vertex_buffer_data = (Scene.vertex * 3)(
Scene.vertex(-0.5, 0.5),
Scene.vertex(0.5, 0.5),
Scene.vertex(0.5, -0.5)
)
self._vertex_buffer = obj()
self._vertex_buffer.buffer = glGenBuffers(1)
self._vertex_buffer.length = len(vertex_buffer_data)
self._vertex_buffer.offset = Scene.vertex.x.offset
self._vertex_buffer.stride = ctypes.sizeof(Scene.vertex)
glBindBuffer(GL_ARRAY_BUFFER, self._vertex_buffer.buffer)
glBufferData(GL_ARRAY_BUFFER, vertex_buffer_data, GL_STATIC_DRAW)
def __del__ (self): # Rename to delete
glDeleteBuffers(1, [self._vertex_buffer.buffer]) # TypeError, if executed in __del__(), but not when executeed in delete()
Scene._instances.remove(self)
if len(Scene._instances) == 0:
Scene.__static_del__()
def render (self):
glClearColor(0.0, 0.1, 0.2, 1.0)
glClear(GL_COLOR_BUFFER_BIT)
# Draw
glUseProgram(Scene._program)
glBindVertexArray(Scene._vertex_array)
glBindVertexBuffer(Scene._attrib_pos, self._vertex_buffer.buffer, self._vertex_buffer.offset, self._vertex_buffer.stride)
glDrawArrays(GL_TRIANGLES, 0, self._vertex_buffer.length)
if __name__ == "__main__":
main(sys.argv[1:])
最佳答案
glDeleteBuffers
raises a TypeError, BUT ONLY if it is called inside a__del__
function:
导致该错误的原因是,在调用析构函数之前,OpenGL 上下文已被销毁。
对于任何其他 OpenGL 指令,glDeleteBuffers
需要有效且当前的 OpenGL 上下文。
如果
scene.delete()
被调用,然后 delete()
并立即调用 glDeleteBuffers
。此时 OpenGL 上下文是当前的,并且操作在任何情况下都会成功。
但是当你这么做的时候
del scene
那么不能保证立即调用析构函数。
参见Python- 3.3.1. Data model - Basic customization
Note
del
x doesn’t directly callx.__del__()
— the former decrements the reference count for x by one, and the latter is only called when x’s reference count reaches zero.
析构函数何时被调用取决于垃圾回收。 Python 不提供任何保证,关于何时调用析构函数,它会在所有引用被删除之后发生,因此可能不必在之后立即发生。
这会导致在 OpenGL 竞赛被销毁之后(在 glfwTerminate()
之后)调用析构函数并且操作失败。
一个安全的方法是直接调用析构函数:
例如
Scene.__del__(scene)
关于python - PyOpenGL.glDeleteBuffers 在 __del__ 函数中的奇怪行为?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54969902/
您好,我正在使用 VBO 加载图像纹理,然后用 C++ 绘制它。 VBO id 生成、绑定(bind)和绘制发生在这里 void ViewManager::render(){ glClear(GL_C
目前,我在 Java 端设置了一个在 onDestroy() 期间调用的 native 函数。我在我为其分配内存的任何 native 端指针上调用 free()。如果我还在这个函数中调用 glDele
正如 Qt 用户所知,使用任何 OpenGL 扩展都非常麻烦。我让它工作的方法是像这样扩展 QGLFunctions 类: class Object3D : protected QGLFunctio
基本上,在我的代码中,我挂接了 glDeleteTextures 和 glBufferData 函数。我存储了一个纹理列表和一个缓冲区列表。缓冲区列表包含校验和和指向缓冲区的指针。下面的代码在数据到达
我在我的 iOS/Android 游戏中遇到了一些性能问题,其中有几个 VBO 必须每隔一段时间更新一次。在对我的游戏进行分析后发现,每次 VBO 更新 glDeleteBuffers() 最多需要
我发现了一些我不理解的 del 行为,希望您能给我一些见解。我正在尝试使用 PyOpenGL 和 glfw 实现 OpenGL 的 hello_triangle。关闭 OpenGL 窗口后,我的程序应
是否需要在删除缓冲区对象之前取消绑定(bind)它?如果我在 VAO 中绑定(bind)它并删除它而不解除绑定(bind)(绑定(bind)到 0),会发生什么?引用还会存在吗? public voi
在 OpenGL ES 2 中使用 VBO 时,我遇到了 glDeleteBuffers , glDeleteShader , 和 glDeleteProgram .我在网上环顾四周,但找不到关于何时
我读到 VBO(顶点缓冲区对象)本质上保留一个引用计数,因此如果将 VBO 的名称指定给 glDeleteBuffers(),如果一个活的VAO(顶点数组对象)仍然引用它。这种行为类似于“智能指针”,
我是一名优秀的程序员,十分优秀!