Skip to content

Latest commit

 

History

History
86 lines (62 loc) · 3.78 KB

atomic_exchange.md

File metadata and controls

86 lines (62 loc) · 3.78 KB

atomic_exchange

  • memory[meta header]
  • std[meta namespace]
  • function template[meta id-type]
  • cpp11[meta cpp]
  • cpp20deprecated[meta cpp]
namespace std {
  template <class T>
  shared_ptr<T> atomic_exchange(shared_ptr<T>* p, shared_ptr<T> r);
}

この関数は、C++20から非推奨となった。アトミックアクセスの対象としているshared_ptrshared_ptrに対するatomic特殊化で置き換えることで同等の機能を使用できる。

概要

shared_ptrオブジェクトを、アトミックに入れ替える。

要件

p != nullptrであること。

戻り値

atomic_exchange_explicit(p, r, memory_order_seq_cst)
  • atomic_exchange_explicit[link atomic_load_explicit.md]
  • memory_order_seq_cst[link /reference/atomic/memory_order.md]

例外

投げない

非推奨・削除の詳細

この関数はフリー関数であるため、この関数によってアトミックにアクセスする対象となるshared_ptrオブジェクトそのものはどこかに配置されている非アトミックオブジェクトである。そのため、アトミックアクセスしたい文脈の外側から通常のアクセスが可能であり、もし別のスレッドからそのようなアクセスが行われているとこの関数を用いていてもデータ競合を引き起こし未定義動作となる。

すなわち、アトミックにアクセスしたいshared_ptrオブジェクトに対する全てのアクセスをプログラマがきちんと管理しなければこの関数の使用は安全ではなく、それはかなり困難であったためこの関数は危険な利用がデフォルトとなっていた。

そのため、この関数(とそのファミリ)は非推奨とされ、代わりにshared_ptrに対するatomic特殊化が追加された。shared_ptrに対するatomic特殊化を利用すれば、アトミックにアクセスする対象となるshared_ptrオブジェクトそのものをアトミックオブジェクトとすることができるため、どこからアクセスしたとしても全てのアクセスは自動的にアトミックアクセスとなり、前述の問題は回避できる。

この関数からshared_ptrに対するatomic特殊化に移行する場合は、元のコードでアトミックアクセス対象となっていたshared_ptrオブジェクトの型をstd::atomic<std::shared_ptr>に変更することで移行でき、その場合はatomicのために用意されているフリー関数が代わりに使用される(宣言されているヘッダが異なるため、<atomic>ヘッダのインクルードが必要となるかもしれない)。

#include <iostream>
#include <memory>

int main()
{
  std::shared_ptr<int> a(new int(1));
  std::shared_ptr<int> b(new int(2));

  std::shared_ptr<int> prev_state = std::atomic_exchange(&a, b);

  std::cout << *a << std::endl;
  std::cout << *prev_state << std::endl;
}
  • std::atomic_exchange[color ff0000]

出力

2
1

バージョン

言語

  • C++11

処理系

参照