【13】JMicro微服务-ID生成与Redis
如非授权,禁止用于商业用途,转载请注明出处
作者:mynewworldyyl
往下看前,建议完成前面1到12小节
1. 微服务中ID地位
如果说前面小节的功能点是微服务的大脑,那么全局唯一ID则是微服务的神经系统,没有ID这个神经系统,再强的大脑也白搭,只有有了这个神经系统,才能有效协调整个微服务系统的正常工作,才不会出现神经错乱。就好像两个或多个人的身份证号码相同,则依赖于这个身份证号唯一性的系统就无法正常工作(无法为具有相同身份证号的这些人服务)。
JMicro中,消息是微服务之间通讯最基本单元,系统将之命名为org.jmicro.api.net.Message,每个消息由唯一ID所标识,不管系统中有多少个服务,服务之间以多高的QPS做交互,也许上万亿的QPS,也不管系统运行多久,也许是1亿年,假设这1亿年中总共有万万亿个消息,那么这万万亿个消息的ID都不能重复,否则这个微服务系统都是不可靠的。例如,两个转账消息,一个转1分钱,一个传1亿,如果两个消息ID相同,那么有50%的可能本应收到1亿的账号却收到了1分钱,而那个本应收到1分钱账号却收到了1亿,如果是比特币转账,那后果就凉了。
2. JMicro中ID使用场景
最典型的是服务调用链路跟踪(【6】JMicro微服务-服务日志监控 ),从调用发起方开始,生成一个全局唯一链路ID,并且将这个链路ID从调用者传给被调用者,直到最终被调用者返回时,也将链路ID返回,直到返回给调用发起方。JMIcro通过将相同的链路ID消息归属为一个微服务调用过程,这个相同ID的链路上发生的所有事件及相关统计数据,都可以通过这个ID作为维度做跟踪分析。比如从调用发起方发送请求到收到响应,消耗很长时间,那么通过链路ID查看每个消息请求和响应时间,即可找到问题的结点,这里的“结点”即是JMicro服务。
还有就是JMicro的RPC调用过程中,为了提高性能,消息都是异步的,但是作为RPC客户端调用者,却是同步的方式使用,比如前面的ISimpleRpc调用hello服务方法,返回一个字符串,是同步返回的。这个底层异步消息提供上层同步使用,就是通过请求ID实现的。每个RPC请求,会生成一个全局唯一ID,并将这个ID传给服务方,服务方处理完成后,也返回给调用方,调用方底层通过这个ID识别这个结果应该返回给那个RPC方法(唤醒调用者线程)。
3. 应用全局唯一ID
应用如果需要使用到全局唯一ID,也可以和JMicro底层一样,使用相同接口,如下代码是JMicro获取当前链路ID代码。
public static Long lid(){
JMicroContext c = get();
Long id = c.getLong(LINKER_ID, null);
if(id != null) {
return id;
}
ComponentIdServer idGenerator = JMicro.getObjectFactory().get(ComponentIdServer.class);
if(idGenerator != null) {
id = idGenerator.getLongId(Linker.class);
c.setLong(LINKER_ID, id);
}
return id;
}
2到6行检查当前上下文是否有链路ID,如果有则直接返回,否则第8行取得ComponentIdServer 实例,并通过ComponentIdServer实例的相关方法获取Linker.class的
ID,下图为ComponentIdServer实例获取ID相关方法:

3种类型6个方法,分别为int类型ID,获取1个id和多个ID两个方法,同理Long和String。
取得ComponentIdServer 实例的另一种方式如下,通过@Inject注解获取:
@Inject
private ComponentIdServer idGenerator;
4. 基于Redis全局唯一ID生成方案
如下代码为基于Redis的lua脚本,获取特定KEY的cnt个ID,使用了Redis调用Lua脚本的原子性。
public JMicroRedisBaseIdGenerator() {
StringBuilder sb = new StringBuilder();
sb.append("local k = KEYS[1];\n");
sb.append("local cnt = ARGV[1];\n");
sb.append("local val = tonumber(redis.call('incrby', k, cnt));\n");
sb.append("return val;\n");
luaScript = sb.toString();
}
获取Int类型的ID
public Integer[] getIntIds(String idKey, int num) {
Jedis r = pool.getResource();
try {
int endId = Integer.parseInt(r.eval(luaScript, 1, idKey,num+"").toString());
Integer[] ids = new Integer[num];
int oriId = endId - num;
for(int i = 0; i < num; i++) {
ids[i] = oriId+i;
}
return ids;
}finally {
r.close();
}
}
5. 使用Redis
在org.jmicro.redis.RegistRedis中,有如下方法实现将Redis相关接口实例注册到IObjectFactory中

