2019-07-28 10:31:56 +08:00
|
|
|
|
#pragma once
|
2018-06-28 21:04:54 +08:00
|
|
|
|
#include "../acl_cpp_define.hpp"
|
|
|
|
|
#include <list>
|
2018-12-25 11:32:42 +08:00
|
|
|
|
#include <stdlib.h>
|
2018-06-28 21:04:54 +08:00
|
|
|
|
#include "thread_mutex.hpp"
|
|
|
|
|
#include "thread_cond.hpp"
|
|
|
|
|
#include "noncopyable.hpp"
|
|
|
|
|
|
|
|
|
|
namespace acl
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
/**
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>֮<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣͨ<EFBFBD>ţ<EFBFBD>ͨ<EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD><EFBFBD><EFBFBD>ʵ<EFBFBD><EFBFBD>
|
2018-06-28 21:04:54 +08:00
|
|
|
|
*
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* ʾ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
2018-06-28 21:04:54 +08:00
|
|
|
|
*
|
|
|
|
|
* class myobj
|
|
|
|
|
* {
|
|
|
|
|
* public:
|
|
|
|
|
* myobj(void) {}
|
|
|
|
|
* ~myobj(void) {}
|
|
|
|
|
*
|
|
|
|
|
* void test(void) { printf("hello world\r\n"); }
|
|
|
|
|
* };
|
|
|
|
|
*
|
|
|
|
|
* acl::tbox<myobj> tbox;
|
|
|
|
|
*
|
|
|
|
|
* void thread_producer(void)
|
|
|
|
|
* {
|
|
|
|
|
* myobj* o = new myobj;
|
|
|
|
|
* tbox.push(o);
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* void thread_consumer(void)
|
|
|
|
|
* {
|
|
|
|
|
* myobj* o = tbox.pop();
|
|
|
|
|
* o->test();
|
|
|
|
|
* delete o;
|
|
|
|
|
* }
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
|
class tbox : public noncopyable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2018-07-27 16:26:27 +08:00
|
|
|
|
/**
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD>췽<EFBFBD><EFBFBD>
|
|
|
|
|
* @param free_obj {bool} <EFBFBD><EFBFBD> tbox <EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʱ<EFBFBD><EFBFBD><EFBFBD>Ƿ<EFBFBD><EFBFBD>Զ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>鲢<EFBFBD>ͷ<EFBFBD>
|
|
|
|
|
* δ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ѵĶ<EFBFBD>̬<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
2018-07-27 16:26:27 +08:00
|
|
|
|
*/
|
2019-01-02 11:25:17 +08:00
|
|
|
|
tbox(bool free_obj = true)
|
2018-07-27 16:26:27 +08:00
|
|
|
|
: size_(0), free_obj_(free_obj), cond_(&lock_) {}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
|
|
|
|
|
~tbox(void)
|
|
|
|
|
{
|
2018-07-27 16:26:27 +08:00
|
|
|
|
clear(free_obj_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>δ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ѵ<EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* @param free_obj {bool} <EFBFBD>ͷŵ<EFBFBD><EFBFBD><EFBFBD> delete <EFBFBD><EFBFBD><EFBFBD><EFBFBD>ɾ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
2018-07-27 16:26:27 +08:00
|
|
|
|
*/
|
|
|
|
|
void clear(bool free_obj = false)
|
|
|
|
|
{
|
|
|
|
|
if (free_obj) {
|
|
|
|
|
for (typename std::list<T*>::iterator it =
|
|
|
|
|
tbox_.begin(); it != tbox_.end(); ++it) {
|
|
|
|
|
|
|
|
|
|
delete *it;
|
|
|
|
|
}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
}
|
2018-07-27 16:26:27 +08:00
|
|
|
|
tbox_.clear();
|
2018-06-28 21:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* @param t {T*} <EFBFBD>ǿ<EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* @param notify_first {bool} <EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϊ true<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>֪ͨ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƚ<EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* <EFBFBD><EFBFBD>֪ͨ<EFBFBD><EFBFBD>ע<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ߵ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
2018-12-25 11:32:42 +08:00
|
|
|
|
* @return {bool}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
*/
|
2019-03-13 10:28:51 +08:00
|
|
|
|
bool push(T* t, bool notify_first = true)
|
2018-06-28 21:04:54 +08:00
|
|
|
|
{
|
2018-12-25 11:32:42 +08:00
|
|
|
|
if (lock_.lock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
tbox_.push_back(t);
|
|
|
|
|
size_++;
|
2019-03-13 10:28:51 +08:00
|
|
|
|
|
|
|
|
|
if (notify_first) {
|
|
|
|
|
if (cond_.notify() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
if (lock_.unlock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (lock_.unlock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
if (cond_.notify() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-12-25 11:32:42 +08:00
|
|
|
|
}
|
2019-03-13 10:28:51 +08:00
|
|
|
|
|
2018-12-25 11:32:42 +08:00
|
|
|
|
return true;
|
2018-06-28 21:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* @param wait_ms {int} >= 0 ʱ<EFBFBD><EFBFBD><EFBFBD>õȴ<EFBFBD><EFBFBD><EFBFBD>ʱʱ<EFBFBD><EFBFBD>(<EFBFBD><EFBFBD><EFBFBD>뼶<EFBFBD><EFBFBD>)<EFBFBD><EFBFBD>
|
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Զ<EFBFBD>ȴ<EFBFBD>ֱ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* @param found {bool*} <EFBFBD>ǿ<EFBFBD>ʱ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƿ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ҫ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݿն<EFBFBD><EFBFBD><EFBFBD>ʱ<EFBFBD>ļ<EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* @return {T*} <EFBFBD><EFBFBD> NULL <EFBFBD><EFBFBD>ʾ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> NULL ʱ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ҫ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ
|
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>飬<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> push <EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD>ն<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>NULL<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ҳ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> NULL<EFBFBD><EFBFBD>
|
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʱ<EFBFBD><EFBFBD>Ȼ<EFBFBD><EFBFBD>Ϊ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֻ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϊ<EFBFBD>ն<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> wait_ms <EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* Ϊ -1 ʱ<EFBFBD><EFBFBD><EFBFBD><EFBFBD> NULL <EFBFBD><EFBFBD>Ȼ<EFBFBD><EFBFBD>Ϊ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> wait_ms <EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD><EFBFBD> 0 ʱ<EFBFBD><EFBFBD><EFBFBD><EFBFBD> NULL<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ӧ<EFBFBD>ü<EFBFBD><EFBFBD><EFBFBD> found <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֵΪ true <EFBFBD><EFBFBD><EFBFBD><EFBFBD> false <EFBFBD><EFBFBD>
|
|
|
|
|
* <EFBFBD>ж<EFBFBD><EFBFBD>Ƿ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
2018-06-28 21:04:54 +08:00
|
|
|
|
*/
|
2018-07-27 13:26:07 +08:00
|
|
|
|
T* pop(int wait_ms = -1, bool* found = NULL)
|
2018-06-28 21:04:54 +08:00
|
|
|
|
{
|
|
|
|
|
long long n = ((long long) wait_ms) * 1000;
|
2018-07-27 13:26:07 +08:00
|
|
|
|
bool found_flag;
|
2018-12-25 11:32:42 +08:00
|
|
|
|
if (lock_.lock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-07-29 18:57:47 +08:00
|
|
|
|
while (true) {
|
2018-07-27 13:26:07 +08:00
|
|
|
|
T* t = peek(found_flag);
|
|
|
|
|
if (found_flag) {
|
2018-12-25 11:32:42 +08:00
|
|
|
|
if (lock_.unlock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-07-27 13:26:07 +08:00
|
|
|
|
if (found) {
|
|
|
|
|
*found = found_flag;
|
|
|
|
|
}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
return t;
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// ע<><D7A2><EFBFBD><EFBFBD><EFBFBD><EFBFBD>˳<EFBFBD><EFBFBD><F2A3ACB1><EFBFBD><EFBFBD>ȵ<EFBFBD><C8B5><EFBFBD> wait <20><><EFBFBD>ж<EFBFBD> wait_ms
|
2018-07-27 16:26:27 +08:00
|
|
|
|
if (!cond_.wait(n, true) && wait_ms >= 0) {
|
2018-12-25 11:32:42 +08:00
|
|
|
|
if (lock_.unlock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-07-27 13:26:07 +08:00
|
|
|
|
if (found) {
|
|
|
|
|
*found = false;
|
|
|
|
|
}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2019-07-28 10:31:56 +08:00
|
|
|
|
* <EFBFBD><EFBFBD><EFBFBD>ص<EFBFBD>ǰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>е<EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
2018-06-28 21:04:54 +08:00
|
|
|
|
* @return {size_t}
|
|
|
|
|
*/
|
|
|
|
|
size_t size(void) const
|
|
|
|
|
{
|
2018-06-28 21:40:24 +08:00
|
|
|
|
return size_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
void lock(void)
|
|
|
|
|
{
|
2018-12-25 11:32:42 +08:00
|
|
|
|
if (lock_.lock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-06-28 21:40:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void unlock(void)
|
|
|
|
|
{
|
2018-12-25 11:32:42 +08:00
|
|
|
|
if (lock_.unlock() == false) {
|
|
|
|
|
abort();
|
|
|
|
|
}
|
2018-06-28 21:04:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
2018-07-27 16:26:27 +08:00
|
|
|
|
std::list<T*> tbox_;
|
|
|
|
|
size_t size_;
|
|
|
|
|
bool free_obj_;
|
2018-06-28 21:04:54 +08:00
|
|
|
|
thread_mutex lock_;
|
|
|
|
|
thread_cond cond_;
|
|
|
|
|
|
2018-07-27 13:26:07 +08:00
|
|
|
|
T* peek(bool& found_flag)
|
2018-06-28 21:04:54 +08:00
|
|
|
|
{
|
|
|
|
|
typename std::list<T*>::iterator it = tbox_.begin();
|
2018-07-27 13:26:07 +08:00
|
|
|
|
if (it == tbox_.end()) {
|
|
|
|
|
found_flag = false;
|
2018-06-28 21:04:54 +08:00
|
|
|
|
return NULL;
|
2018-07-27 13:26:07 +08:00
|
|
|
|
}
|
|
|
|
|
found_flag = true;
|
2018-06-28 21:04:54 +08:00
|
|
|
|
size_--;
|
|
|
|
|
T* t = *it;
|
|
|
|
|
tbox_.erase(it);
|
|
|
|
|
return t;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace acl
|