- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我开始使用 Allegro 5 在 C++ 的 Visual Studio 2017 中创建游戏。为了更轻松地管理图像,我创建了一个 ImageLoader 类,该类将加载和存储所有事件图像,并在必要时销毁它们。它使用将文件名与相应图像匹配的映射来实现这一点。目前我的 main() 代码如下所示:
int main(){
if (!al_init()) {
al_show_native_message_box(NULL, NULL, NULL, "Could not intialize allegro 5.", NULL, NULL);
return -1;
}
ALLEGRO_DISPLAY *display = al_create_display(DEFAULT_SCREEN_WIDTH, DEFAULT_SCREEN_HEIGHT);
al_set_window_title(display, "Game title");
// make usre the display was created correctly
if (!display) {
al_show_native_message_box(display, "Title", "settings", "Could not create Allegro window.", NULL, NULL);
}
// intialize fonts, primitives, keyboard,etc.
al_init_font_addon();
al_init_ttf_addon();
al_init_primitives_addon();
al_install_keyboard();
if(!al_init_image_addon()) {
al_show_native_message_box(display, "Error", "Error", "Failed to initialize al_init_image_addon!",
NULL, ALLEGRO_MESSAGEBOX_ERROR);
return -1;}
ImageLoader image_loader;
ImageLoader 然后为播放器加载图像:
ALLEGRO_BITMAP *image = al_load_bitmap(filename);
if (image == NULL) {
std::cout << filename << std::endl;
std::cout << "loader failed to load image" << std::endl;
}
image_map[filename] = image;
当我测试这部分时,它似乎工作正常,因为图像不为空。请注意,image_map 在 ImageLoader.h 中的声明方式如下:
std::map<const char *, ALLEGRO_BITMAP*> image_map;
当我尝试从 ImageLoader 获取图像时,问题出现在我的主游戏循环中:
for (GameImage image : imageList) {
ALLEGRO_BITMAP *draw_image = image_loader.get_current_image(image);
if (draw_image == NULL) {
//TODO: error handling
std::cout << "no image" << std::endl;
}
else
al_draw_bitmap(draw_image, image.get_x(), image.get_y(), NULL);
}
我的检查总是显示 draw_image
一片空白。这是 get_current_image 的代码:
ALLEGRO_BITMAP * ImageLoader::get_current_image(GameImage image)
{
return image_map[image.get_image_filename()];
}
我已通过检查 if (image_map.find(image.get_image_filename()) == image_map.end())
进行测试,以确保我在此处使用与加载图像并将其存储在 map 中时相同的文件名。 ,但即使这返回 false,ALLEGRO_BITMAP 指针仍然为空。我已经尝试让 map 存储位图而不是指针,但是当我尝试向 map 添加元素时这给了我一个错误,因为我不允许像这样修改 map 值。为什么这些指针在我设置它们的时间和我检索它们的时间之间会变为空?我也乐于接受有关如何存储位图的其他建议。
编辑:我修改了我的项目,将作为 char 数组的文件名实例更改为 std::strings。图像映射现在声明为 std::map<std::string, ALLEGRO_BITMAP*> image_map
在 ImageLoader.h 中,GameImage 存储的文件名现在是 std::string image_filename
. load_image
在 ImageLoader.cpp 中现在看起来像这样:
ALLEGRO_BITMAP *image = al_load_bitmap(filename.c_str());
if (image == NULL) {
std::cout << filename << std::endl;
std::cout << "loader failed to load image" << std::endl;
}
image_map[filename] = image;
最后,get_current_image
还是一样:
ALLEGRO_BITMAP * ImageLoader::get_current_image(GameImage image)
{
return image_map[image.get_image_filename()];
}
但是,同样的问题仍然存在。我还检查了图像映射的大小,它在我的程序的整个持续时间内保持在 1,我插入的图像的文件名作为键,值从指向位图的非空指针开始,并且看起来变成在某些时候为空。
编辑 2:
将字符数组更改为字符串并修复 get_current_image()
之后因此,如果找不到搜索到的文件名,它就不再添加到 map 中,我发现在将图像加载到我最初发布问题时忘记包含的行中时也犯了一个错误:
current_screen.load_images(image_loader);
事实证明,我写了load_images()
像这样:
void MainGameScreen::load_images(ImageLoader loader)
...意味着调用函数对 loader
进行的调用实际上并没有应用于我传入的 ImageLoader。我将其更改为:
void MainGameScreen::load_images(ImageLoader& loader)
...现在一切正常。
最佳答案
你的问题是,使用 const char *
作为键意味着 map
将执行直接地址比较,即“字符串的地址是否等于我保存在内存中的字符串的地址”。这几乎肯定不是您想要执行字符串比较的方式。
这个问题实际上有两种解决方案,具体取决于您的用例。第一种是简单地将 const char *
更改为 std::string
,这是最简单的解决方案,默认情况下您应该这样做。
std::map<std::string, ALLEGRO_BITMAP*> image_map;
而且,从广义上讲,无论在何处使用字符串,都应该使用 std::string
或 std::string const&
。没有理由使用其他任何东西。
...除非您关心性能。如果您正在编写游戏,性能几乎肯定是您关心的事情,这将我们带到了第二个解决方案。必须对 map 进行大量查找将调用大量比较,虽然这通常不是一个大问题,但之所以会出现,是因为每次比较都是基于字符串的完整相等性检查。
解决方案是,当您加载图像时,为每个图像分配一个唯一的 ID(例如 int64_t
或 uint64_t
),并将这些值分配给 GameImage
对象而不是文件名或路径名。然后,当您进行查找时,使用该 ID 执行查找。
这取决于你。将 const char *
替换为 std::string
几乎肯定会修复代码中的逻辑错误并使其按预期方式工作。如果您发现必须进行所有这些字符串比较会显着降低您的程序速度,那么剩下的更像是一个优化问题。
编辑:
你的新问题是 std::map
的 operator[]
函数自动插入一个默认值(在这种情况下,nullptr
) 如果它找不到任何具有所请求名称的预先存在的图像。您想要的代码看起来更像这样:
ALLEGRO_BITMAP * ImageLoader::get_current_image(GameImage image)
{
auto it = image_map.find(image.get_image_filename());
if(it == image_map.end()) return nullptr;
else return it->second;
//return image_map[image.get_image_filename()];
}
这样,找不到图像就不会诱使您使用的任何调试工具认为该位置存储了有效(空)值。
如果您想改用内置异常工具,也可以将其简化为:
ALLEGRO_BITMAP * ImageLoader::get_current_image(GameImage image)
{
//Will throw an exception if nothing is found
return image_map.at(image.get_image_filename());
}
关于c++ - 快板 5 : trouble storing bitmaps in a std map,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43746948/
我正在开发一个小型图书馆,我需要做的一件事是让访问者访问一些数据并返回结果。 在一些较旧的 C++ 代码中,访问者需要声明一个 typedef return_type .例如,boost::stati
我正在尝试使用std:map类型的键和值制作std::any Visual Studio 2017 std::map m("lastname", "Ivanov"); std::cout (m["la
我已经在 C++ 的 map 中声明了一个集合为 std::map> .如何循环访问或打印设定值? 最佳答案 如果你知道如何迭代 std::map或 std::set单独地,您应该可以毫无问题地组合迭
如何循环? 我已经试过了: //----- code std::vector >::iterator it; for ( it = users.begin(); it != users.end();
我有两个用例。 A.我想同步访问两个线程的队列。 B.我想同步两个线程对队列的访问并使用条件变量,因为其中一个线程将等待另一个线程将内容存储到队列中。 对于用例 A,我看到了使用 std::lock_
我正在查看这两种类型特征的文档,但不确定有什么区别。我不是语言律师,但据我所知,它们都适用于“memcpy-able”类型。 它们可以互换使用吗? 最佳答案 不,这些术语不能互换使用。这两个术语都表示
我有以下测试代码,其中有一个参数 fS,它是 ofstream 的容器: #include #include #include #include int
这是这个问题的延续 c++ function ptr in unorderer_map, compile time error 我试图使用 std::function 而不是函数指针,并且只有当函数是
std::unordered_map str_bool_map = { {"a", true}, {"b", false}, {"c", true} }; 我们可以在此映射上使
我有以下对象 std::vector> vectorList; 然后我添加到这个使用 std::vector vec_tmp; vec_tmp.push_back(strDRG); vec_tmp.p
为什么 std::initializer_list不支持std::get<> , std::tuple_size和 std::tuple_element ?在constexpr中用得很多现在的表达式,
我有一个像这样定义的变量 auto drum = std::make_tuple ( std::make_tuple ( 0.3f , Ex
假设我有一个私有(private)std::map在我的类(class)里std::map 。我怎样才能将其转换为std::map返回给用户?我想要下面的原型(prototype) const std
假设我有一个私有(private)std::map在我的类(class)里std::map 。我怎样才能将其转换为std::map返回给用户?我想要下面的原型(prototype) const std
问题 我正在尝试将 lambda 闭包传递给 std::thread,它使用任意封闭参数调用任意封闭函数。 template std::thread timed_thread(Function&& f
我想创建一个模板类,可以容纳容器和容器的任意组合。例如,std::vector或 std::map ,例如。 我尝试了很多组合,但我必须承认模板的复杂性让我不知所措。我编译的关闭是这样的: templ
我有一个 std::vector>我将其分配给相同类型的第二个 vector 。 我收到这个编译器错误: /opt/gcc-8.2.0/include/c++/8.2.0/bits/stl_algob
有时候,我们有一个工厂可以生成一个 std::unique_ptr vector ,后来我们想在类/线程/你命名的之间共享这些指针。因此,最好改用 std::shared_ptr 。当然有一种方法可以
这个问题在这里已经有了答案: Sorting a vector of custom objects (14 个答案) 关闭 6 年前。 我创建了一个 vector vector ,我想根据我定义的参
我有三个类(class)成员: public: std::vector > getObjects(); std::vector > getObjects() const; privat
我是一名优秀的程序员,十分优秀!