150行C代码的comet服务器

2024-06-15 01:48
文章标签 代码 服务器 150 comet

本文主要是介绍150行C代码的comet服务器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Comet 技术就是常见的 Web 服务器"推"技术, 用于向网页实时地推送数据. 最常见的 Comet 技术应用在网页聊天, 当然还可以应用于很多的方面, 如微博更新, 热点新闻推送, 股票即时行情等等, 甚至是网页游戏!

Comet 技术如此重要, 但市面上并没有真正流行通用的 Comet 服务器和解决方案, 比较知名的互联网公司大多是自己开发, 或者基于开源服务器进行二次开发, 例如基于 Jetty(一个开源 Java Web 容器), 而 Facebook 的聊天系统的 Comet 服务器是基于 Mochiweb(一个开源的 Erlang Web 服务器).

当然还有比较知名的以 nginx 模块形式出现的 nginx-push-stream, 但根据实际使用经验, 这个模块无法稳定支撑 10 万个并发连接, 更别谈百万同时在线了. 这也是这个模块为什么没有被普遍大规模应用的原因.

既然大家都开发自己的 Comet 服务器, 那必然有其中的道理, 说是核心技术倒说不上, 不过是为了便于扩展, 能很好地和现有系统整合, 易于运维和管理而已. 那么, 要开发一个 Comet 服务器到底有多难呢? 其实, 一个最简单的 Comet 服务器只需要 150 行 C 语言代码!

先说一下 Comet 技术, 从浏览器支持考虑, long-polling 技术显然是最佳的选择, 又从跨域方面考虑, 那必然是 script tag long-polling 技术获胜. 这也是 Facebook 的选择. 所以, 最简单的 Comet 服务器只支持 Script tag long-polling 即可.

Long-polling 技术要求浏览器的每一个网页和服务器保持一个 HTTP 请求连接(TCP 连接), 服务器收到这样的连接后, 会立即返回 HTTP 首部, 接着通过 chunk 传输编码, 源源不断地将一个个消息发送给浏览器.

一个完整的 chunk 编码的 HTTP 响应如下:

HTTP/1.1 200 OK
Date: Fri, 31 Dec 1999 23:59:59 GMT
Content-Type: text/plain
Transfer-Encoding: chunked1a; ignore-stuff-here
abcdefghijklmnopqrstuvwxyz
10
1234567890abcdef
0
[blank line here]

只要服务器不返回只有"0"的那一行以及紧接着的空白行, 那么就可以保持向网页推数据.

最简单的 Comet 服务器使用了 libevent 框架, 你可以在这里得到它的代码: https://github.com/ideawu/icomet. 欢迎对 Comet 了解的前端工程师贡献 JavaScript 相关的代码!

使用方式:

订阅: curl -v "http://127.0.0.1:8000/sub?id=12"
推送: curl -v "http://127.0.0.1:8000/pub?id=12&content=hi"

