acl  3.5.3.0
fiber_tbox.hpp
浏览该文件的文档.
1 #pragma once
2 #include "fiber_cpp_define.hpp"
3 #include <list>
4 #include <stdlib.h>
5 #include "fiber_event.hpp"
6 #include "fiber_cond.hpp"
7 
8 namespace acl {
9 
10 /**
11  * 用于协程之间,线程之间以及协程与线程之间的消息通信,通过协程条件变量
12  * 及协程事件锁实现
13  *
14  * 示例:
15  *
16  * class myobj
17  * {
18  * public:
19  * myobj(void) {}
20  * ~myobj(void) {}
21  *
22  * void test(void) { printf("hello world\r\n"); }
23  * };
24  *
25  * acl::fiber_tbox<myobj> fiber_tbox;
26  *
27  * void thread_producer(void)
28  * {
29  * myobj* o = new myobj;
30  * fiber_tbox.push(o);
31  * }
32  *
33  * void thread_consumer(void)
34  * {
35  * myobj* o = fiber_tbox.pop();
36  * o->test();
37  * delete o;
38  * }
39  */
40 
41 template<typename T>
43 {
44 public:
45  /**
46  * 构造方法
47  * @param free_obj {bool} 当 fiber_tbox 销毁时,是否自动检查并释放
48  * 未被消费的动态对象
49  */
50  fiber_tbox(bool free_obj = true) : size_(0), free_obj_(free_obj) {}
51 
53  {
54  clear(free_obj_);
55  }
56 
57  /**
58  * 清理消息队列中未被消费的消息对象
59  * @param free_obj {bool} 释放调用 delete 方法删除消息对象
60  */
61  void clear(bool free_obj = false)
62  {
63  if (free_obj) {
64  for (typename std::list<T*>::iterator it =
65  tbox_.begin(); it != tbox_.end(); ++it) {
66 
67  delete *it;
68  }
69  }
70  tbox_.clear();
71  }
72 
73  /**
74  * 发送消息对象
75  * @param t {T*} 非空消息对象
76  * @param notify_first {bool} 如果本参数为 true,则内部添加完消息后
77  * 采用先通知后解锁方式,否则采用先解锁后通知方式,当 fiber_tbox 对象
78  * 的生存周期比较长时,该参数设为 false 的效率更高,如果 fiber_tbox
79  * 对象的生存周期较短(如:等待者调用 pop 后直接销毁 fiber_tbox 对象),
80  * 则本参数应该设为 true,以避免 push 者还没有完全返回前因 fiber_tbox
81  * 对象被提前销毁而造成内存非法访问
82  * @return {bool}
83  */
84  bool push(T* t, bool notify_first = true)
85  {
86  // 先加锁
87  if (event_.wait() == false) {
88  abort();
89  }
90 
91  // 向队列中添加消息对象
92  tbox_.push_back(t);
93  size_++;
94 
95  if (notify_first) {
96  if (cond_.notify() == false) {
97  abort();
98  }
99  if (event_.notify() == false) {
100  abort();
101  }
102  return true;
103  } else {
104  if (event_.notify() == false) {
105  abort();
106  }
107  if (cond_.notify() == false) {
108  abort();
109  }
110  return true;
111  }
112  }
113 
114  /**
115  * 接收消息对象
116  * @param wait_ms {int} >= 0 时设置等待超时时间(毫秒级别),
117  * 否则永远等待直到读到消息对象或出错
118  * @param found {bool*} 非空时用来存放是否获得了一个消息对象,主要用在
119  * 当允许传递空对象时的检查
120  * @return {T*} 非 NULL 表示获得一个消息对象,返回 NULL 时得需要做进一
121  * 步检查,生产者如果 push 了一个空对象(NULL),则消费者也会获得 NULL,
122  * 但此时仍然认为获得了一个消息对象,只不过为空对象;如果 wait_ms 参数
123  * 为 -1 时返回 NULL 依然认为获得了一个空消息对象,如果 wait_ms 大于
124  * 等于 0 时返回 NULL,则应该检查 found 参数的值为 true 还是 false 来
125  * 判断是否获得了一个空消息对象
126  */
127  T* pop(int wait_ms = -1, bool* found = NULL)
128  {
129  bool found_flag;
130  if (event_.wait() == false) {
131  abort();
132  }
133  while (true) {
134  T* t = peek(found_flag);
135  if (found_flag) {
136  if (event_.notify() == false) {
137  abort();
138  }
139  if (found) {
140  *found = found_flag;
141  }
142  return t;
143  }
144 
145  // 注意调用顺序,必须先调用 wait 再判断 wait_ms
146  if (!cond_.wait(event_, wait_ms) && wait_ms >= 0) {
147  if (event_.notify() == false) {
148  abort();
149  }
150  if (found) {
151  *found = false;
152  }
153  return NULL;
154  }
155  }
156  }
157 
158  /**
159  * 返回当前存在于消息队列中的消息数量
160  * @return {size_t}
161  */
162  size_t size(void) const
163  {
164  return size_;
165  }
166 
167 public:
168  void lock(void)
169  {
170  if (event_.wait() == false) {
171  abort();
172  }
173  }
174 
175  void unlock(void)
176  {
177  if (event_.notify() == false) {
178  abort();
179  }
180  }
181 
182 private:
183  fiber_tbox(const fiber_tbox&) {}
184  const fiber_tbox& operator=(const fiber_tbox&);
185 
186 private:
187  std::list<T*> tbox_;
188  size_t size_;
189  bool free_obj_;
190  fiber_event event_;
191  fiber_cond cond_;
192 
193  T* peek(bool& found_flag)
194  {
195  typename std::list<T*>::iterator it = tbox_.begin();
196  if (it == tbox_.end()) {
197  found_flag = false;
198  return NULL;
199  }
200  found_flag = true;
201  size_--;
202  T* t = *it;
203  tbox_.erase(it);
204  return t;
205  }
206 };
207 
208 } // namespace acl
209 
bool wait(fiber_event &event, int timeout=-1)
bool notify(void)
bool notify(void)
T * pop(int wait_ms=-1, bool *found=NULL)
Definition: fiber_tbox.hpp:127
bool wait(void)
bool push(T *t, bool notify_first=true)
Definition: fiber_tbox.hpp:84
void unlock(void)
Definition: fiber_tbox.hpp:175
fiber_tbox(bool free_obj=true)
Definition: fiber_tbox.hpp:50
void clear(bool free_obj=false)
Definition: fiber_tbox.hpp:61
void lock(void)
Definition: fiber_tbox.hpp:168
size_t size(void) const
Definition: fiber_tbox.hpp:162