所以我们可以通过@Inject将以上实例对象注入到我们的代码中,如下代码所示:
@Inject(required=true)
private JedisPool pool;
到目前为止,我们在JMIcro环境中,自动拥有了ZK及Redis环境支持,后面还会有MyBatis相关支持,用以操作数据库,如果有兴趣,可提前通过源码查看JMIcror插件式的模块扩展机制。Github链接如下:
https://github.com/mynewworldyyl/jmicro/tree/master/jmicro.ext/jmicro.mybatis
https://github.com/mynewworldyyl/jmicro/tree/master/jmicro.redis
https://github.com/mynewworldyyl/jmicro/tree/master/jmicro.zk
【13】JMicro微服务-ID生成与Redis的更多相关文章
- 【2】JMicro微服务-Hello World
如非授权,禁止用于商业用途,转载请注明出处作者:mynewworldyyl 1. 首先完成 JMicro微服务-RPC体验 的1到5步. 按默认方式启动ZK及Redis: JDK需要Java8及以上. ...
- 微服务架构中的Redis
了解如何将Redis与Spring Cloud和Spring Data一起使用以提供配置服务器,消息代理和数据库. Redis可以广泛用于微服务架构中.它可能是少数流行的软件解决方案之一,你的应用程序 ...
- 【12】JMicro微服务-Zookeeper
如非授权,禁止用于商业用途,转载请注明出处作者:mynewworldyyl 往下看前,建议完成前面1到11小节 1. CuratorFramework支持 JMicro目前基于Zookeeper实现统 ...
- 【11】JMicro微服务-配置管理
如非授权,禁止用于商业用途,转载请注明出处作者:mynewworldyyl 往下看前,建议完成前面1到10小节 JMicro目前仅支持基于Zookeeper做配置管理,全部配置信息可以在ZK做增删改查 ...
- 【10】JMicro微服务-API网关
如非授权,禁止用于商业用途,转载请注明出处作者:mynewworldyyl 往下看前,建议完成前面1到9小节 1. Api网关基本特性: Api网关作为对外网提供服务的基本入口,地位类似于NGINX, ...
- 【8】JMicro微服务-JMicro ZKUI
ZKUI是一个开源项目,是一个查看,修改ZK数据非常方便的工具.JMicro基于ZK做服务治理,配置管理,因此使用ZKUI会提供非常大的方便. Github地址:https://github.com/ ...
- 【3】JMicro微服务-服务超时,重试,重试间隔
如非授权,禁止用于商业用途,转载请注明出处作者:mynewworldyyl 接下来的内容都基于[2]JMicro微服务-Hello World做Demo 微服务中,超时和重试是一个最基本问题下面Dem ...
- JMicro微服务之超时&重试
JMicro是本人开发的基于Java实现的微服务框架,当前正式版本为0.0.3,并已发布到maven中央仓库.项目源码github:https://github.com/mynewworldyyl/j ...
- 你还在手撕微服务?快试试 go-zero 的微服务自动生成
0. 为什么说做好微服务很难? 要想做好微服务,我们需要理解和掌握的知识点非常多,从几个维度上来说: 基本功能层面 并发控制&限流,避免服务被突发流量击垮 服务注册与服务发现,确保能够动态侦测 ...
随机推荐
- shell的基本语法
一 赋值运算符 1 += :使用方法是,((x+=需要增加的数字))算和值. 2 *= :使用方法是,((x*=需要怎加的倍数))算乘值. 3 %= :使用方法是,((x%=需要除以的数字))算余数 ...
- 着重基础之—MySql 不能遗忘的索引操作
着重基础之—MySql 不能遗忘的索引操作 关于MySql索引的基础知识我就不在这里写了,我不太想当信息的搬运工. 技巧分享:Workbench 作为一款专为MySQL设计的ER/数据库建模工具.除了 ...
- js数组合并(一个数组添加到另一个数组里面)方法
js定义两个数组. var arrA=[1,2,3]; var arrB=[4,5,6]; 要实现[1,2,3,4,5,6],如果直接arrA.push(arrB); 则arrB只会作为了arrA的一 ...
- python类的继承-1
#!/usr/bin/python3 #类定义 class people: #定义基本属性 name = '' age = 0 #定义私有属性,私有属性在类外部无法直接进行访问 __weight = ...
- UVa 12003 Array Transformer (分块)
题意:给定一个序列,然后有 m 个修改,问你最后的序列是什么,修改是这样的 l r v p 先算出从 l 到 r 这个区间内的 小于 v 的个数k,然后把第 p 个的值改成 k * u / (r - ...
- UVa 11384 Help is needed for Dexter (递归)
题意:给定一个n表示1到n的序列,让你用最小的步数把这个序列都变为0,每个操作可以从序列中选择一个或多个个,同时减掉一个正整数,求最少的步数. 析:一看这个题,感觉挺高深的,但是静下心来想想,其实挺简 ...
- (二分匹配 模板 KM)奔小康赚大钱--hdu--2255
链接: http://acm.hdu.edu.cn/showproblem.php?pid=2255 代码: #include <iostream> #include <cstdio ...
- 探求Floyd算法的动态规划本质
Floyd–Warshall(简称Floyd算法)是一种著名的解决任意两点间的最短路径(All Paris Shortest Paths,APSP)的算法.从表面上粗看,Floyd算法是一个非常简单的 ...
- Bellman_ford货币兑换——正权回路判断
POJ1860 题目大意:你在某一点有一些钱,给定你两点之间钱得兑换规则,问你有没有办法使你手里的钱增多.就是想看看转一圈我的钱能不能增多,出现这一点得条件就是有兑换钱得正权回路,所以选择用bellm ...
- poj1836 Alignment
Alignment Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 11707 Accepted: 3730 Descri ...