这个 Comet 服务器的最大并发数并没有进行测试, 但 last.fm 的 CTO 对一个同样是基于 libevent 的类似程序进行测试, 100 万连接只需要 2GB 内存.

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <signal.h>
#include <fcntl.h>
#include <unistd.h>
#include <event.h>
#include <evhttp.h>
#include <event2/event.h>
#include <event2/http.h>
#include <event2/buffer.h>
#include <event2/util.h>
#include <event2/keyvalq_struct.h>
#include <netinet/in.h>
#include <arpa/inet.h>#define MAX_CHANNELS    1000struct Channel{int id;struct evhttp_request *req;
};struct Channel channels[MAX_CHANNELS];void init(){int i;for(i=0; i<MAX_CHANNELS; i++){channels[i].id = i;channels[i].req = NULL;}
}// called when user disconnects
void cleanup(struct evhttp_connection *evcon, void *arg){struct Channel *sub = (struct Channel *)arg;printf("disconnected uid %d\n", sub->id);sub->req = NULL;
}void sub_handler(struct evhttp_request *req, void *arg)
{struct evkeyvalq params;const char *uri = evhttp_request_get_uri(req);evhttp_parse_query(uri, &params);struct evbuffer *buf;int uid = -1;struct evkeyval *kv;for(kv = params.tqh_first; kv; kv = kv->next.tqe_next){if(strcmp(kv->key, "id") == 0){uid = atoi(kv->value);}}if(uid < 0 || uid >= MAX_CHANNELS){buf = evbuffer_new();evhttp_send_reply_start(req, HTTP_NOTFOUND, "Not Found");evbuffer_free(buf);return;}printf("sub: %d\n", uid);struct Channel *sub = &channels[uid];sub->req = req;buf = evbuffer_new();evhttp_send_reply_start(req, HTTP_OK, "OK");evhttp_add_header(req->output_headers, "Content-Type", "text/html; charset=utf-8");evbuffer_add_printf(buf, "{type: \"welcome\", id: \"%d\", content: \"hello world!\"}\n", uid);evhttp_send_reply_chunk(req, buf);evbuffer_free(buf);evhttp_connection_set_closecb(req->evcon, cleanup, &channels[uid]);
}void pub_handler(struct evhttp_request *req, void *arg){struct evkeyvalq params;const char *uri = evhttp_request_get_uri(req);evhttp_parse_query(uri, &params);struct evbuffer *buf;int uid = -1;const char *content = "";struct evkeyval *kv;for(kv = params.tqh_first; kv; kv = kv->next.tqe_next){if(strcmp(kv->key, "id") == 0){uid = atoi(kv->value);}else if(strcmp(kv->key, "content") == 0){content = kv->value;}}struct Channel *sub = NULL;if(uid < 0 || uid >= MAX_CHANNELS){sub = NULL;}else{sub = &channels[uid];}if(sub && sub->req){printf("pub: %d content: %s\n", uid, content);// push to browserbuf = evbuffer_new();evbuffer_add_printf(buf, "{type: \"data\", id: \"%d\", content: \"%s\"}\n", uid, content);evhttp_send_reply_chunk(sub->req, buf);evbuffer_free(buf);// response to publisherbuf = evbuffer_new();evhttp_add_header(req->output_headers, "Content-Type", "text/html; charset=utf-8");evbuffer_add_printf(buf, "ok\n");evhttp_send_reply(req, 200, "OK", buf);evbuffer_free(buf);}else{buf = evbuffer_new();evbuffer_add_printf(buf, "id: %d not connected\n", uid);evhttp_send_reply(req, 404, "Not Found", buf);evbuffer_free(buf);}
}int main(int argc, char **argv){signal(SIGPIPE, SIG_IGN);struct event_base *base;struct evhttp *http;struct evhttp_bound_socket *handle;unsigned short port = 8000;init();base = event_base_new();http = evhttp_new(base);evhttp_set_cb(http, "/sub", sub_handler, NULL);evhttp_set_cb(http, "/pub", pub_handler, NULL);//evhttp_set_gencb(http, request_handler, NULL);handle = evhttp_bind_socket_with_handle(http, "0.0.0.0", port);printf("server listen at 127.0.0.1:%d\n", port);event_base_dispatch(base);return 0;
}

Related posts:

  1. Libevent 2 HTTP 客户端示例
  2. C#封装log4net
  3. 使用 jemalloc 编译过程出错的问题
  4. Ideawu.P2P API 简介
  5. 基于列的数据库

这篇关于150行C代码的comet服务器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1062089

相关文章

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

精选20个好玩又实用的的Python实战项目(有图文代码)

《精选20个好玩又实用的的Python实战项目(有图文代码)》文章介绍了20个实用Python项目,涵盖游戏开发、工具应用、图像处理、机器学习等,使用Tkinter、PIL、OpenCV、Kivy等库... 目录① 猜字游戏② 闹钟③ 骰子模拟器④ 二维码⑤ 语言检测⑥ 加密和解密⑦ URL缩短⑧ 音乐播放

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Python极速搭建局域网文件共享服务器完整指南

《Python极速搭建局域网文件共享服务器完整指南》在办公室或家庭局域网中快速共享文件时,许多人会选择第三方工具或云存储服务,但这些方案往往存在隐私泄露风险或需要复杂配置,下面我们就来看看如何使用Py... 目录一、android基础版:HTTP文件共享的魔法命令1. 一行代码启动HTTP服务器2. 关键参

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

SpringBoot改造MCP服务器的详细说明(StreamableHTTP 类型)

《SpringBoot改造MCP服务器的详细说明(StreamableHTTP类型)》本文介绍了SpringBoot如何实现MCPStreamableHTTP服务器,并且使用CherryStudio... 目录SpringBoot改造MCP服务器(StreamableHTTP)1 项目说明2 使用说明2.1

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

mysql中的服务器架构详解

《mysql中的服务器架构详解》:本文主要介绍mysql中的服务器架构,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、mysql服务器架构解释3、总结1、背景简单理解一下mysqphpl的服务器架构。2、mysjsql服务器架构解释mysql的架