std::atomic<T>::fetch_add

来自cppreference.com
< cpp‎ | atomic‎ | atomic
 
 
并发支持库
线程
(C++11)
(C++20)
(C++20)
this_thread 命名空间
(C++11)
(C++11)
(C++11)
互斥
(C++11)
(C++11)  
通用锁管理
(C++11)
(C++11)
(C++11)
(C++11)
(C++11)
条件变量
(C++11)
信号量
闩与屏障
(C++20)
(C++20)
未来体
(C++11)
(C++11)
(C++11)
(C++11)
安全回收
(C++26)
(C++26)
风险指针





原子类型
(C++11)
(C++20)
原子类型的初始化
(C++11)(C++20 中弃用)
(C++11)(C++20 中弃用)
内存定序
原子操作的自由函数
原子标志的自由函数
 
 
atomic<整数类型 > 特化atomic<浮点类型 > 特化 (C++20 起)才有的成员
T fetch_add( T arg, std::memory_order order =
                        std::memory_order_seq_cst ) noexcept;
(1) (C++11 起)
T fetch_add( T arg, std::memory_order order =
                        std::memory_order_seq_cst ) volatile noexcept;
(2) (C++11 起)
atomic<T*> 部分特化才有的成员
T* fetch_add( std::ptrdiff_t arg,

              std::memory_order order =

                  std::memory_order_seq_cst ) noexcept;
(3) (C++11 起)
T* fetch_add( std::ptrdiff_t arg,

              std::memory_order order =

                  std::memory_order_seq_cst ) volatile noexcept;
(4) (C++11 起)

以值和 arg 的算术加法结果原子地替换当前值。运算是读-修改-写操作。按照 order 的值影响内存。

1,2) 对于有符号整数类型,定义算术使用补码表示进行。没有未定义的结果。

对于浮点类型,生效的浮点环境可能异于调用方线程的浮点环境。操作不必遵循对应的 std::numeric_limits 特征,但鼓励这么做。如果结果不是它的类型所能表示的值,那么结果未指定,但该操作没有未定义行为。

(C++20 起)
3,4) 结果可能是未定义的地址,但该运算无未定义行为。
如果 T 不是完整对象类型,那么程序非良构。


重载 (2)(4) 在参与重载决议且 std::atomic<T>::is_always_lock_freefalse 时被弃用。

(C++20 起)

参数

arg - 算术加法的另一实参
order - 强制的内存定序约束

返回值

*this修改顺序中,紧接此函数生效之前的值。

示例

#include <array>
#include <atomic>
#include <iostream>
#include <thread>
 
std::atomic<long long> data{10};
std::array<long long, 5> return_values{};
 
void do_work(int thread_num)
{
    long long val = data.fetch_add(1, std::memory_order_relaxed);
    return_values[thread_num] = val;
}
 
int main()
{
    {
        std::jthread th0{do_work, 0};
        std::jthread th1{do_work, 1};
        std::jthread th2{do_work, 2};
        std::jthread th3{do_work, 3};
        std::jthread th4{do_work, 4};
    }
 
    std::cout << "结果:" << data << '\n';
 
    for (long long val : return_values)
        std::cout << "看到返回值:" << val << std::endl;
}

可能的输出:

结果:15
看到返回值:11
看到返回值:10
看到返回值:14
看到返回值:12
看到返回值:13

缺陷报告

下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。

缺陷报告 应用于 出版时的行为 正确行为
P0558R1 C++11 允许进行指向(可有 cv 限定的)void 或函数的指针运算 使之非良构

参阅

将非原子值加到原子对象上,并获得原子对象的先前值
(函数模板)
令原子值增加或减少一
(公开成员函数)