在 lambda 中传递参数时无法调用指向成员的指针

Unable to call pointer-to-member when passing arguments inside lambda

我正在尝试为 std::thread 制作一个包装器,它允许我查看正在调用的线程上抛出了哪些异常。我在加入线程时向上传播它们。这在调用带参数的函数或不带参数的 class 成员函数时效果很好。但是,如果我尝试使其与带有参数的 class 成员函数一起使用,我会收到一个我无法理解的错误。下面是一些示例代码,可重现我所看到的问题。

#include <iostream>
#include <string>

#include <thread>
#include <exception>
#include <utility>

class BlazingThread {
public:
    template< typename Function, typename... Args >
    explicit BlazingThread( Function&& f, Args&&... args ){
        exception = nullptr;
        thread = std::thread(([&](){
            try{
                auto caller = std::forward<Function>(f);

                caller(args...);
            }catch(...){
                //non handled exception rethrow so we can stop whoever started this mess
                //exception = std::make_exception_ptr(BlazingException("An unknown error occurred!"));
            }

        }));

    }

    template< typename Function >
    explicit BlazingThread( Function&& f){
        exception = nullptr;
        thread = std::thread(([&](){
            try{
                //std::bind(f);
                std::forward<Function>(f);

            }catch(...){
                //non handled exception rethrow so we can stop whoever started this mess
            //  exception = std::make_exception_ptr(BlazingException("An unknown error occurred!"));
            }

        }));

    }

    BlazingThread(BlazingThread && other){
        this->thread = std::move(other.thread);
        this->exception = std::move(other.exception);
    }

    BlazingThread(){

    }

    BlazingThread& operator=(const BlazingThread&) = delete;
    BlazingThread& operator=(BlazingThread && other){
        this->thread = std::move(other.thread);
        this->exception = std::move(other.exception);
        return *this;
    }
    BlazingThread(const BlazingThread& other) = delete;
    virtual ~BlazingThread();

    void join();
    static unsigned int hardware_concurrency(){
        return std::thread::hardware_concurrency();
    }
private:
    std::thread thread;
    std::exception_ptr exception;
};





void BlazingThread::join(){
    thread.join();
    if(this->exception != nullptr){
        //an exception was thrown in the thread, lets rethrow it
        std::rethrow_exception(this->exception);
    }
}

BlazingThread::~BlazingThread() {
    // TODO Auto-generated destructor stub
}





void testFunction(int x, int y){
    int z = x + y;
    std::cout<<"x + y = "<<z<<std::endl;
    //throw BlazingException("A planned error!");
}

class TestClass{
public:
    TestClass(int newX){
        this->x = newX;
    }

    void talk(){
        std::cout<<"this is "<<x<<std::endl;
    }
    void talking(){

        BlazingThread thread2(&TestClass::talk);
        thread2.join();
        std::string msg = "something else ";
        BlazingThread thread3(&TestClass::talkSomething, msg);
        thread3.join();



    }

    void talkSomething(std::string testMsg){
        std::cout<<testMsg<<x<<std::endl;
    }
private:
    int x;
};

int main() {
    try{
        TestClass test(3);
        test.talk();

        BlazingThread thread(testFunction,2,4);

        thread.join();
    }catch(...){
        //std::cout<<"Found BlazingException, what = "<<e.what()<<std::endl;
    }
    return 0;
}

    BlazingThread thread3(&TestClass::talkSomething, msg);
    thread3.join();

阻止编译。我看到的错误是

