c++ - c++ 在使用 std::multimap 时出现运行时错误的可能性很小
问题描述
有时我在使用时遇到运行时错误multimap
std::async
。调试模式下的 Visual2019 显示此错误:
表达式:无法取消引用结束映射/设置迭代器。
产生错误的代码示例:
#include <iostream>
#include <map>
#include <future>
#include <mutex>
#include <Windows.h>
class MyClass
{
public:
MyClass()
{
mp.emplace(mapsize, 'f');
mapsize += 1;
ft = std::async([this]() {
mx.lock();
while (true) {
for (int i = 0; i < mapsize; i++) {
auto pr = mp.equal_range(i);
for (auto j = pr.first; j != pr.second; j++)
std::cout << j->second << "\n";}}
mx.unlock(); });
}
private:
std::mutex mx;
static int mapsize;
std::future <void>ft;
static std::multimap <int, char> mp;
};
int MyClass::mapsize;
std::multimap <int, char> MyClass::mp;
int main()
{
for (int i = 0; i < 100000; i++)
new MyClass();
}
编辑:我做了一些同步,但它仍然产生同样的错误
解决方案
std::async
默认情况下在单独的线程中运行。因此,您在mp
没有同步的情况下从多个线程访问同一个对象 ( )。这被称为竞态条件,一种未定义的行为。
只有在 (1) 多个读取器、0 个写入器或 (2) 0 个读取器、1 个写入器的情况下,才能对同一对象进行非同步并行访问。在所有其他情况下,访问应该被序列化,例如使用mutex
.
但请注意,当使用互斥体时,对共享对象的所有访问都必须受到同一个互斥体的保护。所以应该制作互斥体并在andstatic
周围使用。mp.emplace
mapsize += 1
此外,为了更好的异常安全性,请使用unique_lock
or lock_guard
(RAII) 而不是手动锁定互斥锁:
class MyClass
{
public:
MyClass()
{
std::lock_guard<std::mutex> lock(mtx);
mp.emplace(mapsize, 'f');
mapsize += 1;
ft = std::async([this]() {
while (true) {
std::lock_guard<std::mutex> lock(mtx);
for (int i = 0; i < mapsize; i++) {
auto pr = mp.equal_range(i);
for (auto j = pr.first; j != pr.second; j++)
std::cout << j->second << "\n";
}
}
});
}
private:
std::future <void>ft;
static std::mutex mtx; // protects everything from here on down
static int mapsize;
static std::multimap <int, char> mp;
};
int MyClass::mapsize;
std::mutex MyClass::mtx;
std::multimap <int, char> MyClass::mp;
int main()
{
for (int i = 0; i < 100000; i++)
new MyClass();
}
推荐阅读
- python - BytesIO() 对象是完全加载到内存中还是流式传输?
- go - 解锁前返回的 mutex.Lock 函数
- c# - 无法将位图 TIFF 图像加载到矩阵
- python - 比较用户给出的两个单词作为烧瓶中的 Web 应用程序的相似度
- javascript - HTML 中的井字棋盘
- typescript - 如何区分打字稿中的日期?
- javascript - 组件正在更改要控制的非受控输入。这可能是由于值从未定义更改为已定义值
- kubernetes - 我需要部署什么版本的软件才能获得 kubernetes 1.19.7
- windows - 如何从损坏的地方恢复 WinSCP 中的文件传输
- python - 尝试在集合中查找项目时出现 Python mongodb/motor“'ObjectId' 对象不可迭代”错误