【问题标题】:Smart pointers in SDL wrapper headerSDL 包装头中的智能指针
【发布时间】:2018-12-02 01:51:08
【问题描述】:

SDL 类型如何作为成员用于包装类中的智能指针? (使用 SDL2、mingw 和 gnu-make)

我知道这个问题并不简单,所以这里是当前工作的上下文:

main.cpp:

#include "WindowWrapper.h"

int main(int argnum, char* argv[])
{
  //stuff...

  WindowWrapper myWindow;

  //more stuff...
}

WindowWrapper.h

//include guard...

class SDL_Window;

class WindowWrapper
{
  public:
    //constructor destructor, stuff....

  private:
    SDL_Window* window;
}

WindowWrapper.cpp

#include "WindowWrapper.h"
#include "SDL.h"

WindowWrapper::WindowWrapper()
{
  int initResult = SDL_Init(SDL_INIT_VIDEO);
  //error check, boring stuff...

  window = SDL_CreateWindow(/* whatever, params... you know... */)
}

WindowWrapper::~WindowWrapper()
{
  SDL_DestroyWindow(window);
  window = nullptr;

  SDL_Quit();
}

正如我所说,这工作正常,但我想将诸如窗口(表面、渲染器、纹理、通常)之类的指针更改为智能指针,例如 std::unique_ptr 或类似的。

但是…… 给定一个像这样的自定义删除器:

struct CustomDeleter
{
  void operator()(SDL_Window* window)
  {
    SDL_DestroyWindow(window);
  }

  void operator()(SDL_Window** window) //sounds crazy, but you would try even crazier stuff when experimenting
  {
    SDL_DestroyWindow(*window);
  }
};

我在以下所有方面都失败了(全部用于 WindowWrapper.h):

#incldue "SDL.h"  //--> Fails to include in any header,
// causes the famous "Undefined Reference to WinMain" problem

/*
 * Whatever definitions
 */


#include <memory>

typedef struct SDL_Window SDL_Window;
class WindowWrapper
{
 /*
  * Stuff
  */
 private:
   std::unique_ptr<SDL_Window, CustomDeleter> window;  //--> obviously fails, incomplete type 
}


#include <memory>

class SDL_Window;
class WindowWrapper
{
 /*
  * Stuff
  */
 private:
   std::unique_ptr<SDL_Window*, CustomDeleter> window;  //--> acts like SDL_Window**... 
   //compiles, but almost impossible to get it work without extra intermediate objects, and other nasty hacks
}

我还尝试制作一个自定义指针模板类,该类将 SDL_Whatever* 和 deleter 函数作为构造参数,直到我想复制/移动包装对象(比如将它们存储在地图或其他容器中)之前它都很好)。那时所有的复制/移动语义都变得非常混乱,实现一个像 shared_ptr 这样的引用计数器现在已经超出了我的范围,无论如何,这就是实现 std 智能指针的原因。

总结一下:

  1. 有没有办法将 SDL 类型本身包含在标头中? 我猜头文件中包含 SDL.h 失败的原因是因为与 cpp 文件不同,头文件在编译过程中没有链接在一起,所以它永远找不到主函数的定义。但是,如果只包含类型(尽管它们在 SDL 包中没有自己的包含文件 -.-" ),那将很容易解决这个问题。

  2. 有没有一种干净的方法来解决智能指针的“您只能声明原始指针和引用而没有完整类型”的规则? 我知道唯一指针有一个例外,如果你可以保证一堆关于析构函数的东西,但我认为像这样定义的自定义删除器实际上违反了这些规则。但我仍然认为智能指针应该有办法解决这个问题,因为它们实际上是带有附加功能的指针。我就是不知道怎么做。

【问题讨论】:

    标签: c++ sdl smart-pointers


    【解决方案1】:

    您可以使用自定义删除器使其正常工作:https://wandbox.org/permlink/K3L7BvJSf7OuKfFM

    关键是你声明删除器operator()和包装器一起在标题中,但实现它的功能在关联的.cpp中。这样它是一个完整的类型,但您仍然将所有实际功能封装在包装器 .cpp 中,您可以在其中访问实际的 SDL。

    【讨论】:

    • 谢谢!它就是这样工作的。实际上我设法将自定义删除器提取到他们自己的单个类中,所以现在我到处都有一个很好的 codestd::unique_ptr myPointer;code 语法。只需保留您编写的此标头-源代码分离即可。看起来很简单,可惜我没有弄清楚。
    猜你喜欢
    • 2014-08-06
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-01-24
    • 1970-01-01
    • 2019-09-21
    • 2016-10-09
    • 1970-01-01
    相关资源
    最近更新 更多