SpringBoot+LayIM+t-io 实现好友申请通知流程

SpringBoot+LayIM+t-io 实现好友申请通知流程,第1张

概述前言在上一篇Springboot+LayIM+t-io文件上传、监听用户状态的实现中,已经介绍了两个小细节:用户的在离线状态和群人数的状态变化。今天的主要内容就是用户加好友的实现。

@H_403_2@前言

在上一篇 Spring boot + LayIM + t-io 文件上传、 监听用户状态的实现 中,已经介绍了两个小细节:用户的在离线状态和群人数的状态变化。今天的主要内容就是用户加好友的实现。

@H_403_2@简介

加好友,大家用过QQ都知道,无非是发起好友申请,对方收到消息通知,然后处理。不过,本篇只讲前半部分,消息通知的处理留到下一篇去讲。因为内容有点多,怕是一时半会消化不了。在介绍主体流程之前,先给大家介绍一下准备工作。

@H_403_2@准备工作

首先,为了让数据更贴近实战,所以我用了比较“真实”的用户数据。结合fly模板,完善了用户中心头部的用户信息的数据绑定。数据绑定部分判断了是否已经是好友,来决定是否出现“加为好友”的按钮。示例如下,当用户自己看到自己的主页时,是这样的:

 

看到非好友的用户主页,是这样的:

 

绑定数据部分,简单给大家介绍一下,就是用thymleaf模板绑定。后台访问页面的时候,将 Model 赋值即可。

