引子
最近在开发一个社交软件的过程中用到了环信即时通讯。然后发现系统第一次启动的时候添加好友会很慢。随即跟了下代码发现是创建Token的过程比较耗时间。于是引出了两个问题需要解决:
1、系统启动后第一次创建token速度慢的问题
2、token过期(token默认有效期为7天)后,重新创建速度慢的问题
源码分析
首先我查看了它的源代码,发现里面封装的结构其实是比较简单的
1、入口类EasemobIMUsers,里面有一个静态的全局变量ClientSecretCredential
private static Credential credential = new ClientSecretCredential(Constants.APP_CLIENT_ID,
Constants.APP_CLIENT_SECRET, Roles.USER_ROLE_APPADMIN);
2、EasemobIMUsers,里面有一个Token,这个Token就是我们操作用的token
protected Token token;
@Override
public Token getToken() {
if (null == token || token.isExpired()) {
try {
ObjectNode objectNode = factory.objectNode();
objectNode.put("grant_type", "client_credentials");
objectNode.put("client_id", tokenKey1);
objectNode.put("client_secret", tokenKey2);
List<NameValuePair> headers = new ArrayList<NameValuePair>();
headers.add(new BasicNameValuePair("Content-Type", "application/json"));
HttpPost httpPost = new HttpPost();
httpPost.setURI(CLIENTSECRETCREDENTAIL_TOKEN_URL.toURI());
for (NameValuePair nameValuePair : headers) {
httpPost.addHeader(nameValuePair.getName(), nameValuePair.getValue());
}
httpPost.setEntity(new StringEntity(objectNode.toString(), "UTF-8"));
HttpResponse tokenResponse = client.execute(httpPost);
HttpEntity entity = tokenResponse.getEntity();
String results = EntityUtils.toString(entity, "UTF-8");
LOGGER.info("-----------------------------返回结果-------------------------------statuscode:"
+ tokenResponse.getStatusLine().toString());
if (tokenResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {
ObjectMapper mapper = new ObjectMapper();
JsonFactory factory = mapper.getJsonFactory();
JsonParser jp = factory.createJsonParser(results);
JsonNode json = mapper.readTree(jp);
String accessToken = json.get("access_token").asText();
Long expiredAt = System.currentTimeMillis() + json.get("expires_in").asLong() * 1000;
token = new Token(accessToken, expiredAt);
}
} catch (Exception e) {
throw new RuntimeException("Some errors occurred while fetching a token by username and password .");
}
}
return token;
}
关键就在这个getToken()方法。我们不要去看他复杂的处理,只要知道干了什么事:token存在且没过期时,直接返回token;否则,重新创建一个token。性能的瓶颈就在于这个重新创建的地方。问题找到了,接下来我们来寻求解决方案
解决方法
第一个问题很好解决,就在是系统启动的时候访问访问ClientSecretCredential的getToken()。
接下来第二个问题,我目前能能想到的解决方法有四个
1、直接设置token为永不失效,但是没能查到环信是否支持这种设置的信息。
2、把对环信的操作放到一个单独的线程中,无需在等待返回结果,我们项目里面这样做无法保证环信的操作和我们自己服务器上操作的一致性,而且线程会暂用比较多的资源。
3、把token取到自己的后台,利用定期(比如6天激发一次更新token的操作),这样实现起来可能会稍微有点麻烦
4、设置一个更新token的时间点,比如token有效期不足一天时,提前更新。具体是每次调用完token后,判断token是否已经需要更新,如果需要在另起一个线程用于更新token。(具体更新方法是,设置有效期为过期,然后访问getToken()方法)。
我目前采用的是第四种解决方案。各位如果谁看到这篇文章,并且了解第一种方案是否可行的话,请留言告诉我(先说声谢谢分享)。如果还有什么好的解决方法也可以分享出来,如果这篇文章能解决你的一些问题我也会很高兴。