习惯了微信聊天,利用WebSocket手动实现个聊天功能怎么样?

msg.put(“msg”, “连接成功”);

msg.put(“status”, “SUCCESS”);

msg.put(“userId”, userId);

sendMessage(JSON.toJSONString(msg));

} catch (Exception e) {

logger.debug(“IO异常”);

}

}

/**

  • 连接关闭调用的方法

*/

@OnClose

public void onClose(@PathParam(“userId”) String userId ) {

//从set中删除

webSocketSet.remove(this);

onlineCount.decrementAndGet(); // 在线数减1

logger.info(“用户”+ userId +“退出聊天!当前在线用户数为” + onlineCount.get());

}

/**

  • 收到客户端消息后调用的方法

  • @param message 客户端发送过来的消息

*/

@OnMessage

public void onMessage(String message, @PathParam(“userId”) String userId ) {

//客户端输入的消息message要经过处理后封装成新的message,后端拿到新的消息后进行数据解析,然后判断是群发还是单发,并调用对应的方法

logger.info(“来自客户端” + userId + “的消息:” + message);

try {

MyMessage myMessage = JSON.parseObject(message, MyMessage.class);

String messageContent = myMessage.getMessage();//messageContent:真正的消息内容

String messageType = myMessage.getMessageType();

if(“1”.equals(messageType)){ //单聊

String recUser = myMessage.getUserId();//recUser:消息接收者

sendInfo(messageContent,recUser,userId);//messageContent:输入框实际内容 recUser:消息接收者 userId 消息发送者

}else{ //群聊

sendGroupInfo(messageContent,userId);//messageContent:输入框实际内容 userId 消息发送者

}

} catch (Exception e) {

logger.error(“解析失败:{}”, e);

}

}

/**

  • 发生错误时调用的方法

  • @OnError

**/

@OnError

public void onError(Throwable error) {

logger.debug(“Websocket 发生错误”);

error.printStackTrace();

}

public synchronized void sendMessage(String message) {

this.session.getAsyncRemote().sendText(message);

}

/**

  • 单聊

  • message : 消息内容,输入的实际内容,不是拼接后的内容

  • recUser : 消息接收者

  • sendUser : 消息发送者

*/

public void sendInfo( String message , String recUser,String sendUser) {

JSONObject msgObject = new JSONObject();//msgObject 包含发送者信息的消息

for (ChatWebSocketController item : webSocketSet) {

if (StringUtil.equals(item.userId, recUser)) {

logger.info(“给用户” + recUser + “传递消息:” + message);

//拼接返回的消息,除了输入的实际内容,还要包含发送者信息

msgObject.put(“message”,message);

msgObject.put(“sendUser”,sendUser);

item.sendMessage(JSON.toJSONString(msgObject));

}

}

}

/**

  • 群聊

  • message : 消息内容,输入的实际内容,不是拼接后的内容

  • sendUser : 消息发送者

*/

public void sendGroupInfo(String message,String sendUser) {

JSONObject msgObject = new JSONObject();//msgObject 包含发送者信息的消息

if (StringUtil.isNotEmpty(webSocketSet)) {

for (ChatWebSocketController item : webSocketSet) {

if(!StringUtil.equals(item.userId, sendUser)) { //排除给发送者自身回送消息,如果不是自己就回送

logger.info(“回送消息:” + message);

//拼接返回的消息,除了输入的实际内容,还要包含发送者信息

msgObject.put(“message”,message);

msgObject.put(“sendUser”,sendUser);

item.sendMessage(JSON.toJSONString(msgObject));

}

}

}

}

/**

  • Map/Set的key为自定义对象时,必须重写hashCode和equals。

  • 关于hashCode和equals的处理,遵循如下规则:

  • 1)只要重写equals,就必须重写hashCode。

  • 2)因为Set存储的是不重复的对象,依据hashCode和equals进行判断,所以Set存储的对象必须重写这两个方法。

  • 3)如果自定义对象做为Map的键,那么必须重写hashCode和equals。

  • @param o

  • @return

*/

@Override

public boolean equals(Object o) {

if (this == o) {

return true;

}

if (o == null || getClass() != o.getClass()) {

return false;

}

ChatWebSocketController that = (ChatWebSocketController) o;

return Objects.equals(session, that.session);

}

@Override

public int hashCode() {

return Objects.hash(session);

}

}

  • 3.4 声明Controller中的MyMessage实体类

public class MyMessage implements Serializable {

private static final long serialVersionUID = 1L;

private String userId;

private String message;//消息内容

private String messageType;//消息类型 1 代表单聊 2 代表群聊

public String getUserId() {

return userId;

}

public void setUserId(String userId) {

this.userId = userId;

}

public String getMessage() {

return message;

}

public void setMessage(String message) {

this.message = message;

}

public String getMessageType() {

return messageType;

}

public void setMessageType(String messageType) {

this.messageType = messageType;

}

}

  • 3.5 声明Controller中的StringUtil工具类

