当前位置: 首页 > news >正文

做网站的资料修改网页设计代码的基本格式

做网站的资料修改,网页设计代码的基本格式,随县网站建设,想自己做网站文章目录 延时队列简介应用场景案例#xff1a;考虑#xff1a;实现#xff1a;整体思路#xff1a;具体实现生产者消费者 运行结果 redis分布式延时队列优势redis分布式延时队列劣势 延时队列简介 延时队列是一种特殊的消息队列#xff0c;它允许将消息在一定的延迟时间… 文章目录 延时队列简介应用场景案例考虑实现整体思路具体实现生产者消费者 运行结果 redis分布式延时队列优势redis分布式延时队列劣势 延时队列简介 延时队列是一种特殊的消息队列它允许将消息在一定的延迟时间后再进行消费。延时队列的主要特点是可以延迟消息的处理时间以满足定时任务或者定时事件的需求。 总之延时队列通过延迟消息的消费时间提供了一种方便、可靠的方式来处理定时任务和定时事件。它在分布式系统中具有重要的作用能够提高系统的可靠性和性能。 延时队列的实现方式可以有多种本文介绍一种redis实现的分布式延时队列。 应用场景 定时任务可以将需要在特定时间执行的任务封装为延时消息通过延时队列来触发任务的执行。 订单超时处理可以将订单消息发送到延时队列中并设置订单的超时时间超过时间后消费者从队列中获取到超时的订单消息进行相应的处理。 消息重试机制当某个消息处理失败时可以将该消息发送到延时队列中并设置一定的重试时间超过时间后再次尝试处理。 案例 12306火车票购买抢了订单后45分钟没有支付自动取消订单 考虑 数据持久化redis是支持的可以使用rdb也可以使用aof 有序存储因为只要最小的没过期后面的肯定就没过期这样的话检查最小的节点就行了考虑使用redis中的zset结构 高可用考虑哨兵或者cluster 高伸缩因为12306用户量非常大可能导致redis中存储的任务空间非常大所以考虑扩展节点从这个角度来说使用cluster集群模式哨兵只有一个节点即主节点写数据。 实现 整体思路 生产消费者模型因为12306的用户量非常大所以考虑生产者和消费者有多个节点采用cluster模式实现高可用以及高伸缩性采用zset存储延时任务zadd key score memberscore表示时间为了让数据均匀分布在cluster集群中的多个主节点中构建多个zset每个zset对应一个消费者生产者随机向某个zset中生产数据。 具体实现 生产者 需要安装hiredis-cluster集群安装编译如下 git clone https://github.com/Nordix/hiredis-cluster.git cd hiredis-cluster mkdir build cd build cmake -DCMAKE_BUILD_TYPERelWithDebInfo - DENABLE_SSLON .. make sudo make install sudo ldconfig需要安装libevent库最后编译时执行gcc producer.c -o producer -levent -lhiredis_cluster -lhiredis -lhiredis_ssl编译生产者可执行程序 #include hiredis_cluster/adapters/libevent.h #include hiredis_cluster/hircluster.h #include event.h #include event2/listener.h #include event2/bufferevent.h #include event2/buffer.h #include stdio.h #include stdlib.h #include stdint.h #include string.h #include sys/time.hint64_t g_taskid 0;#define MAX_KEY 10static int64_t hi_msec_now() {int64_t msec;struct timeval now;int status;status gettimeofday(now, NULL);if (status 0) {return -1;}msec (int64_t)now.tv_sec * 1000LL (int64_t)(now.tv_usec / 1000LL);return msec; }static int _vscnprintf(char *buf, size_t size, const char *fmt, va_list args) {int n;n vsnprintf(buf, size, fmt, args);if (n 0) {return 0;}if (n (int)size) {return n;}return (int)(size-1); }static int _scnprintf(char *buf, size_t size, const char *fmt, ...) {va_list args;int n;va_start(args, fmt);n _vscnprintf(buf, size, fmt, args);va_end(args);return n; }void connectCallback(const redisAsyncContext *ac, int status) {if (status ! REDIS_OK) {printf(Error: %s\n, ac-errstr);return;}printf(Connected to %s:%d\n, ac-c.tcp.host, ac-c.tcp.port); }void disconnectCallback(const redisAsyncContext *ac, int status) {if (status ! REDIS_OK) {printf(Error: %s\n, ac-errstr);return;}printf(Disconnected from %s:%d\n, ac-c.tcp.host, ac-c.tcp.port); }void addTaskCallback(redisClusterAsyncContext *cc, void *r, void *privdata) {redisReply *reply (redisReply *)r;if (reply NULL) {if (cc-errstr) {printf(errstr: %s\n, cc-errstr);}return;}int64_t now hi_msec_now() / 10;printf(add task success reply: %lld now%ld\n, reply-integer, now); }int addTask(redisClusterAsyncContext *cc, char *desc) {/* 转化为厘米秒 */int64_t now hi_msec_now() / 10;g_taskid;/* key */char key[256] {0};// 为了让数据均匀分布在cluster集群中的多个主节点中 ​ // 构建多个zset每个zset对应一个消费者生产者随机向某个zset中生产数据// 生产者可以有很多个只需要保证向task_group:0-task_group:9中均匀的生产数据即可int len _scnprintf(key, 255, task_group:%ld, g_taskid % MAX_KEY);key[len] \0;/* member */char mem[1024] {0};len _scnprintf(mem, 1023, task:%ld:%s, g_taskid, desc);mem[len] \0;int status;// 为每一个任务延时5秒中去处理status redisClusterAsyncCommand(cc, addTaskCallback, ,zadd %s %ld %s, key, now500, mem);printf(redisClusterAsyncCommand:zadd %s %ld %s\n, key, now500, mem);if (status ! REDIS_OK) {printf(error: err%d errstr%s\n, cc-err, cc-errstr);}return 0; }void stdio_callback(struct bufferevent *bev, void *arg) {redisClusterAsyncContext *cc (redisClusterAsyncContext *)arg;struct evbuffer *evbuf bufferevent_get_input(bev);char *msg evbuffer_readln(evbuf, NULL, EVBUFFER_EOL_LF);if (!msg) return;if (strcmp(msg, quit) 0) {printf(safe exit!!!\n);exit(0);return;}if (strlen(msg) 1024-5-13-1) {printf([err]msg is too long, try again...\n);return;}addTask(cc, msg);printf(stdio read the data: %s\n, msg); }int main(int argc, char **argv) {printf(Connecting...\n);// 连接cluster集群可以从cluster集群中任意一个节点出发连接集群redisClusterAsyncContext *cc redisClusterAsyncConnect(127.0.0.1:7006, HIRCLUSTER_FLAG_NULL);printf(redisClusterAsyncContext...\n);if (cc cc-err) {printf(Error: %s\n, cc-errstr);return 1;}struct event_base *base event_base_new();redisClusterLibeventAttach(cc, base);redisClusterAsyncSetConnectCallback(cc, connectCallback);redisClusterAsyncSetDisconnectCallback(cc, disconnectCallback);// nodeIterator ni;// initNodeIterator(ni, cc-cc);// cluster_node *node;// while ((node nodeNext(ni)) ! NULL) {// printf(node %s:%d role:%d pad:%d\n, node-host, node-port, node-role, node-pad);// }struct bufferevent *ioev bufferevent_socket_new(base, 0, BEV_OPT_CLOSE_ON_FREE);bufferevent_setcb(ioev, stdio_callback, NULL, NULL, cc);bufferevent_enable(ioev, EV_READ | EV_PERSIST);printf(Dispatch..\n);event_base_dispatch(base);printf(Done..\n);redisClusterAsyncFree(cc);event_base_free(base);return 0; }// 需要安装 hiredis-cluster libevent // gcc producer.c -o producer -levent -lhiredis_cluster -lhiredis -lhiredis_ssl说明 这里构建了10个zset分别是task_group:0task_group:1…task_group:9作为10个zset的keyzset的数据其实就代表着消费者的数量通常消费者的功能是一摸一样的生产者就不管你有多少个了只需要将任务均匀的打散在不同的zset中就行了具体实现可以搞一个全局的id每一次添加任务时id,然后再对zset个数10取模最终可以得到0-9之间的一个数然后再与task_group拼接这样就可以将任务均匀的打散在不同的zset中。 消费者 消费者是采用skynetlua脚本实现的每个消费者会不断的去检查redis中的任务有没有过期如果过期就取出来删除这里只是demo只是打印之后删除任务 local skynet require skynetlocal function table_dump( object )if type(object) table thenlocal s { for k,v in pairs(object) doif type(k) ~ number then k string.format(%q, k) ends s .. [..k..] .. table_dump(v) .. ,endreturn s .. } elseif type(object) function thenreturn tostring(object)elseif type(object) string thenreturn string.format(%q, object)elsereturn tostring(object)end endlocal mode, key ... if mode slave thenlocal rediscluster require skynet.db.redis.clusterlocal function onmessage(data,channel,pchannel)print(onmessage,data,channel,pchannel)endskynet.start(function ()local db rediscluster.new({{host127.0.0.1,port7001},},{read_slavetrue,authnil,db0,},onmessage)assert(db, redis-cluster startup error)skynet.fork(function ()while true dolocal res db:zrange(key, 0, 0, withscores)if not next(res) thenskynet.sleep(50)elselocal expire tonumber(res[2])local now skynet.time()*100if now expire thenprint((%s is comsumed:expire_time:%d):format(res[1], expire))db:zrem(key, res[1])elseskynet.sleep(10)endendendend)end)elseskynet.start(function () -- // 启动10个程序并把slave传入modetask_group:i传入到key中即每个程序只消费一个for i0,9 doskynet.newservice(SERVICE_NAME, slave, task_group:..i)运行结果 redis分布式延时队列优势 1.Redis zset支持高性能的 score 排序。 2.Redis是在内存上进行操作的速度非常快。 3.Redis可以搭建集群当消息很多时候我们可以用集群来提高消息处理的速度提高可用性。 4.Redis具有持久化机制当出现故障的时候可以通过AOF和RDB方式来对数据进行恢复保证了数据的可靠性 redis分布式延时队列劣势 使用 Redis 实现的延时消息队列也存在数据持久化, 消息可靠性的问题 没有重试机制 - 处理消息出现异常没有重试机制, 这些需要自己去实现, 包括重试次数的实现等没有 ACK 机制 - 例如在获取消息并已经删除了消息情况下, 正在处理消息的时候客户端崩溃了, 这条正在处理的这些消息就会丢失, MQ 是需要明确的返回一个值给 MQ 才会认为这个消息是被正确的消费了。 总结如果对消息可靠性要求较高, 推荐使用 MQ 来实现
http://www.yutouwan.com/news/79954/

