美文网首页
boost::exception_ptr类的使用,跨线程获取异常

boost::exception_ptr类的使用,跨线程获取异常

作者: FredricZhu | 来源:发表于2021-04-10 15:05 被阅读0次

程序代码如下,非常具有自解释性,这里就不多说了。

CMakeLists.txt

cmake_minimum_required(VERSION 2.6)
project(lexical_cast)

add_definitions(-std=c++14)

include_directories("/usr/local/include")
link_directories("/usr/local/lib")
file( GLOB APP_SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/*.cpp)
foreach( sourcefile ${APP_SOURCES} )
    file(RELATIVE_PATH filename ${CMAKE_CURRENT_SOURCE_DIR} ${sourcefile})
    string(REPLACE ".cpp" "" file ${filename})
    add_executable(${file} ${sourcefile})
    target_link_libraries(${file} boost_filesystem boost_thread boost_system boost_serialization pthread boost_chrono)
endforeach( sourcefile ${APP_SOURCES} )

utils/tasks_processor_base.hpp

#ifndef _CHAPTOR06_02_TASKS_PROCESSOR_BASE_HPP_
#define _CHAPTOR06_02_TASKS_PROCESSOR_BASE_HPP_
#include <boost/noncopyable.hpp>
#include <boost/thread/thread.hpp>
#include <boost/asio/io_service.hpp>

#include <iostream>

namespace detail {
    template <class T>
    struct task_wrapped {
        private:
            T task_unwrapped_;
    
        public:
            explicit task_wrapped(const T& task_unwrapped): task_unwrapped_(task_unwrapped) {}

            void operator()() const {
                // 设置中断点,线程会在中断点退出
                try {
                    boost::this_thread::interruption_point();
                } catch (const boost::thread_interrupted&) {}

                try {
                    // 执行task
                    task_unwrapped_();
                } catch (const std::exception& e) {
                    std::cerr << "Exception: " << e.what() << '\n';
                } catch (const boost::thread_interrupted&) {
                    std::cerr << "Thread Interrupted...\n"; 
                } catch(...) {
                    std::cerr << "Unknown exception...\n";
                }
            }
    };

    template <class T>
    inline task_wrapped<T> make_task_wrapped(const T& task_unwrapped) {
        return task_wrapped<T>(task_unwrapped);
    }
}; // namespace detail

namespace tp_base {
    class tasks_processor: private boost::noncopyable {
        protected:
            boost::asio::io_service ios_;
            boost::asio::io_service::work work_;
        
            tasks_processor(): ios_(),
                work_(ios_) 
            {}

        public:
            static tasks_processor& get();

            template <class T>
            inline void push_task(const T& task_unwrapped) {
                ios_.post(detail::make_task_wrapped(task_unwrapped));
            }

            void start() {
                ios_.run();
            }

            void stop() {
                ios_.stop();
            }
    };

    tasks_processor& tasks_processor::get() {
        static tasks_processor proc;
        return proc;
    }
}; // namespace tp_base


#endif

main.cpp

#include "utils/tasks_processor_base.hpp"

#include <boost/exception_ptr.hpp>
#include <boost/lexical_cast.hpp>
#include <stdexcept>

using namespace tp_base;

// 前向声明
void func_test2();

void process_exception(const boost::exception_ptr& exec_ptr) {
    try {
        boost::rethrow_exception(exec_ptr);
    } catch(const boost::bad_lexical_cast& /*e*/) {
        std::cerr << "Lexical cast exception detected \n";
        tasks_processor::get().push_task(&func_test2);
    } catch(...) {
        std::cerr << "Can not handle such exceptions: \n"
            << boost::current_exception_diagnostic_information()
            << "\n";
        // 停止io_service::run
        tasks_processor::get().stop();
    }
}

void func_test1() {
    try {
        boost::lexical_cast<int>("oops!");
    } catch(...) {
        auto exec = boost::current_exception();
        tasks_processor::get().push_task([exec](){
            process_exception(exec);
        });
    }
}

void func_test2() {
    try {
        BOOST_THROW_EXCEPTION(std::logic_error("Some fatal logic error"));
    } catch(...) {  
        auto exec = boost::current_exception();
        tasks_processor::get().push_task([exec](){
            process_exception(exec);
        });
    }
}

void run_sub_thread_throw(boost::exception_ptr& ptr) {

    
    try {
        boost::lexical_cast<float>("not-a-float");
    } catch(...) {
        ptr = boost::current_exception();
    }
}

int main(int argc, char* argv[]) {
    tasks_processor::get().push_task(&func_test1);
    tasks_processor::get().start();

    boost::exception_ptr ptr;
    boost::thread t([&ptr](){
        run_sub_thread_throw(ptr);
    });
    t.join();

    // 子线程有异常,在主线程中抛出异常
    if(ptr) {
        std::cerr << "bad exception occurred in sub thread\n";
        boost::rethrow_exception(ptr);
    }
}

程序输出如下,


图片.png

然后抽了两个小时时间看了下boost::exception_ptr类和boost::exception类的实现,画了下类图,受益匪浅,以后出问题查源码可能用的着。

exception_wrapper的继承链,


图片.png

boost::exception类的构造


图片.png

boost::exception_ptr类的构造


图片.png

相关文章

网友评论

      本文标题:boost::exception_ptr类的使用,跨线程获取异常

      本文链接:https://www.haomeiwen.com/subject/haknkltx.html