Skip to content

Latest commit

 

History

History
222 lines (181 loc) · 6.35 KB

README.md

File metadata and controls

222 lines (181 loc) · 6.35 KB

CppThreadPool

Introduction

The original intention of this project is to learn the new C++20 standard in use. Therefore, make sure your compiler supports C++20 or later standards.

TODO :

  • Smart pointer as parameter
    • Solve memory leaking of shared_ptr + lambda
  • New function wrapper that supports move semantics
  • Locked task queue
    • Add dispatcher struct
  • Traditional mode support
  • Unlocked task queue
  • Coroutine supported
  • (Try) High concurrency support
    • High concurrency test

Installation

  • Clone the repo.
git clone https://github.com/JieRen98/CppThreadPool.git
cd CppThreadPool
  • Include the directory in your project.
# ...
include_directories("path/to/the/dir/CppThreadPool")
# ...
  • Enjoy.

Example

Compile test program

mkdir test

echo "
#include <iostream>
#include <ThreadPool.hpp>

void run_ref(int &x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(10));
    x++;
    std::cout << "value address: " << &x << " in run_ref" << std::endl;
}

void run_cp(int x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(20));
    x++;
    std::cout << "value address: " << &x << " in run_cp" << std::endl;
}

std::unique_ptr<int> ret_unique(int x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(30));
    x++;
    std::cout << "value address: " << &x << " in ret_unique" << std::endl;
    return std::make_unique<int>(x);
}

std::string run_ref_ret_value(int &x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(40));
    x++;
    std::cout << "value address: " << &x << " in run_ref_ret_value" << std::endl;
    return "string from run_ref_ret_value";
}

int &run_ref_ret_ref(int &x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(50));
    x++;
    std::cout << "value address: " << &x << " in run_ref_ret_ref" << std::endl;
    return x;
}

int main() {
    ThreadPool::ThreadPool tp{5};
    auto sptr = std::make_shared<int>(0);
    auto uptr = std::make_unique<int>(0);
    std::cout << "value address: " << sptr.get() << ", start with: " << *sptr << std::endl;
    tp.start();

    auto future1 = tp.submit(run_ref, sptr)->get_future();
    auto future2 = tp.submit(run_cp, sptr)->get_future();
    auto future3 = tp.submit(ret_unique, sptr)->get_future();
    auto future4 = tp.submit(run_ref_ret_value, std::move(uptr))->get_future();
    auto sptr5 = std::make_shared<int>(0);
    auto future5 = tp.submit(run_ref_ret_ref, sptr5)->get_future();
    future1.wait();
    future2.wait();
    future3.wait();
    future4.wait();
    future5.wait();
    std::cout << "sptr: " << *sptr << ", use count: " << sptr.use_count() << std::endl;
    auto result3 = std::move(future3.get());
    std::cout << "Return of future3: " << *result3 << std::endl;
    std::cout << future4.get() << std::endl;
    auto &future5_value = future5.get();
    std::cout << "run_ref_ret_ref start with value: " << *sptr5 << std::endl;
    future5_value++;
    std::cout << "run_ref_ret_ref add 1: " << *sptr5 << std::endl;
    tp.shutdown();
}
" > test/test.cpp

echo "add_executable(test test.cpp)" > test/CMakeLists.txt

mkdir build && cd build

cmake .. && make

Run the program

cd test
./test

result:

value address: 0x15da110, start with: 0
value address: 0x15da110 in run_ref
value address: 0x7f09429e67ac in run_cp
value address: 0x7f09421e5774 in ret_unique
value address: 0x15da120 in run_ref_ret_value
value address: 0x15db1d0 in run_ref_ret_ref
sptr: 1, use count: 1
Return of future3: 1
string from run_ref_ret_value
run_ref_ret_ref start with value: 1
run_ref_ret_ref add 1: 2

Explanation

The test code as following:

#include <iostream>
#include <ThreadPool.hpp>

void run_ref(int &x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(10));
    x++;
    std::cout << "value address: " << &x << " in run_ref" << std::endl;
}

void run_cp(int x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(20));
    x++;
    std::cout << "value address: " << &x << " in run_cp" << std::endl;
}

std::unique_ptr<int> ret_unique(int x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(30));
    x++;
    std::cout << "value address: " << &x << " in ret_unique" << std::endl;
    return std::make_unique<int>(x);
}

std::string run_ref_ret_value(int &x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(40));
    x++;
    std::cout << "value address: " << &x << " in run_ref_ret_value" << std::endl;
    return "string from run_ref_ret_value";
}

int &run_ref_ret_ref(int &x) {
    std::this_thread::sleep_for(std::chrono::milliseconds(50));
    x++;
    std::cout << "value address: " << &x << " in run_ref_ret_ref" << std::endl;
    return x;
}

int main() {
    ThreadPool::ThreadPool tp{5};
    auto sptr = std::make_shared<int>(0);
    auto uptr = std::make_unique<int>(0);
    std::cout << "value address: " << sptr.get() << ", start with: " << *sptr << std::endl;
    tp.start();

    auto future1 = tp.submit(run_ref, sptr)->get_future();
    auto future2 = tp.submit(run_cp, sptr)->get_future();
    auto future3 = tp.submit(ret_unique, sptr)->get_future();
    auto future4 = tp.submit(run_ref_ret_value, std::move(uptr))->get_future();
    auto sptr5 = std::make_shared<int>(0);
    auto future5 = tp.submit(run_ref_ret_ref, sptr5)->get_future();
    future1.wait();
    future2.wait();
    future3.wait();
    future4.wait();
    future5.wait();
    std::cout << "sptr: " << *sptr << ", use count: " << sptr.use_count() << std::endl;
    auto result3 = std::move(future3.get());
    std::cout << "Return of future3: " << *result3 << std::endl;
    std::cout << future4.get() << std::endl;
    auto &future5_value = future5.get();
    std::cout << "run_ref_ret_ref start with value: " << *sptr5 << std::endl;
    future5_value++;
    std::cout << "run_ref_ret_ref add 1: " << *sptr5 << std::endl;
    tp.shutdown();
}

As we can see, we support unique_ptr and shared_ptr. Notably, we should use std::move to transfer the control of the unique_ptr.

At the same time, thanks to the pointer setting, we can easily use lvalue reference in our submitted function. Several existing thread pools have difficulty in passing reference. Some require users to pass reference by using std::ref that will cause some problems in type deduction.