相关文章:

  • 秦皇岛做网站外包电子商务主要是什么
  • 本地集团网站建设做网站的公司那家好。
  • 如何用自己公司网站做邮箱国投集团网站开发
  • 怎样做号网站优化哪个网站可以免费学编程
  • 漳州建设网站wordpress pdf预览
  • 网站建设需求说明书怎么写银川360推广 网站建设
  • asp.net做网站吗虚拟主机购买网站
  • 学网站建设 去那里文登区住房和城乡建设局网站
  • dw可以做有后台的网站么用ps给旅游网站做前端网页
  • 我想做网站怎么做企业官网定制
  • 网站建设下载模板之后怎么修改wordpress分类页面不显示内容
  • 东莞网站关键字郑州铭功路网站建设
  • 网站的手机客户端怎样做python手机版
  • 途牛旅游网站建设目的网站建设使用的基本技术
  • 请问聊城做网站网站建设后应该干什么
  • 邢台做企业网站wordpress导航链接地址都是主页
  • 为什么访问外国网站速度慢qq空间做宣传网站
  • 做网站协调国内搜索引擎排名第一的是
  • 品牌网站建设预算网站加入搜索引擎怎么做
  • 灵台县住房和城乡建设局网站wordpress seo模块
  • 网站工信部不备案吗c4d培训
  • 网站设计的公司概况简介discuz网站模板
  • 如何建立一个网站英语作文机关门户网站建设意义
  • 电子商务网站设计的书甜品售卖网站网页设计
  • 营业执照上有以上除网站制作网站设计与制作前景
  • 做一个自己的免费网站吗长春网站制作小程序
  • 环保网站建设公司排名甘肃平凉建设局网站
  • 网站排名西安自己怎样做网站文章关键词内链
  • 怎么做集团网站网页设计主题内容
  • 开发高端网站建设手机做炫光头像图的网站