共计 24946 个字符,预计需要花费 63 分钟才能阅读完成。
1. 性能实现
1. 批改性能(明码、昵称、共性签名)
2. 增加好友、删除好友
3. 单聊性能
4. 判断好友是否在线
2. 模块划分
==
3. 应用的常识
- netty
- swing
- 汇合等同步阻塞队列 synchronousQueue
- 数据库 MySQL 中的 CRUD
- C3p0 连接池
- JSON 字符串
==
4. 局部代码实现
1.nettyController.java
- 接管到来自客户端的音讯,与 dao 层进行交互
- dao 层与之数据库进行交互
批改明码
增加好友
从增加好友逻辑实现上我走了很多的弯路频繁的拜访数据库,这是一件很不好的事件
package chat.Project.controller;
import chat.Project.bean.information;
import chat.Project.constant.EnMsgType;
import chat.Project.dao.*;
import chat.utils.CacheUtil;
import chat.utils.JsonUtils;
import com.fasterxml.jackson.databind.node.ObjectNode;
import io.netty.channel.Channel;
import java.util.ArrayList;
import java.util.Iterator;
public class NettyController {private static UserDao userDao = new UserDaoImpl();
private static informationDao informationDao = new informationDaoImpl();
private static friendDao friendDao = new friendDaoImpl();
public static String processing(String message, Channel channel){
// 解析客户端发送的音讯
ObjectNode jsonNodes = JsonUtils.getObjectNode(message);
String msgtype = jsonNodes.get("msgtype").asText();
if (EnMsgType.EN_MSG_LOGIN.toString().equals(msgtype)){
// 登录操作
return loginOperation(jsonNodes,channel);
}else if (EnMsgType.EN_MSG_MODIFY_SIGNATURE.toString().equals(msgtype)){
// 批改签名
return modifySignature(jsonNodes);
}else if (EnMsgType.EN_MSG_MODIFY_NICKNAME.toString().equals(msgtype)){
// 批改昵称
return modifyNickname(jsonNodes);
}else if (EnMsgType.EN_MSG_GETINFORMATION.toString().equals(msgtype)){
// 获取登录信息
return getInformation(jsonNodes);
}else if (EnMsgType.EN_MSG_VERIFY_PASSWORD.toString().equals(msgtype)){
// 进行批改明码
return verifyPasswd(jsonNodes);
}else if (EnMsgType.EN_MSG_CHAT.toString().equals(msgtype)){
// 单聊模式
return SingleChat(jsonNodes);
}else if (EnMsgType.EN_MSG_GET_ID.toString().equals(msgtype)){
// 获取 id
return getId(jsonNodes);
}else if (EnMsgType.EN_MSG_GET_FRIEND.toString().equals(msgtype)){
// 获取好友列表
return getFriend(jsonNodes);
}else if (EnMsgType.EN_MSG_ADD_FRIEND.toString().equals(msgtype)){
// 增加好友
return addFriends(jsonNodes);
}else if (EnMsgType.EN_MSG_DEL_FRIEND.toString().equals(msgtype)){
// 删除好友
return delFriend(jsonNodes);
}else if (EnMsgType.EN_MSG_ACTIVE_STATE.toString().equals(msgtype)){
// 判断好友的在线状态
return friendIsActive(jsonNodes);
}
return "";
}
// 判断好友在线状态
private static String friendIsActive(ObjectNode jsonNodes) {int friendId = jsonNodes.get("friendId").asInt();
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_ACTIVE_STATE.toString());
// 客户端保障用户独立存在且是好友
Channel channel = CacheUtil.get(friendId);
// 判断用户是否在线
if (channel == null){
// 用户不在线
objectNode.put("code",200);
}else {
// 用户在线
objectNode.put("code",300);
}
return objectNode.toString();}
// 增加好友
private static String delFriend(ObjectNode jsonNodes) {Integer friendId = jsonNodes.get("friendId").asInt();
int userId = jsonNodes.get("id").asInt();
String localName = jsonNodes.get("localName").asText();
// 封装发回客户端的 JSON
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_DEL_FRIEND.toString());
objectNode.put("code",200);
// 验证是否存在以后好友
information information = informationDao.getInformation(friendId);
String friendName = information.getNickname();
// 查问本人是否有该好友
boolean exist = friendDao.isExist(friendName,userId);
if (exist){
// 存在以后好友进行删除操作
friendDao.delFriend(userId,friendName);
friendDao.delFriend(friendId,localName);
objectNode.put("code",300);
}
return objectNode.toString();}
// 增加好友
private static String addFriends(ObjectNode jsonNodes) {Integer friendId = jsonNodes.get("friendId").asInt();
int userId = jsonNodes.get("id").asInt();
String localName = jsonNodes.get("localName").asText();
// 验证是否有 ID
boolean exists = userDao.verifyExistFriend(friendId);
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_ADD_FRIEND.toString());
objectNode.put("code",200);
if(exists){
// 示意存在此 id
objectNode.put("code",300);
// 获取好友昵称
information information = informationDao.getInformation(friendId);
String friendNickname = information.getNickname();
// 进行增加好友的操作 两个对应的信息都应该增加
friendDao.addFriends(userId,localName,friendNickname);
friendDao.addFriends(friendId,friendNickname,localName);
}
return objectNode.toString();}
// 获取好友列表
private static String getFriend(ObjectNode jsonNodes) {int uid = jsonNodes.get("uid").asInt();
// 返回 ArrayLis 汇合
ArrayList<String> friends = friendDao.getFriends(uid);
// 封装 JSON
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_GET_FRIEND.toString());
// 写回 friend 汇合
Iterator<String> iterator = friends.iterator();
int i = 0;
while (iterator.hasNext()){objectNode.put("res"+i,iterator.next());
i++;
}
// 记录好友个数
objectNode.put("count",i);
return objectNode.toString();}
// 获取 id
private static String getId(ObjectNode jsonNodes) {String nickname = jsonNodes.get("nickname").asText();
information information = informationDao.nicknameGetId(nickname);
// 联系人的 id
int uid = information.getUid();
// 封装 JSON
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_GET_ID.toString());
objectNode.put("uid",uid);
return objectNode.toString();}
// 单聊模式
private static String SingleChat(ObjectNode jsonNodes) {int id = jsonNodes.get("id").asInt();
// 依据 id 在 friend 表获取登录用户名
// 封装 JSON 数据服务端转发数据
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_CHAT.toString());
// 客户端保障用户独立存在且是好友
Channel channel = CacheUtil.get(id);
// 判断用户是否在线
if (channel == null){
// 用户不在线
objectNode.put("code",200);
}else{
// 用户在线
objectNode.put("code",300);
// 音讯转发
channel.writeAndFlush(jsonNodes.toString());
}
return objectNode.toString();}
// 批改明码
private static String verifyPasswd(ObjectNode jsonNodes) {int id = jsonNodes.get("id").asInt();
String oldPasswd = jsonNodes.get("oldPasswd").asText();
String newPasswd = jsonNodes.get("newPasswd").asText();
boolean exits = userDao.verifyPassword(oldPasswd, id);
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_VERIFY_PASSWORD.toString());
objectNode.put("code",200);
if (exits){
// 验证胜利
userDao.modifyPasswd(newPasswd,id);
objectNode.put("code",300);
}
return objectNode.toString();}
// 获取信息
private static String getInformation(ObjectNode jsonNodes) {int id = jsonNodes.get("id").asInt();
information information = informationDao.getInformation(id);
// 封装 JSON 发回客户端
ObjectNode objectNode = JsonUtils.getObjectNode();
objectNode.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
objectNode.put("srctype",EnMsgType.EN_MSG_GETINFORMATION.toString());
objectNode.put("Nickname",information.getNickname());
objectNode.put("Signature",information.getSignature());
return objectNode.toString();}
// 批改昵称
private static String modifyNickname(ObjectNode jsonNodes) {int id = jsonNodes.get("id").asInt();
String nickname = jsonNodes.get("nickname").asText();
// 进行存储
informationDao.storeNickname(nickname,id);
return "";
}
// 批改签名
private static String modifySignature(ObjectNode jsonNodes) {int id = jsonNodes.get("id").asInt();
String signature = jsonNodes.get("signature").asText();
// 进行存储
informationDao.storeSignature(signature,id);
return "";
}
// 登录操作
private static String loginOperation(ObjectNode objectNode,Channel channel) {int id = objectNode.get("id").asInt();
String passwd = objectNode.get("passwd").asText();
// 进行数据库查问
boolean exits = userDao.getInformation(id, passwd);
ObjectNode jsonNodes = JsonUtils.getObjectNode();
jsonNodes.put("msgtype",EnMsgType.EN_MSG_ACK.toString());
jsonNodes.put("srctype",EnMsgType.EN_MSG_LOGIN.toString());
jsonNodes.put("code",300);
// 返回状态码
if (exits){jsonNodes.put("code",200);
// 增加用户的在线信息
CacheUtil.put(id,channel);
}
return jsonNodes.toString();}
}
2.ClientHandler.java
客户端承受来自服务端返回的音讯
依据返回的状态码来判断是否操作胜利
package chat.Project.netty;
import chat.Frame.chat.ChatFrame;
import chat.Frame.chat.linkmen;
import chat.Frame.chat.login;
import chat.Project.constant.EnMsgType;
import chat.util.JsonUtils;
import com.fasterxml.jackson.databind.node.ObjectNode;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import java.util.concurrent.SynchronousQueue;
public class ClientHandler extends SimpleChannelInboundHandler<String> {
// 定义一个同步阻塞队列状态码
public static SynchronousQueue<Object> queue = new SynchronousQueue<>();
public static String Nickname;
public String Signature;
@Override
protected void channelRead0(ChannelHandlerContext channelHandlerContext, String s) throws Exception { }
// 客户端接收数据
@Override
public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {System.out.println(msg);
// 解析服务端发送的音讯
ObjectNode jsonNodes = JsonUtils.getObjectNode((String) msg);
String msgtype = jsonNodes.get("msgtype").asText();
if (EnMsgType.EN_MSG_ACK.toString().equals(msgtype)) {String srctype = jsonNodes.get("srctype").asText();
if (EnMsgType.EN_MSG_LOGIN.toString().equals(srctype)) {
// 登录操作
queue.offer(jsonNodes.get("code").asInt());
}else if(EnMsgType.EN_MSG_GETINFORMATION.toString().equals(srctype)){
// 存取信息
Nickname = jsonNodes.get("Nickname").asText();
Signature = jsonNodes.get("Signature").asText();
linkmen.label_1.setText(Nickname);
linkmen.field.setText(Signature);
}else if (EnMsgType.EN_MSG_CHAT.toString().equals(srctype)){
// 发送端返回音讯
queue.offer(jsonNodes.get("code").asInt());
}else if (EnMsgType.EN_MSG_GET_ID.toString().equals(srctype)){int uid = jsonNodes.get("uid").asInt();
queue.offer(uid);
}else if (EnMsgType.EN_MSG_GET_FRIEND.toString().equals(srctype)){
// 获取登录用户的好友
int count = jsonNodes.get("count").asInt();
login.friend = new String[count];
for (int i = 0;i<count;i++){login.friend[i] = jsonNodes.get("res"+i).asText();
System.out.println(jsonNodes.get("res"+i));
}
}else if (EnMsgType.EN_MSG_ADD_FRIEND.toString().equals(srctype)){
// 增加好友
queue.offer(jsonNodes.get("code").asInt());
}else if (EnMsgType.EN_MSG_DEL_FRIEND.toString().equals(srctype)){
// 删除好友
queue.offer(jsonNodes.get("code").asInt());
}else if (EnMsgType.EN_MSG_ACTIVE_STATE.toString().equals(srctype)){
// 好友在线状态
queue.offer(jsonNodes.get("code").asInt());
}
}else if (EnMsgType.EN_MSG_VERIFY_PASSWORD.toString().equals(msgtype)){
// 批改明码
int code = 0;
code = jsonNodes.get("code").asInt();
queue.offer(code);
}else if (EnMsgType.EN_MSG_CHAT.toString().equals(msgtype)){
// 接收端承受音讯 封装敌人昵称
String message = ""+ jsonNodes.get("message").asText();
// 聊天显示框读取音讯
ChatFrame.sb.append(message+"n");
ChatFrame.displayTextPanel.setText(ChatFrame.sb.toString());
}
}
}
3.linkmen.java
这是登录胜利的界面
package chat.Frame.chat;
import chat.Frame.operation.alterColumn.changeNickname;
import chat.Frame.operation.alterColumn.changePassword;
import chat.Frame.operation.alterColumn.changeSignature;
import chat.Frame.operation.friendHandle.addFriend;
import chat.Frame.operation.friendHandle.delFriend;
import chat.Frame.tipFrame;
import chat.Project.services.sendServers;
import io.netty.channel.Channel;
import javax.swing.*;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;
import java.awt.*;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;
/**
* 联系人界面
*/
public class linkmen extends JFrame {
// 容器
private JFrame frame;
// 标签
private JLabel label_2, label_3, label_4, label;
// 昵称
public static JLabel label_1;
// 状态框
private JComboBox box, box_1, box_2;
// 图片
private ImageIcon icon_1, icon;
// 文本
private JTextField field_1;
// 共性签名
public static JTextField field;
// 面板
private JPanel panel_1, panel_3, panel;
// 滚动面板
public JScrollPane panel_2;
// 列表
public static JList list;
// 与服务端通信的通道
private Channel channel;
// 用户的 id
private Integer id;
// 暂存 oldPasswd
public static JLabel label_5,label_6;
// 好友列表数组
private String[] fd;
// 列表
public static DefaultListModel<String> model;
public linkmen(Integer id, Channel channel,String[] fd) {
this.id = id;
this.channel = channel;
this.fd = fd;
}
public void init() {
// 初始化面板 1 并设置信息
panel_1 = new JPanel();
panel_1.setLayout(null);
panel_1.setLocation(0, 0);
panel_1.setBorder(BorderFactory.createTitledBorder("资料卡"));
panel_1.setSize(new Dimension(295, 148));
panel_1.setOpaque(false);
// 初始化面板 3 并设置信息
panel_3 = new JPanel();
panel_3.setLayout(null);
panel_3.setBorder(BorderFactory.createTitledBorder("零碎设置"));
panel_3.setLocation(0, 617);
panel_3.setSize(new Dimension(295, 55));
panel_3.setOpaque(false);
// 设置头像标签
label_2 = new JLabel(new ImageIcon("E: 聊天软件 untitledsrcimageSource4.png"));
label_2.setCursor(Cursor.getPredefinedCursor(Cursor.HAND_CURSOR));
label_2.setBounds(15, 15, 100, 100);
panel_1.add(label_2);
// 初始暂存标签
label_5 = new JLabel();
label_6 = new JLabel();
// 设置昵称标签
label_1 = new JLabel("");
label_1.setBounds(130, 10, 100, 30);
label_1.setFont(new Font("宋体", Font.PLAIN, 18));
panel_1.add(label_1);
list = new JList<String>(model);
// 设置每个列表的高
list.setFixedCellHeight(20);
list.setSelectionBackground(new Color(0xD8FF2F));
list.addListSelectionListener(new ListSelectionListener() {
@Override
public void valueChanged(ListSelectionEvent e) {
// 关上一个聊天窗口
if (e.getValueIsAdjusting()) {for (int i = 0; i < model.size(); i++) {if (model.get(i).equals(list.getSelectedValue())){
// 获取 id 有谬误
int ids = new sendServers(channel).getId((String) list.getSelectedValue());
if (ids!=0) {new sendServers(channel).friendIsActive(ids);
new ChatFrame(ids, channel).setVisible(true);
}else{System.out.println("好友不存在");
}
}
}
}
}
});
// 初始化面板二
panel_2 = new JScrollPane(list);
panel_2.setBorder(BorderFactory.createTitledBorder("联系人"));
panel_2.setLocation(0, 147);
panel_2.setSize(new Dimension(295, 470));
panel_2.getViewport().setOpaque(false);
list.setOpaque(false);
panel_2.setOpaque(false);
// 设置在线状态 bBox();
box = new JComboBox();
box.addItem("✅在线");
box.addItem("uD83DuDCBF 隐身");
box.addItem("uD83DuDCBB 繁忙");
box.addItem("❎离线");
box.setBounds(200, 10, 70, 30);
panel_1.add(box);
// 设置共性签名的标签
label_4 = new JLabel("共性签名:");
label_4.setFont(new Font("宋体", Font.PLAIN, 16));
label_4.setForeground(Color.BLUE);
label_4.setBounds(120, 50, 100, 20);
panel_1.add(label_4);
// 设置文本
field = new JTextField("");
field.setBounds(120, 80, 160, 30);
panel_1.add(field);
label_3 = new JLabel("uD83DuDD0D");
label_3.setForeground(Color.RED);
label_3.setBounds(10, 122, 20, 20);
panel_1.add(label_3);
// 设置搜寻栏
field_1 = new JTextField();
field_1.setBounds(30, 120, 250, 25);
panel_1.add(field_1);
// 对面板三进行初始化
box_1 = new JComboBox();
box_1.addItem("uD83DuDD12uD83DuDD28uD83DuDD13");
box_1.addItem("批改明码");
box_1.addItem("批改昵称");
box_1.addItem("批改签名");
box_1.setBounds(8, 20, 100, 25);
panel_3.add(box_1);
box_1.addItemListener(new ItemListener() {
@Override
public void itemStateChanged(ItemEvent e) {if ("批改签名".equals(box_1.getSelectedItem())) {
// 执行一次
if (e.getStateChange() == ItemEvent.SELECTED) {changeSignature changeSignature = new changeSignature(linkmen.this);
changeSignature.setVisible(true);
field.setText(changeSignature.jTextField.getText());
String signature = field.getText();
// 存储签名的办法
new sendServers(channel).modifySignature(signature, id);
}
}
if ("批改明码".equals(box_1.getSelectedItem())) {if (e.getStateChange() == ItemEvent.SELECTED) {changePassword changePassword = new changePassword(linkmen.this);
changePassword.setVisible(true);
label_5.setText(changePassword.oldPassword.getText());
String oldPasswd = label_5.getText();
label_6.setText(new String(changePassword.newPassword.getPassword()));
String newPasswd = label_6.getText();
// 进行验证
new sendServers(channel).verifyPasswd(oldPasswd, id,newPasswd);
}
}
if ("批改昵称".equals(box_1.getSelectedItem())) {if (e.getStateChange() == ItemEvent.SELECTED) {changeNickname changeNickname = new changeNickname(linkmen.this);
changeNickname.setVisible(true);
label_1.setText(changeNickname.jTextField.getText());
String nickname = label_1.getText();
// 存储昵称
new sendServers(channel).modifyNickname(nickname, id);
}
}
}
});
// 增加好友、删除好友
box_2 = new JComboBox();
box_2.addItem("uD83DuDC65");
box_2.addItem("增加好友");
box_2.addItem("删除好友");
box_2.setBounds(170, 20, 100, 25);
box_2.addItemListener(new ItemListener() {
@Override
public void itemStateChanged(ItemEvent e) {if ("增加好友".equals(box_2.getSelectedItem())) {if (e.getStateChange() == ItemEvent.SELECTED) {addFriend addFriend = new addFriend(linkmen.this);
addFriend.setVisible(true);
// 读取要搜寻的 ID
String friendIds = addFriend.jTextField.getText();
// 判断是否是字符串
if (judgeDigit(friendIds)){int friendId = Integer.parseInt(friendIds);
// 搜寻数据库
new sendServers(channel).addFriendOperate(friendId,id,label_1.getText());
}else {new tipFrame().init("输出参数谬误");
}
}
}
if ("删除好友".equals(box_2.getSelectedItem())) {if (e.getStateChange() == ItemEvent.SELECTED) {delFriend delFriend = new delFriend(linkmen.this);
delFriend.setVisible(true);
// 对其数据库进行删除操作
String friendIds = delFriend.TextField.getText();
// 判断是否是字符串
if(judgeDigit(friendIds)){int friendId = Integer.parseInt(friendIds);
// 操作数据库
new sendServers(channel).delFriendOperate(friendId,id,label_1.getText());
}else{new tipFrame().init("输出参数谬误");
}
}
}
}
});
panel_3.add(box_2);
// 设置 frame 信息
frame = new JFrame();
// 设置窗体信息
frame.setTitle("腾讯 QQ");
// 给窗体设置图片
icon_1 = new ImageIcon("E: 聊天软件 untitledsrcimageSource3.png");
frame.setIconImage(icon_1.getImage());
icon = new ImageIcon("E: 聊天软件 untitledsrcimageSource5.png");
label = new JLabel(icon);
// 获取窗口的第二层,将 label 放入
frame.getLayeredPane().add(label, new Integer(Integer.MIN_VALUE));
// 获取 frame 的顶层容器, 并设置为通明
panel = (JPanel) frame.getContentPane();
panel.setOpaque(false);
frame.setLayout(null);
frame.setLocation(750, 150);
frame.setSize(287, 700);
frame.setVisible(true);
frame.setResizable(false);
label.setBounds(0, 0, 287, 700);
frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
frame.add(panel_1);
frame.add(panel_2);
frame.add(panel_3);
}
public void mian() {
// 初始化面板 2 并设置信息
model = new DefaultListModel<>();
for (int i = 0; i < fd.length; i++) {model.addElement(fd[i]);
}
init();
// 更新昵称和签名
new sendServers(channel).update(id);
// 获取用户的昵称, 和好友列表
// 设置签名和昵称字体初始款式和大小
label_1.setFont(new Font("宋体", Font.PLAIN, 18));
field.setFont(new Font("宋体", Font.PLAIN, 18));
}
// 判断是否是数字
private static boolean judgeDigit(String string){for (int i = 0; i < string.length(); i++) {if (!Character.isDigit(string.charAt(i))){return false;}
}
return true;
}
}
4.tipFrame
提醒操作状态窗口
package chat.Frame;
import chat.Frame.chat.linkmen;
import chat.Frame.operation.alterColumn.changeNickname;
import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
public class tipFrame extends JDialog {
private Container container;
// 显示错误信息
public JLabel label;
// 确认按钮
private JButton button;
public tipFrame(){}
public void init(String msg){container = getContentPane();
label = new JLabel(msg);
label.setBounds(70,0,200,70);
label.setFont(new Font("微软雅黑",Font.PLAIN,20));
container.add(label);
button = new JButton("确认");
button.setBounds(35,50,140,40);
container.add(button);
setBounds(780,170,220,140);
setLayout(null);
setVisible(true);
container.setBackground(new Color(0xD8FFD5));
// 提醒窗口前置
setAlwaysOnTop(true);
button.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {tipFrame.this.dispose();
}
});
}
}
==
5. 运行例图
1. 登录界面
注册账号和遗记明码没有增加事件当初就是个陈设
—
2. 联系人界面
这外面的所有性能都能够应用
—
3. 聊天界面
这个外面表情按钮没弄好
4. 通信的过程
5. 批改操作
6. 好友的操作
我的项目举荐:
2000 多 G 的计算机各行业电子资源分享(继续更新)
2020 年微信小程序全栈我的项目之喵喵交友【附课件和源码】
Spring Boot 开发小而美的集体博客【附课件和源码】
Java 微服务实战 296 集大型视频 - 谷粒商城【附代码和课件】
Java 开发微服务畅购商城实战【全 357 集大我的项目】- 附代码和课件
最全最具体数据结构与算法视频 -【附课件和源码】
正文完