libuv学习笔记4------tcp服务器的实现

在开始之前先回顾一下用Linux提供的基本的API函数来实现tcp服务器的流程:

1.调用socket得到一个文件描述符;

2.调用bind绑定IP和端口(在绑定之前需要填充一个struct sockaddr_in结构体);

3.调用accept接受连接请求;

4.调用read/write来收发数据。

其中,accept、read在默认情况下还是阻塞的,我们还可能需要调用用select, poll, epoll来对多个客户端进行处理。想一想,这一套下来其实蛮复杂的。

libuv对tcp提供的API与标准的linux提供的API相比,更为简单。仅仅需要调用几个API函数,完成几个回调函数的编写,便可编写出一个性能强大、运行可靠的TCP服务器。

libuv对TCP消息的处理,同样是基于stream的。

下面进入正题,开始介绍如何利用libuv编写一个tcp服务器。先介绍一下基本流程:

1.uv_tcp_init建立tcp句柄

2.uv_tcp_bind绑定

3.uv_listen建立监听,当有新的连接到来时,激活调用回调函数

4.uv_accept接收链接

5.使用stream操作来和客户端通信

1.API函数介绍

由于tcp服务器与客户端之间的数据交互依赖于stream,如果不了解的话可以看这里,在本文中不再进行介绍。

1.1.初始化TCP服务器对象

int uv_tcp_init(uv_loop_t, uv_tcp_t handle);

uv_tcp_t server;uv_tcp_init(loop,&server);

1.2.填充struct sockaddr_in结构体

int uv_ip4_addr(const char* ip, int port, struct sockaddr_in* addr);

参数1:本机IP

参数2:端口号

参数3:待填充的结构体的地址

struct sockaddr_in addr;uv_ip4_addr("0.0.0.0",7000,&addr);

1.3.绑定IP和端口

int uv_tcp_bind(uv_tcp_t* handle,const struct sockaddr* addr,unsigned int flags);

参数1:tcp服务器对象

参数2:指向sockaddr_in结构体

参数3:一般写0。在使用IPv6时,此参数可写为UV_TCP_IPV6ONLY

uv_tcp_bind(&server,(const struct sockaddr*)&addr,0);

1.4.建立监听

int uv_listen(uv_stream_t* stream, int backlog, uv_connection_cb cb);

参数1:要监听的对象

参数2:最大长度的待处理的请求连接队列。

参数3:回调函数,在回调函数内进行uv_accept、uv_read_start操作。

int r = uv_listen((uv_stream_t*)&server,DEFAULT_BACKLOG,on_new_connection);
 
if(r)
{
    fprintf(stderr, "Listen error %s\n", uv_strerror(r));
    return 1;
}

2.uv_listen的回调函数介绍

void (uv_connection_cb)(uv_stream_t server, int status);

server:服务器对象

status:状态,status小于0,则代表连接失败

void on_new_connection(uv_stream_t* server,int status)
{
    if(status < 0)
    {
        fprintf(stderr,"New connection error %s\n",uv_strerror(status));
 
 
        return;
    }
 
    uv_tcp_t *client = (uv_tcp_t *)malloc(sizeof(uv_tcp_t));
 
    uv_tcp_init(loop,client);
 
    //判断accept是否成功
    if(uv_accept(server,(uv_stream_t*)client) == 0)
    {
        uv_read_start((uv_stream_t*)client,alloc_buffer,echo_read);
    }
    else 
    {
        uv_close((uv_handle_t*) client, NULL);
    }
}

3.完整代码

#include <stdio.h>
#include <uv.h>
#include <stdlib.h>
 
uv_loop_t *loop;
#define DEFAULT_PORT 7000
 
//连接队列最大长度
#define DEFAULT_BACKLOG 128
 
typedef struct{
    uv_write_t req;
    uv_buf_t buf;
}write_req_t;
 
//负责为新来的消息申请空间
void alloc_buffer(uv_handle_t* handle,size_t suggested_size,uv_buf_t* buf)
{
    buf->len = suggested_size;
    buf->base = (char *)malloc(suggested_size);
}
 
void on_close(uv_handle_t* handle)
{
    if(handle != NULL)
        free(handle);
}
 
void free_write_req(uv_write_t *req)
{
    write_req_t *wr = (write_req_t*)req;
 
    free(wr->buf.base);
    free(wr);
}
 
void echo_write(uv_write_t* req, int status)
{
    if(status)
    {
        fprintf(stderr, "Write error %s\n", uv_strerror(status));
    }
 
    free_write_req(req);
}
 
//负责处理新来的消
void echo_read(uv_stream_t* client,ssize_t nread,const uv_buf_t* buf)
{
    if(nread > 0)
    {
        buf->base[nread] = 0;
        fprintf(stdout,"recv:%s",buf->base);
 
        write_req_t *req = (write_req_t *)malloc(sizeof(write_req_t));
        req->buf = uv_buf_init(buf->base,nread);
 
        uv_write((uv_write_t *)req,client,&req->buf,1,echo_write);
 
        return;
    }
    else if(nread < 0)
    {
        if(nread != UV_EOF)
        {
            fprintf(stderr,"Read error %s\n",uv_err_name(nread));
        }
        else
        {
            fprintf(stderr,"client disconnect\n");
        }
        uv_close((uv_handle_t*)client,on_close);
    }
 
    if(buf->base != NULL)
    {
        free(buf->base);
    }
}
 
void on_new_connection(uv_stream_t* server,int status)
{
    if(status < 0)
    {
        fprintf(stderr,"New connection error %s\n",uv_strerror(status));
 
 
        return;
    }
 
    uv_tcp_t *client = (uv_tcp_t *)malloc(sizeof(uv_tcp_t));
 
    uv_tcp_init(loop,client);
 
    //判断accept是否成功
    if(uv_accept(server,(uv_stream_t*)client) == 0)
    {
        uv_read_start((uv_stream_t*)client,alloc_buffer,echo_read);
    }
    else 
    {
        uv_close((uv_handle_t*) client, NULL);
    }
}
 
int main(int argc,char **argv)
{
    loop = uv_default_loop();
 
    uv_tcp_t server;
    uv_tcp_init(loop,&server);
 
    struct sockaddr_in addr;
 
    uv_ip4_addr("0.0.0.0",DEFAULT_PORT,&addr);
 
    uv_tcp_bind(&server,(const struct sockaddr*)&addr,0);
 
    int r = uv_listen((uv_stream_t*)&server,DEFAULT_BACKLOG,on_new_connection);
 
    if(r)
    {
        fprintf(stderr, "Listen error %s\n", uv_strerror(r));
        return 1;
    }
 
    return uv_run(loop,UV_RUN_DEFAULT);
}

此代码实现了一个echo服务器,支持多客户端连接。需要说明的是,当客户端发起四次挥手断开连接时,服务器端会触发echo_read回调函数。在回调函数内部需要根据nread判断当前状态,如果nread小于0,并且nread等于UV_EOF,则代表客户端断开连接。此时对客户端对象进行close操作即可。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353

推荐阅读更多精彩内容