/** * 属性赋值 * */ private voID setModel(User user,Model model){ long currentUserID = getUserID(); long visitUserID = user.getID(); //是否是自己 boolean isSelf = currentUserID == visitUserID; //两个用户是否已经是好友 boolean isFrIEnd = groupService.isFrIEnd(currentUserID,visitUserID); Map<String,Object> userMap = new HashMap<>(8); userMap.put("avatar",user.getAvatar()); userMap.put("name",user.getUsername()); userMap.put("addtime",TimeUtil.formatDate(user.getCreateAt())+" 加入"); if(user.getSign()==null ||user.getSign().length()==0) {  userMap.put("sign",""); }else {  userMap.put("sign","(" + user.getSign() + ")"); } userMap.put("uID",user.getID()); userMap.put("self",isSelf || isFrIEnd); model.addAttribute("user",userMap); }

然后页面上,将model中的数据取出来。

<div  > <input type="hIDden" th:value="${user.uID}" ID="visitUID"/> <img src="" th:src="${user.avatar}" th:alt="${user.name}"/> <h1>  <p th:text="${user.name}"></p>  <i ></i> </h1> <p >  <!--<i  title="飞吻"></i><span >67206飞吻</span>-->  <i ></i><span th:text="${user.addtime}"></span>  <!--<i ></i><span>来自杭州</span>-->  <i  th:if="${user.self==false}"></i><a lay-event="addFrIEnd" href="#" rel="external nofollow" title="添加TA为好友" th:if="${user.self==false}">加为好友</a> </p> <p  th:text="${user.sign}"></p> </div>

ok,以上就是简单的准备工作。想了解详情代码的可以去文末的github地址去搜寻。

@H_403_2@发起好友申请

我们先根据layim的业务分析。首先,要知道我们要加谁(toID)为好友。然后在加上一个备注(remark)。这些东西交给后台就OK了。为了避免连表查询,对于系统消息的存储我做了用户名和用户头像的冗余。表主要包含字段:用户ID,用户头像,用户名,被申请用户ID,申请时间,申请类型,备注,已读等其他属性。

所以,发起好友申请就很简单了。就是一个添加功能,前端传的就是被申请人用户ID和申请备注,后端组织数据插入到数据库,代码如下:

/** * 提交好友申请 * */ public JsonResult saveFrIEndApply(long toID,String remark){ remark = HTMLUtils.HTMLEscape(remark); ContextUser user = ShiroUtil.getCurrentUser(); long userID = Long.parseLong(user.getUserID()); int record = applyRepository.countByToIDAndUIDAndTypeAndResult(toID,userID,ApplyType.frIEnd,0); if(record > 0){  return JsonResult.fail("已经申请过"); } Apply apply = new Apply(); apply.setType(ApplyType.frIEnd); apply.setToID(toID); apply.setRemark(remark); apply.setUID(userID); apply.setAvatar(user.getAvatar()); apply.setname(user.getUsername()); apply.setRead(false); apply.setResult(0); return saveApply(apply); }

OK,申请完了,下面我们要做啥?没错,通知对方,喂,我向你发送了申请,快快处理。在这里呢我遇到了一个问题。由于springboot程序占用端口 8080,而t-io占用端口8888,也就是说,如果我想在8080端口的业务中主动调用8888的服务推送,我不知道如何获取相应的channelContext。不过经过询问作者之后,一句话解决了我的问题。

拿到 ServerGroupContext ,问题迎刃而解。

在之前的程序启动的时候注册了 LayimWebsocketStarter 这个bean。所以,在8080业务端如果能拿到它的话就没问题了。

得到 LayimWebsocketStarter ,就能得到  ServerGroupContext, 然后就能在服务端做主动推送了。

当然可能没有开发过这个东西,对于上文中的问题不是很理解,没关系,其实我就想说明,如果从服务端主动向客户端推送消息的话,使用ServerGroupContext即可。

@H_403_2@服务端主动推送

以下代码在  com.fyp.layim.im.common.util.PushUtil 中

OK,接上文,我们按照步骤来。

第一步,获取 LayimWebsocketStarter 。

/** * 获取starter*/private static LayimWebsocketStarter getStarter(){ return (LayimWebsocketStarter)SpringUtil.getBean("layimWebsocketStarter"); }

第二步,获取 ServerGroupContext

private static ServerGroupContext getServerGroupContext(){ return getStarter().getServerGroupContext(); }

第三步,获取 ChannelContext。

/** * 获取channelContext * */ private static ChannelContext getChannelContext(String toID) { ServerGroupContext context = getServerGroupContext(); //找到用户 ChannelContext channelContext = context.users.find(context,toID); return channelContext; }

第四步,发射,这里的代码就和聊天中的那部分代码差不多了。核心部分就是,获取ChannelContext,然后给他发送消息。如果不在线就不用管。

/** * 服务端主动推送消息 * */ public static voID pushApplyMessage(String toID) { logger.info("执行到了发送方法:pushApplyMessage"); LayimToClIEntNoticeMsgBody body = new LayimToClIEntNoticeMsgBody(); ChannelContext channelContext = getChannelContext(toID); //先判断是否在线,再去查询数据库,减少查询次数 if (channelContext != null && !channelContext.isClosed()) {  int count = getApplyService().getUnreadMsgCount(Long.parseLong(toID));  body.setCount(count);  push(channelContext,body); } } /** * 服务端主动推送消息 * */ private static voID push(ChannelContext channelContext,Object msg) { try {  WsResponse response = BodyConvert.getInstance().convertToTextResponse(msg);  Aio.send(channelContext,response); }catch (IOException ex){ } }

现在推送已经搞定了,那么什么时候推送呢?由于这个系统消息的推送可以不用那么即时,于是我看了下,springboot里面有类似的事件机制,于是乎 ApplyEvent 就诞生了。

public class ApplyEvent extends ApplicationEvent { public ApplyEvent(Object source) { super(source); } private long toID; public long getToID(){ return toID; } public ApplyEvent(Object source,long toID) { super(source); this.toID = toID; }}

在创建一个Listener,监听事件。

public class ApplyListener implements ApplicationListener<ApplyEvent> { private Logger logger = LoggerFactory.getLogger(ApplyListener.class); @OverrIDe public voID onApplicationEvent(ApplyEvent applyEvent) { new Thread(){  public voID run(){  Long toID = applyEvent.getToID();  //这里就要调用上文中的推送了  PushUtil.pushApplyMessage(toID.toString());  } }.start(); }}

不过我有个疑问,发现Listener中执行的时候是同步的。后来加了@Async 和@EnableAsync 也没用,于是我就用了new Thread().start()实现异步,确保不影响主要申请流程。(这是个疑问,自己没搞明白的地方)

最后,别忘了在Application启动的时候把Listener加上。

public static voID main(String[] args) { SpringApplication springApplication = new SpringApplication(LayimApplication.class); /**  * 这里监听增加Listener,Listener才会触发  * ApplyListener 是监听好友申请的事件  * */ springApplication.addListeners(new ApplyListener()); springApplication.run(args); }

@H_403_2@功能拼接

马上就要成功了,我们在把事件串起来,在好友申请成功之后,发布事件。

/** * 好友申请 * */ @PostMapPing(value = "/apply-frIEnd") public JsonResult apply(@RequestParam("toID") Long toID,@RequestParam("remark") String remark){ JsonResult result = applyService.saveFrIEndApply(toID,remark); //申请成功,发布申请事件,通知 toID处理消息,如果不在线,不会进行处理 if(result.isSuccess()){  applicationContext.publishEvent(new ApplyEvent("apply",toID)); } return result; }

@H_403_2@功能演示

讲了那么多,给大家看一下成品效果。(用户场景:安小鸟加皇上为好友,皇上接收消息并查看)

 

皇上收到消息,系统d出左下角的小数字4。(调用 layim.msgBox(msgCount) 方法)

 

皇上点开消息盒子:

 

皇上收到了四位爱妃的申请,寝食难安,他会怎么处理呢?欲知后事如何,且听下回分解~~~

@H_403_2@总结

本篇主要介绍了一个加好友的流程的实现。

好友申请按钮出不出现取决于用户是否为自己,是否已经是好友。(后端也要做验证) t-io的服务端主动推送,如何调用。关键词: ServerGroupContext event的使用,除了applicationEvent,还可以拓展其他类型,如消息队列,eventbus等。 各种细节处理,比如先判断对方是否在线,在去查询数据库。或者结合缓存等 由于是自己摸索,难免有代码繁杂混乱之处,

文中代码地址: https://github.com/fanpan26/SpringBootLayIM

                           http://xiazai.jb51.net/201712/yuanma/SpringBootLayIM-master.rar

以上所述是小编给大家介绍的SpringBoot+LayIM+t-io 实现好友申请通知流程,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对编程小技巧网站的支持!

总结

以上是内存溢出为你收集整理的SpringBoot+LayIM+t-io 实现好友申请通知流程全部内容,希望文章能够帮你解决SpringBoot+LayIM+t-io 实现好友申请通知流程所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1227365.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-05
下一篇 2022-06-05

发表评论

登录后才能评论

评论列表(0条)

保存