2
私はスレッドとスレッドの並行性を実践しています。私は自分のスレッドプールを作りたいと思います。私はスレッドのベクトルを持っているため、これらのスレッドは次の関数を別のベクトルから実行するために条件変数で待機します。スレッドプールに関数を渡します。
しかし、未知の引数を持つ関数の格納/渡し/実行に問題があります。誰かが私にこれを行う方法やこれに必要なことをヒントを与えることができますか?
ThreadPool tp(10);
tp.execute(std::bind(print, i));
void print(int i) {
std::cout << i << std::endl;
}
void ThreadPool::execute(const std::function<void()> function) {
l.lock();
if (running) {
queue.push_back(function);
cv.notify_one();
} else {
// TODO error
std::cout << "err execute()\n";
}
l.unlock();
}
糸ループ:
// set up threads in constructor
for (int i = 0; i < size; i++) {
threads.push_back(std::thread(&ThreadPool::thread_loop, std::ref(*this), i));
}
static void ThreadPool::thread_loop(ThreadPool &pool, int id) {
pool.o.lock();
std::cout << "pool-" << pool.id << "-thread-" << id << " started\n";
pool.o.unlock();
std::function<void()> function;
std::unique_lock<std::mutex> ul(pool.m, std::defer_lock);
while (pool.running || pool.queue.size() > 0) {
ul.lock();
while (pool.queue.size() == 0) {
pool.cv.wait(ul);
}
pool.l.lock();
if (pool.queue.size() == 0) {
pool.l.unlock();
continue;
}
function = pool.queue.at(0);
pool.queue.erase(pool.queue.begin());
pool.l.unlock();
ul.unlock();
function();
}
}
これは 'のstd ::機能'を頼みます。これで、0引数の呼び出し可能なもの(関数、関数オブジェクト、ラムダ、 'std :: bind'の結果、あなたの名前)をスレッドプールに渡すことができます。 –
m_threads.push_back(std :: thread(&InitThreadsFoo、m_threads.size()));を実行できます。既存のスレッドをベクトルから再割り当てして別の関数を引き継ぐ方法や、すでに使用されている既存のスレッドに関数を割り当てる方法も知っています。 – ReturnVoid
@Andrey Turkinありがとう、これは素晴らしい作品です! – Pali