java.好友发送验证申请_SpringBoot+LayIM+t-io 实现好友申请通知流程

前言

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

简介

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

准备工作

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

cfad063b28a40e917466173a7fb4d73d.png

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

e664954c4ed93f506d9036e93a1fb044.png

绑定数据部分,简单给大家介绍一下,就是用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 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中的数据取出来。

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

发起好友申请

我们先根据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即可。

服务端主动推送

以下代码在  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 {

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);

}

功能拼接

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

/**

* 好友申请

* */

@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;

}

功能演示

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

064313fe386ea9d0e0e344cc56063426.png

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

e2ef7591e4d860f504f17db3b1a63d58.png

皇上点开消息盒子:

c3dfcde0ffe97222412b50e417fc8355.png

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

总结

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

好友申请按钮出不出现取决于用户是否为自己,是否已经是好友。(后端也要做验证)

t-io的服务端主动推送,如何调用。关键词: servergroupcontext

event的使用,除了applicationevent,还可以拓展其他类型,如消息队列,eventbus等。

各种细节处理,比如先判断对方是否在线,在去查询数据库。或者结合缓存等

由于是自己摸索,难免有代码繁杂混乱之处,

以上所述是小编给大家介绍的springboot+layim+t-io 实现好友申请通知流程,希望对大家有所帮助

希望与广大网友互动??

点此进行留言吧!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot、Vue-Element-Admin和MyBatis是一种常见的技术组合,用于构建现代化的Web应用程序。下面是一个简要的入门教程。 1. 首先,我们需要设置Spring Boot项目。可以使用Spring Initializr(https://start.spring.io/)来初始化一个基本的Spring Boot项目。在依赖项中添加Spring Web、Spring Data JPA和MyBatis等必要的依赖项。 2. 在Spring Boot项目中,我们需要创建实体类和数据库表的映射。使用JPA注解来定义实体类,并使用MyBatis注解来指定数据库表的映射。 3. 接下来,我们需要创建一个数据访问层(DAO)来处理与数据库的交互。使用MyBatis的注解或XML映射文件来定义SQL查询和操作。 4. 在Service层,编写业务逻辑代码来处理DAO返回的数据,并与其他组件进行交互。 5. 在Controller层,处理HTTP请求和响应,将数据传递给前端页面或接收前端发送的数据。使用Spring MVC注解来定义请求映射和参数解析。 6. 在前端方面,可以使用Vue-Element-Admin来构建用户界面。Vue是一种流行的JavaScript框架,用于构建灵活的单页面应用程序。Element-Admin是一个基于Vue的组件库,提供了丰富的UI组件和布局。 7. 在Vue-Element-Admin中,我们可以使用Vue Router来实现页面之间的导航和路由。使用axios来发送HTTP请求与后端进行数据交互。 8. 在Vue组件中,我们可以通过调用后端的API来获取数据并渲染到前端页面上。使用Element-Admin提供的布局和UI组件来美化页面。 通过上述步骤,我们可以实现一个基本的Spring Boot、Vue-Element-Admin和MyBatis的入门教程。这个教程可以帮助初学者了解如何搭建和使用这个技术组合来构建现代化的Web应用程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值