Welcome to OGeek Q&A Community for programmer and developer-Open, Learning and Share
Welcome To Ask or Share your Answers For Others

Categories

0 votes
650 views
in Technique[技术] by (71.8m points)

c++ - Can I make a thread-safe std::atomic<vector<int>>?

I'm having a function that needs to be executed n=1000 times. This functions does a Monte Carlo style simulation and returns an int as the result. I'd like to run nthreads=4 in parallel. Whenever a thread finishes one cycle, it should put the result in a std::vector<int>. Thus, after 1000 cycles, I've a vector of 1000 ints that can be examined by statistics.

Since a std::vector is not thread-safe, I thought about std::mutex (which would surely work).

But I wonder if I can declare a vector to be atomic and thus get around mutexes? Is it possible to have a std::atomic<std::vector<int>>? And can I use push_back etc. on it?

See Question&Answers more detail:os

与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome To Ask or Share your Answers For Others

1 Reply

0 votes
by (71.8m points)

C++11 §29.5/1 says

There is a generic class template atomic. The type of the template argument T shall be trivially copyable (3.9).

What does trivially copyable mean?

§3.9 tells

Scalar types, trivially copyable class types (Clause 9), arrays of such types, and cv-qualified versions of these types (3.9.3) are collectively called trivially copyable types.

For class types (of which std::vector is):

A trivially copyable class is a class that:

  • has no non-trivial copy constructors
  • has no non-trivial move constructors
  • has no non-trivial copy assignment operators
  • has no non-trivial move assignment operators
  • has a trivial destructor

According to this list std::vector is not trivially copyable and so you cannot use std::atomic<std::vector<int>>.

Since you know the size in advance and since you do not need to use methods that would require the vector be reallocated in a different location (like push_back). You can use std::vector<int>::resize or the size constructor to preallocate and preconstruct the required ints. Therefore your concurrent threads do not need to operate on the vector itself but on the elements.

If there is no access from different threads to the same element there is no race condition.

The same goes for int k[1000] which is trivially copyable. But you do not need it to be since the threads do not change the array/vector/list itself but the elements.


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
OGeek|极客中国-欢迎来到极客的世界,一个免费开放的程序员编程交流平台!开放,进步,分享!让技术改变生活,让极客改变未来! Welcome to OGeek Q&A Community for programmer and developer-Open, Learning and Share
Click Here to Ask a Question

...