Written a lua threadpool
工作原理
由于lua只能单线程运行,该lib要求所有lua代码在单线程,而多线程部分只能为c代码
具体用法上要求多线程部分必须用c实现
相关模块
线程池
异步函数实现框架
Now only a sleep function is provided
Usage:
function test2_threadpool()
local tp = Dll.MyTHdPool()
local n =
local function f()
n = n+
print('f ' .. n)
if(n==) then return end
tp:sleep1(, f)
end f()
tp:join()
end
C codes:
#include "stdafx.h"
#include <luabind.hpp>
#include <vector>
#include <queue>
#include <boost/thread.hpp> using namespace luabind; #include "stdafx.h" #include <boost/asio.hpp>
#include <boost/bind.hpp>
#include <boost/thread/thread.hpp> #include <deque> class ThreadPool
{
boost::asio::io_service ioService;
boost::thread_group threadpool;
boost::asio::io_service::work work;
public:
ThreadPool() :work(ioService)
{
/*
* This will start the ioService processing loop. All tasks
* assigned with ioService.post() will start executing.
*/
//boost::asio::io_service::work work(ioService); /*
* This will add threads to the thread pool. (You could just put it in a for loop)
*/
threadpool.create_thread(
boost::bind(&boost::asio::io_service::run, &ioService)
);
threadpool.create_thread(
boost::bind(&boost::asio::io_service::run, &ioService)
); }
~ThreadPool()
{ } void post(boost::function<void()> f)
{
ioService.post(f);
} void join()
{
threadpool.join_all();
}
private: }; namespace bamthread
{
typedef std::unique_ptr<boost::asio::io_service::work> asio_worker; struct ThreadPool {
ThreadPool(size_t threads) :service(), working(new asio_worker::element_type(service)) {
while (threads--)
{
auto worker = boost::bind(&boost::asio::io_service::run, &(this->service));
g.add_thread(new boost::thread(worker));
}
} template<class F>
void post(F f){
service.post(f);
} ~ThreadPool() {
working.reset(); //allow run() to exit
g.join_all();
service.stop();
} private:
boost::asio::io_service service; //< the io_service we are wrapping
asio_worker working;
boost::thread_group g; //< need to keep track of threads so we can join them
};
} void my_task()
{
Sleep();
printf("mytask");
} void test1()
{
bamthread::ThreadPool tp();
tp.post(boost::bind(my_task));
//tp.join();
} void test()
{
/*
* Create an asio::io_service and a thread_group (through pool in essence)
*/
boost::asio::io_service ioService;
boost::thread_group threadpool; /*
* This will start the ioService processing loop. All tasks
* assigned with ioService.post() will start executing.
*/
boost::asio::io_service::work work(ioService); /*
* This will add threads to the thread pool. (You could just put it in a for loop)
*/
threadpool.create_thread(
boost::bind(&boost::asio::io_service::run, &ioService)
);
threadpool.create_thread(
boost::bind(&boost::asio::io_service::run, &ioService)
); /*
* This will assign tasks to the thread pool.
* More about boost::bind: "http://www.boost.org/doc/libs/1_54_0/libs/bind/bind.html#with_functions"
*/
ioService.post(boost::bind(my_task)); /*
* This will stop the ioService processing loop. Any tasks
* you add behind this point will not execute.
*/
ioService.stop(); /*
* Will wait till all the threads in the thread pool are finished with
* their assigned tasks and 'join' them. Just assume the threads inside
* the threadpool will be destroyed by this method.
*/
threadpool.join_all();
} template <typename T>
class queue
{
private:
boost::mutex d_mutex;
boost::condition_variable d_condition;
std::deque<T> d_queue;
public:
void push(T const& value) {
{
boost::unique_lock<boost::mutex> lock(this->d_mutex);
d_queue.push_front(value);
}
this->d_condition.notify_one();
}
T pop() {
boost::unique_lock<boost::mutex> lock(this->d_mutex);
this->d_condition.wait(lock, [=]{ return !this->d_queue.empty(); });
T rc(std::move(this->d_queue.back()));
this->d_queue.pop_back();
return rc;
}
}; class MyTHdPool
{
bamthread::ThreadPool tp; boost::mutex m;
std::map<int, boost::function<void()> > f2s; // key: taskid, value: post processing //boost::thread t_; queue<int> q_;
int taskid_; public:
MyTHdPool() :tp(), taskid_(){} ~MyTHdPool(){
join();
} void Call(boost::function<void()> f1, boost::function<void()> f2)
{
int taskid = taskid_++; printf("begin call task %d\n", taskid); boost::function<void()> f = [=]() mutable {
f1(); q_.push(taskid);
printf("done task %d\n", taskid);
}; {
boost::lock_guard<boost::mutex> lock(m);
f2s[taskid] = (f2);
} tp.post(f);
printf("end post task %d\n", taskid);
} void join()
{
while (true)
{
boost::function<void()> f2;
int taskid = ;
{
{
boost::lock_guard<boost::mutex> lock(m);
if (f2s.empty())
return;
} printf("start pop a task from queue\n");
int taskid = q_.pop();
printf("got a task %d from queue\n", taskid); {
boost::lock_guard<boost::mutex> lock(m);
auto it = f2s.find(taskid);
assert(it != f2s.end());
f2 = it->second;
f2s.erase(it);
}
} printf("exec task post ftn %d\n", taskid);
f2();
}
} void sleep1(double n, object f2)
{
Call([n](){Sleep(n * ); }, [f2, this]() mutable {
f2();
});
} void sleep2(double n)
{
Call([n](){Sleep(n * ); }, [](){});
}
private: }; void callback(object o)
{
printf("before callback\n");
o();
printf("after callback\n");
} int luaopen_Dll(lua_State* L)
{ luaL_openlibs(L);
open(L); // define a module in _G["t"]
module(L, "Dll")[ class_<MyTHdPool>("MyTHdPool")
.def(constructor<>())
.def("sleep1", &MyTHdPool::sleep1)
.def("sleep2", &MyTHdPool::sleep2)
.def("join", &MyTHdPool::join), def("test1", &test1), def("callback", &callback)
]; // push _G["t"] to stack
lua_getglobal(L, "Dll"); // set _G["t"]=nil
lua_pushnil(L);
lua_setglobal(L, "Dll"); return ;
}
Written a lua threadpool的更多相关文章
- [转]LUA元表
lua元表和元方法 <lua程序设计> 13章 读书笔记 lua中每个值都有一个元表,talble和userdata可以有各自独立的元表,而其它类型的值则共享其类型所属的单一元表.lua在 ...
- Lua 笔记
lua命令: #enter shell lua #excute script file lua xxx.lua lua脚本: #!/usr/local/bin/lua 核心概念: As a exten ...
- lua function
This is the main reference for the World of Warcraft Lua Runtime. Note that these are mostly standar ...
- Lua参考手册
英文原版: http://www.lua.org/manual/5.1/ 中文版下面2个地址都有:一样的 manual.luaer.cn lua在线手册 lua参考手册Lua参考手册的中文翻译(云风翻 ...
- lua: Learning Official Doc notes
dynamically typed vars: basic types: nil, boolean, number, string, function, userdata, thread & ...
- Lua学习系列(五)
calling C functions from Lua 5.2 这篇文章也不错: http://blog.csdn.net/x356982611/article/details/26688287 h ...
- Lua的各种资源1
Libraries And Bindings LuaDirectory > LuaAddons > LibrariesAndBindings This is a list of l ...
- luaprofiler探索
什么是luaprofiler? http://luaprofiler.luaforge.net/manual.html LuaProfiler is a time profiler designed ...
- Popular Deep Learning Tools – a review
Popular Deep Learning Tools – a review Deep Learning is the hottest trend now in AI and Machine Lear ...
随机推荐
- cmd 查看端口
windows + R 调出cmd窗口 输入 netstat -nao 查看端口使用情况 输入netstat -nao | findstr 80查找指定端口 杀死指定端口的进程taskkill /pi ...
- Python 编程第一步
Python 编程第一步 在前面的教程中我们已经学习了一些 Python3 的基本语法知识,下面我们尝试来写一个斐波纳契数列. # Fibonacci series: 斐波纳契数列 # 两个元素的总 ...
- SpringBoot 配置阿里巴巴Druid连接池
在Spring Boot下默认提供了若干种可用的连接池(dbcp,dbcp2, tomcat, hikari),当然并不支持Druid,Druid来自于阿里系的一个开源连接池,它提供了非常优秀的监控功 ...
- html5画心
- Android测试(一)——Apk文件结构以及Android组件介绍
APK文件结构: assests目录:一般存放的是不会被编译处理的文件,一般是资源性质的文件或者配置文件: libs目录:程序依赖的native库,包含针对特定处理器软件层的编译代码: res目录:存 ...
- js验证银行卡号 luhn校验规则
<script type="text/javascript"> //luhn校验规则:16位银行卡号(19位通用): // 1.将未带校验位的 15(或18)位卡号从右 ...
- STL 小白学习(6) queue
//queue 一端插入 另一端删除 //不能遍历(不提供迭代器) 不支持随机访问 #include <queue> #include <iostream> using nam ...
- java使用valueOf的方法反转字符串输出
public class FanZhuan { public static void main(String[] args) { String s = "987654321088123abo ...
- python之路--迭代器和生成器
迭代: 迭代器协议: 1.迭代器协议是指:对象必须提供一个next方法,执行该方法要么返回迭代中的下一项,要么就引起一个StopIteration异常,以终止迭代 (只能往后走不能往前退) 2.可迭代 ...
- python中字符串格式化
username='小黑'age=18high=1.88s2='欢迎%s,年龄是%d,身高是%.2f'%(username,age,high)#%s是通用的,%d就必须传整数,%f就必须是小数,想保留 ...