public final class StringUtil {

/**

  • 对象为空

  • @param object

  • @return

*/

public static boolean isEmpty(Object object) {

if (object == null) {

return true;

}

if (object instanceof String && “”.equals(((String) object).trim())) {

return true;

}

if (object instanceof List && ((List) object).size() == 0) {

return true;

}

if (object instanceof Map && ((Map) object).isEmpty()) {

return true;

}

if (object instanceof CharSequence && ((CharSequence) object).length() == 0) {

return true;

}

if (object instanceof Arrays && (Array.getLength(object) == 0)) {

return true;

}

return false;

}

/**

  • 对象不为空

  • @param object

  • @return

*/

public static boolean isNotEmpty(Object object) {

return !isEmpty(object);

}

/**

  • 查询字符串中某个字符首次出现的位置 从1计数

  • @param string 字符串

  • @param c

  • @return

*/

public static int strFirstIndex(String c, String string) {

Matcher matcher = Pattern.compile©.matcher(string);

if (matcher.find()) {

return matcher.start() + 1;

} else {

return -1;

}

}

/**

  • 两个对象是否相等

  • @param obj1

  • @param obj2

  • @return

*/

public static boolean equals(Object obj1, Object obj2) {

if (obj1 instanceof String && obj2 instanceof String) {

obj1 = ((String) obj1).replace(“\*”, “”);

obj2 = ((String) obj2).replaceAll(“\*”, “”);

if (obj1.equals(obj2) || obj1 == obj2) {

return true;

}

}

if (obj1.equals(obj2) || obj1 == obj2) {

return true;

}

return false;

}

/**

  • 根据字节截取内容

  • @param bytes 自定义字节数组

  • @param content 需要截取的内容

  • @return

*/

public static String[] separatorByBytes(double[] bytes, String content) {

String[] contentArray = new String[bytes.length];

double[] array = new double[bytes.length + 1];

array[0] = 0;

//复制数组

System.arraycopy(bytes, 0, array, 1, bytes.length);

for (int i = 0; i < bytes.length; i++) {

content = content.substring((int) (array[i] * 2));

contentArray[i] = content;

}

String[] strings = new String[bytes.length];

for (int i = 0; i < contentArray.length; i++) {

strings[i] = contentArray[i].substring(0, (int) (bytes[i] * 2));

}

return strings;

}

/**

  • 获取指定字符串出现的次数

  • @param srcText 源字符串

  • @param findText 要查找的字符串

  • @return

*/

public static int appearNumber(String srcText, String findText) {

int count = 0;

Pattern p = Pattern.compile(findText);

Matcher m = p.matcher(srcText);

while (m.find()) {

count++;

}

return count;

}

/**

  • 将字符串str每隔2个分割存入数组

  • @param str

  • @return

*/

public static String[] setStr(String str) {

int m = str.length() / 2;

if (m * 2 < str.length()) {

m++;

}

String[] strings = new String[m];

int j = 0;

for (int i = 0; i < str.length(); i++) {

if (i % 2 == 0) {

//每隔两个

strings[j] = “” + str.charAt(i);

} else {

strings[j] = strings[j] + str.charAt(i);

j++;

}

}

return strings;

}

/**

  • 定义一个StringBuffer,利用StringBuffer类中的reverse()方法直接倒序输出

  • 倒叙字符串

  • @param s

*/

public static String reverseString2(String s) {

if (s.length() > 0) {

StringBuffer buffer = new StringBuffer(s);

return buffer.reverse().toString();

} else {

return “”;

}

}

/**

  • 截取字符串中的所有日期时间

  • @param str

  • @return

*/

public static List dateTimeSubAll(String str) {

try {

List dateTimeStrList = new ArrayList<>();

String regex = “[0-9]{4}[-][0-9]{1,2}[-][0-9]{1,2}[ ][0-9]{1,2}[:][0-9]{1,2}[:][0-9]{1,2}”;

Pattern pattern = compile(regex);

Matcher matcher = pattern.matcher(str);

while (matcher.find()) {

String group = matcher.group();

dateTimeStrList.add(group);

}

return dateTimeStrList;

} catch (Exception e) {

e.getMessage();

return null;

}

}

/**

  • 截取字符串中的所有日期

  • @param str

  • @return

*/

public static List dateSubAll(String str) {

try {

List dateStrList = new ArrayList<>();

Pattern pattern = compile(“[0-9]{4}[-][0-9]{1,2}[-][0-9]{1,2}”);

Matcher matcher = pattern.matcher(str);

while (matcher.find()) {

String group = matcher.group();

dateStrList.add(group);

}

return dateStrList;

} catch (Exception e) {

e.getMessage();

return null;

}

}
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

最后

基础知识是前端一面必问的,如果你在基础知识这一块翻车了,就算你框架玩的再6,webpack、git、node学习的再好也无济于事,因为对方就不会再给你展示的机会,千万不要因为基础错过了自己心怡的公司。前端的基础知识杂且多,并不是理解就ok了,有些是真的要去记。当然了我们是牛x的前端工程师,每天像背英语单词一样去背知识点就没必要了,只要平时工作中多注意总结,面试前端刷下题目就可以了。

什么?你问面试题资料在哪里,这不是就在你眼前吗(滑稽

自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!**

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

[外链图片转存中…(img-EuzyJ5g9-1712570125461)]

[外链图片转存中…(img-T2g3cf8s-1712570125461)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

[外链图片转存中…(img-C4qa6pe1-1712570125461)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

最后

基础知识是前端一面必问的,如果你在基础知识这一块翻车了,就算你框架玩的再6,webpack、git、node学习的再好也无济于事,因为对方就不会再给你展示的机会,千万不要因为基础错过了自己心怡的公司。前端的基础知识杂且多,并不是理解就ok了,有些是真的要去记。当然了我们是牛x的前端工程师,每天像背英语单词一样去背知识点就没必要了,只要平时工作中多注意总结,面试前端刷下题目就可以了。

什么?你问面试题资料在哪里,这不是就在你眼前吗(滑稽

资料领取方式:戳这里获取

  • 27
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值