/usr/include/c++/5/functional: In instantiation of ‘struct std::_Bind_check_arity<void (TestClass::*)(std::__cxx11::basic_string<char>), std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >&>’:
    /usr/include/c++/5/functional:1439:12:   required from ‘struct std::_Bind_helper<false, void (TestClass::*)(std::__cxx11::basic_string<char>), std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >&>’
    /usr/include/c++/5/functional:1462:5:   required by substitution of ‘template<class _Func, class ... _BoundArgs> typename std::_Bind_helper<std::__is_socketlike<_Func>::value, _Func, _BoundArgs ...>::type std::bind(_Func&&, _BoundArgs&& ...) [with _Func = void (TestClass::*)(std::__cxx11::basic_string<char>); _BoundArgs = {std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >&}]’
    ../src/simple-thread.cpp:24:34:   required from ‘BlazingThread::BlazingThread(Function&&, Args&& ...)::<lambda()> [with Function = void (TestClass::*)(std::__cxx11::basic_string<char>); Args = {std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >&}]’
    ../src/simple-thread.cpp:24:80:   required from ‘struct BlazingThread::BlazingThread(Function&&, Args&& ...) [with Function = void (TestClass::*)(std::__cxx11::basic_string<char>); Args = {std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >&}]::<lambda()>’
    ../src/simple-thread.cpp:21:10:   required from ‘BlazingThread::BlazingThread(Function&&, Args&& ...) [with Function = void (TestClass::*)(std::__cxx11::basic_string<char>); Args = {std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >&}]’
    ../src/simple-thread.cpp:121:55:   required from here
    /usr/include/c++/5/functional:1426:7: error: static assertion failed: Wrong number of arguments for pointer-to-member
           static_assert(_Varargs::value
           ^

下面是一个工作版本,我需要修复它,以便它需要 std::ref 作为参考,例如 std::thread

    #include <iostream>
    #include <string>

    #include <thread>
    #include <exception>
    #include <utility>

    class BlazingThread {
    public:




        template< typename Function, typename... Args >
        explicit BlazingThread( Function&& f, Args&&... args ){
            exception = nullptr;

            thread = std::thread(([&f,args...](){
                try{
                    //auto caller = std::forward<Function>(f);

                    //caller(std::forward<Args>(args)...);
                    auto functionCall = std::bind(std::forward<Function>(f),args...);
                    functionCall();
                }catch(...){
                    //non handled exception rethrow so we can stop whoever started this mess
                    //exception = std::make_exception_ptr(BlazingException("An unknown error occurred!"));
                }

            }));

        }

        template< typename Function >
        explicit BlazingThread( Function&& f){
            exception = nullptr;
            thread = std::thread(([&](){
                try{
                    //std::bind(f);
                    std::forward<Function>(f)();

                }catch(...){
                    //non handled exception rethrow so we can stop whoever started this mess
                //  exception = std::make_exception_ptr(BlazingException("An unknown error occurred!"));
                }

            }));

        }

        BlazingThread(BlazingThread && other){
            this->thread = std::move(other.thread);
            this->exception = std::move(other.exception);
        }

        BlazingThread(){

        }

        BlazingThread& operator=(const BlazingThread&) = delete;
        BlazingThread& operator=(BlazingThread && other){
            this->thread = std::move(other.thread);
            this->exception = std::move(other.exception);
            return *this;
        }
        BlazingThread(const BlazingThread& other) = delete;
        virtual ~BlazingThread();

        void join();
        static unsigned int hardware_concurrency(){
            return std::thread::hardware_concurrency();
        }
    private:
        std::thread thread;
        std::exception_ptr exception;
    };





    void BlazingThread::join(){
        thread.join();
        if(this->exception != nullptr){
            //an exception was thrown in the thread, lets rethrow it
            std::rethrow_exception(this->exception);
        }
    }

    BlazingThread::~BlazingThread() {
        // TODO Auto-generated destructor stub
    }


    void testFunction(){
        std::cout<<"tester"<<std::endl;
        //throw BlazingException("A planned error!");
    }


    void testFunction2(int x, int y){
        int z = x + y;
        std::cout<<"x + y = "<<z<<std::endl;
        //throw BlazingException("A planned error!");
    }

    class TestClass{
    public:
        TestClass(int newX){
            this->x = newX;
        }

        void talk(){
            std::cout<<"this is "<<x<<std::endl;
        }
        void talking(){

            BlazingThread thread2(&TestClass::talk,this);
            thread2.join();
            std::string msg = "something else ";
            BlazingThread thread3(&TestClass::talkSomething,this, msg);
            thread3.join();



        }

        void talkSomething(std::string testMsg){
            std::cout<<testMsg<<x<<std::endl;
        }
    private:
        int x;
    };

    int main() {
        try{
            TestClass test(3);
            test.talking();

            BlazingThread thread(testFunction2,2,4);

            thread.join();
            BlazingThread thread2(testFunction);

            thread2.join();
        }catch(...){
            //std::cout<<"Found BlazingException, what = "<<e.what()<<std::endl;
        }
        return 0;
    }

改变

BlazingThread thread2(&TestClass::talk);

BlazingThread thread2(&TestClass::talk, this);

和类似的其他地方。您需要传递要处理的对象。

当 lambda 超出当前作用域时(就像将它传递给 std 线程时一样),也停止使用 [&] 捕获。但这只是一个运行时错误。