2

我目前正在使用 GDI+ 制作游戏,我知道这不是开发游戏的最佳解决方案,但由于它是一个学校项目,我别无选择。

大约每运行一次游戏,图形就会呈现在屏幕左上角的表单之外。

如果这有助于缩小问题范围,我正在使用双缓冲。

渲染代码如下所示:

while (true)
{       
    // Create buffer if it don't exist already
    if (context == null)
    {
        context = BufferedGraphicsManager.Current;
        this.buffer = context.Allocate(CreateGraphics(), this.DisplayRectangle);
    }

    // Clear the screen with the forms back color
    this.buffer.Graphics.Clear(this.BackColor);

    // Stuff is written to the buffer here, example of drawing a game object:
    this.buffer.Graphics.DrawImage(
        image: SpriteSheet,
        destRect: new Rectangle(
            this.Position.X
            this.Position.Y
            this.SpriteSheetSource.Width,
            this.SpriteSheetSource.Height),
        srcX: this.SpriteSheetSource.X,
        srcY: this.SpriteSheetSource.Y,
        srcWidth: this.SpriteSheetSource.Width,
        srcHeight: this.SpriteSheetSource.Height,
        srcUnit: GraphicsUnit.Pixel);

    // Transfer buffer to display - aka back/front buffer swapping 
    this.buffer.Render();
}

用截图更容易解释:

游戏 gfx 外赢形式问题

4

2 回答 2

5

在 Winforms 中将 BufferedGraphicsXxx 类公开是一个设计错误。它们是 Winforms 中双缓冲支持的一个实现细节,并且它们对于错误使用它们并不是非常有弹性。

您肯定使用的是从 Allocate() 错误返回的 BufferedGraphics。您在游戏循环中以高速率创建缓冲区。但是您忘记在循环结束时处理您使用的缓冲区。这将以高速率消耗设备上下文 (HDC)。这不会永远持续下去,如果你的程序没有让垃圾收集器运行,那么 Windows 会拔掉插头,不会让你创建新的设备上下文。内部 CreateCompatibleDC() 调用将失败并返回 NULL。否则 BufferedGraphicsContext 类会错过检查此错误的代码并继续使用 NULL 句柄。并开始绘制到桌面窗口而不是窗体。

一个解决方法是将 Allocate() 调用移到循环之外,这样您只需执行一次。但是现在当用户更改窗口大小时,您将遇到一个新问题,缓冲区不再是正确的大小。

更好的捕鼠器是不使用 BufferedGraphics 类,而是将其留给 Winforms 来解决。在 Winforms 中获得游戏循环的方法有多种,但最简单的一种是使用 OnPaint() 方法渲染场景并立即请求另一个绘制,这样它就会不断地被一遍又一遍地调用。与此类似:

public partial class Form1 : Form {
    public Form1() {
        InitializeComponent();
        this.DoubleBuffered = true;
        this.ResizeRedraw = true;
    }
    protected override void OnPaint(PaintEventArgs e) {
        RenderScene(e.Graphics);
        this.Invalidate();
    }
}

RenderScene() 应该使用传递的 Graphics 实例绘制游戏对象。请注意,您不再需要使用 Clear(),这已经完成了。

于 2012-12-30T18:15:32.870 回答
2

大约每运行一次游戏,图形就会呈现在屏幕左上角的表单之外。

从屏幕截图和您的描述中,您偶尔会绘制到 Window 的桌面设备上下文 (DC);这是在获取 DC 时使用零窗口句柄 (IntPtr.Zero) 的效果。这使我相信您可能在创建表单窗口之前开始游戏循环,从而导致图形上下文指向零窗口句柄。

正如评论中所确认的那样,您正在为您的游戏循环使用单独的线程,从而导致这种情况发生的随机行为。一旦处理线程,在启动和完成线程的时间(尤其是当线程可以通过多核/cpu 计算机并行运行时)时,您并不总是两次得到相同的结果。每次运行游戏应用程序时,游戏循环线程都有可能在 UI 线程上的窗体窗口被创建和显示之前启动并执行。

于 2012-12-30T17:26:17.073 回答