《C++ Concurrency in Action》笔记24 std::atomic<T*>、std::atomic<整型>

5.2.4 std::atomic<T*>相关操作

某种类型指针的原子类型就是std::atomic<T*>,就像bool类型的原子类型是std::atomic<bool>一样。接口从本质上来说是一样的。它不支持拷贝构造和拷贝赋值,尽管他可以从对应的非原子类型的指针构造以及赋值。它也有load(),store() ,exchange(),compare_exchange_weak(),以及compare_exchange_strong()。

std::atomic<T*>增加了指针相关的操作。基本的操作是 fetch_add()和 fetch_sub(),分别代表地址相加和地址相减。还有+=、-=,还有前置及后置++、--。

这些重载的运算符就像你期待的那样,与內建类型一样:如果 x是std::atomic<Foo*>类型的对象,并且它指向一个Foo数组,那么x+=3代表把它向后移动3个位置,返回的是一个原始的Foo*,指向相对于之前位置的第4个元素,并不返回std::atomic<Foo*>对象的引用。但是 fetch_add()和fetch_sub()返回的却是执行函数前的原始指针。这些操作是一种read-modify-write类型的操作。

除了返回值不同以外,fetch_add()以及fetch_sub()这些命名函数与重载运算符函数的另一个不同之处在于,命名函数可以指定内存指令参数:

p.fetch_add(3,std::memory_order_release);

而重载运算符无法指定多余参数,它们的内存指令总是memory_order_seq_cst。

5.2.5 std::atomic<整型>

标准整型原子类型本质上都一样,拥有相同的一套操作。它们除了拥有load() ,  store() ,  exchange() ,  compare_exchange_weak() , compare_exchange_strong()这些函数外,它们还有一系列有用的函数:fetch_add() ,  fetch_sub() ,  fetch_and() ,  fetch_or() ,fetch_xor(),以及相应的重载函数 += ,  -= ,  ++x,x++,--x,x--,&= ,  |= , ^=。这些操作几乎涵盖了所有整型的操作需要,除了除法、乘法、位移动这3个操作。因为整型原子类型几乎被用来作为计数器或者位掩码,所以也称不上是什么缺失。如果需要的话,可以循环使用compare_exchange_weak()函数去实现这3个操作。



以下是一些可能的 C++ 语言优化方法: 1. 对于文件不存在的情况,可以使用异常机制来处理,避免直接退出程序。 ```c++ try { std::ifstream infile(filename); if (!infile) { throw std::runtime_error("文件打开失败"); } // 读取文件内容并输出到屏幕上 std::string data; while (std::getline(infile, data)) { std::cout << data << std::endl; } // 关闭文件 infile.close(); } catch (std::exception& ex) { std::cerr << ex.what() << std::endl; return 1; } ``` 2. 在读取文件时,可以使用文件流对象的 read 函数进行二进制读取,避免字符编码的转换。 ```c++ std::ifstream infile(filename, std::ios::binary); if (!infile) { std::cerr << "文件打开失败" << std::endl; return 1; } // 获取文件大小 infile.seekg(0, std::ios::end); std::size_t filesize = infile.tellg(); infile.seekg(0, std::ios::beg); // 读取文件内容并输出到屏幕上 std::vector<char> buffer(filesize); infile.read(buffer.data(), filesize); std::cout.write(buffer.data(), filesize); // 关闭文件 infile.close(); ``` 3. 在读取文件时,可以使用多线程进行处理,提高读取效率。 ```c++ std::ifstream infile(filename, std::ios::binary); if (!infile) { std::cerr << "文件打开失败" << std::endl; return 1; } // 获取文件大小 infile.seekg(0, std::ios::end); std::size_t filesize = infile.tellg(); infile.seekg(0, std::ios::beg); // 读取文件内容并输出到屏幕上 std::vector<char> buffer(filesize); std::vector<std::future<void>> futures; for (int i = 0; i < std::thread::hardware_concurrency(); ++i) { std::size_t start = i * filesize / std::thread::hardware_concurrency(); std::size_t end = (i + 1) * filesize / std::thread::hardware_concurrency(); futures.push_back(std::async(std::launch::async, [&infile, &buffer, start, end]() { infile.seekg(start, std::ios::beg); infile.read(buffer.data() + start, end - start); })); } for (auto& future : futures) { future.wait(); } std::cout.write(buffer.data(), filesize); // 关闭文件 infile.close(); ``` 4. 在读取文件时,可以使用内存映射方式进行读取。需要注意的是,内存映射方式适用于较大的文件,且需要使用操作系统提供的 API。 ```c++ #include <sys/mman.h> #include <fcntl.h> #include <unistd.h> std::string filename = argv[1]; int fd = open(filename.c_str(), O_RDONLY); if (fd == -1) { std::cerr << "文件打开失败" << std::endl; return 1; } struct stat sb; if (fstat(fd, &sb) == -1) { std::cerr << "获取文件大小失败" << std::endl; close(fd); return 1; } char* addr = (char*)mmap(NULL, sb.st_size, PROT_READ, MAP_PRIVATE, fd, 0); if (addr == MAP_FAILED) { std::cerr << "内存映射失败" << std::endl; close(fd); return 1; } std::cout.write(addr, sb.st_size); munmap(addr, sb.st_size); close(fd); ``` 需要注意的是,内存映射方式需要手动进行内存释放和文件关闭操作。另外,对于 Windows 系统,需要使用 CreateFile 和 CreateFileMapping 等 API 来进行内存映射。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值