接下來要談的是
libevent 要如何使用,不過為了方便起見,我們直接寫一個很簡單的 Time Server 來當作例子:當你連上去以後 Server 端直接提供時間,然後結束連線。
在這些例子裡面我以
FreeBSD 6.0 當作測試的平台,另外使用
libevent 1.1a 當作 Event-based Library,Compile 時請使用
gcc -I/usr/local/include -o timeserver timeserver.c -L/usr/local/lib -levent
(如果
libevent 的 Header 與 Library 放在
/usr/include
與
/usr/lib
下的話可以省略這兩個參數)。
原始程式碼在文章的最後頭。
event_init()
表示初始化
libevent 所使用到的變數。
event_set(&ev, s, EV_READ | EV_PERSIST, connection_accept, &ev)
把
s
這個 File Description 放入
ev
(第一個參數與第二個參數),並且告知當事件 (第三個參數的
EV_READ
) 發生時要呼叫
connection_accept()
(第四個參數),呼叫時要把
ev
當作參數丟進去 (第五個參數)。
其中的
EV_PERSIST
表示當呼叫進去的時候不要把這個 event 拿掉 (繼續保留在 Event Queue 裡面),這點可以跟
connection_accept()
內在註冊
connection_time()
的程式碼做比較。
而
event_add(&ev, NULL)
就是把
ev
註冊到 event queue 裡面,第二個參數指定的是 Timeout 時間,設定成
NULL
表示忽略這項設定。
最後的
event_dispatch()
表示進入 event loop,當 Queue 裡面的任何一個 File Description 發生事件的時候就會進入 callback function 執行。
這隻程式非常粗糙,有很多地方沒有注意到 Blocking 的問題,這點我們就先不管了。當跑起來以後你就可以連到 port 7000,就會出現類似下面的結果:gslin@netnews [~] [3:14/W5] t 0 7000
gslin@netnews [~/work/C] [3:15/W3] t 0 7000
Trying 0.0.0.0...
Connected to 0.
Escape character is '^]'.
Fri Nov 25 03:15:10 2005
Connection closed by foreign host.
最基本的使用就是這樣了,你可以
man event
看到完整的說明。
這是
timeserver.c
:
#include <netinet><br>#include <sys><br>#include <sys><br>#include <event.h><br>#include <stdio.h><br>#include <time.h><br><br>void connection_time(int fd, short event, struct event *arg)<br>{<br> char buf[32];<br> struct tm t;<br> time_t now;<br><br> time(&now);<br> localtime_r(&now, &t);<br> asctime_r(&t, buf);<br><br> write(fd, buf, strlen(buf));<br> shutdown(fd, SHUT_RDWR);<br><br> free(arg);<br>}<br><br>void connection_accept(int fd, short event, void *arg)<br>{<br> /* for debugging */<br> fprintf(stderr, "%s(): fd = %d, event = %d./n", __func__, fd, event);<br><br> /* Accept a new connection. */<br> struct sockaddr_in s_in;<br> socklen_t len = sizeof(s_in);<br> int ns = accept(fd, (struct sockaddr *) &s_in, &len);<br> if (ns perror("accept");<br> return;<br> }<br><br> /* Install time server. */<br> struct event *ev = malloc(sizeof(struct event));<br> event_set(ev, ns, EV_WRITE, (void *) connection_time, ev);<br> event_add(ev, NULL);<br>}<br><br>int main(void)<br>{<br> /* Request socket. */<br> int s = socket(PF_INET, SOCK_STREAM, 0);<br> if (s perror("socket");<br> exit(1);<br> }<br><br> /* bind() */<br> struct sockaddr_in s_in;<br> bzero(&s_in, sizeof(s_in));<br> s_in.sin_family = AF_INET;<br> s_in.sin_port = htons(7000);<br> s_in.sin_addr.s_addr = INADDR_ANY;<br> if (bind(s, (struct sockaddr *) &s_in, sizeof(s_in)) perror("bind");<br> exit(1);<br> }<br><br> /* listen() */<br> if (listen(s, 5) perror("listen");<br> exit(1);<br> }<br><br> /* Initial libevent. */<br> event_init();<br><br> /* Create event. */<br> struct event ev;<br> event_set(&ev, s, EV_READ | EV_PERSIST, connection_accept, &ev);<br><br> /* Add event. */<br> event_add(&ev, NULL);<br><br> event_dispatch();<br><br> return 0;<br>}</time.h></stdio.h></event.h></sys></sys></netinet>
Network Programming Using Libevent - (III)
這次要談的跟 Network Programming 沒有直接的關係。
在寫 Nonblocking Network Program 通常要處理 Buffering 的問題,但並不好寫,主要是因為
read()
或
recv()
不保證可以一次讀到一行的份量進來。
在
libevent 裡面提供相當不錯的 Buffer Library 可以用,完整的說明在
man event
的時候可以看到,最常用的應該就是以
evbuffer_add()
、
evbuffer_readline()
這兩個 Function,其他的知道存在就可以了,需要的時候再去看詳細的用法。
下面直接提供
libevent-buff.c
當作範例,編譯後看執行結果,再回頭來看 source code 應該就有感覺了:
#include <sys><br>#include <event.h><br>#include <stdio.h><br><br>void printbuf(struct evbuffer *evbuf)<br>{<br> for (;;) {<br> char *buf = evbuffer_readline(evbuf);<br> printf("* buf = %p, the string = /"/e[1;33m%s/e[m/"/n", buf, buf);<br> if (buf == NULL)<br> break;<br> free(buf);<br> }<br>}<br><br>int main(void)<br>{<br> struct evbuffer *evbuf;<br><br> evbuf = evbuffer_new();<br> if (evbuf == NULL) {<br> fprintf(stderr, "%s(): evbuffer_new() failed./n", __func__);<br> exit(1);<br> }<br><br> /* Add "gslin" into buffer. */<br> u_char *buf1 = "gslin";<br> printf("* Add /"/e[1;33m%s/e[m/"./n", buf1);<br> evbuffer_add(evbuf, buf1, strlen(buf1));<br> printbuf(evbuf);<br><br> u_char *buf2 = " is reading./nAnd he is at home./nLast.";<br> printf("* Add /"/e[1;33m%s/e[m/"./n", buf2);<br> evbuffer_add(evbuf, buf2, strlen(buf2));<br> printbuf(evbuf);<br><br> evbuffer_free(evbuf);<br>}</stdio.h></event.h></sys>
内容来源:http://www.cublog.cn/u/5251/showart_270584.html
顺便建议参考:http://unx.ca/log/category/libevent/