• 表白墙/留言墙 —— 中级SpringBoot项目,MyBatis技术栈MySQL数据库开发,练手项目前后端开发(带完整源码) 全方位全步骤手把手教学


    🧸欢迎来到dream_ready的博客,📜相信你对这篇博客也感兴趣o (ˉ▽ˉ;)

    📜表白墙/留言墙初级Spring Boot项目(此篇博客的简略版,不带MyBatis数据库开发)

    目录

    1、项目前端页面及项目文件架构展示

    1.1、项目前端页面展示:

    1.2、项目文件架构展示:

    2、首先,定义前后端交互接口

    3、然后,创建Spring Boot项目,导入下列依赖

    4、编写前端页面

    5、配置并连接数据库(数据库相关工作)

    5.1、数据库分析及建库建表语句

    5.2、连接数据库(yml文件)并配置相关配置

    6、编写后端代码

    6.1、数据库实体类(Model)

    6.2、Controller —— 控制层

    方法publishMessage处理逻辑:

    方法getMessageInfo处理逻辑:

    6.3、Service —— 业务逻辑层

    6.4、Dao(此处命名为Mapper) —— 持久层

    方法insertMessage处理逻辑:

    方法selectAllMessage处理逻辑:

    7、全部代码(超级全,含建库建表语句)含搭建教程:

    依赖

    建表:

    yml配置文件书写

    实体类:

    Controller:

    Service:

    Dao(此处是Mapper)

    前端:


    1、项目前端页面及项目文件架构展示

    1.1、项目前端页面展示:

    1.2、项目文件架构展示:

    此处用到了后端极其常见的三层架构

    详细信息请看下面这个博客,其实如果是初学者的话,先简单按着SpringMVC的理解来也行,但两者是有区别的:

    什么是SpringMVC?简单好理解!什么是应用分层?SpringMVC与应用分层的关系? 什么是三层架构?SpringMVC与三层架构的关系?

    2、首先,定义前后端交互接口

    讲解:

    1.         此项目注重前后端信息的交互,舍去了登录的操作
    2.         提交留言 —— 点击提交后,前端从输入框获取三个参数的值,向后端发送这MessageInfo的三个参数的值,后端接收后保存到MySQL数据库中
    3.         查看所有留言 —— 前端发送无参的请求,后端查询数据库,将查询后的数据封装到List集合中,前端接收数据后将其展示到页面上

    3、然后,创建Spring Boot项目,导入下列依赖

    如何在idea中创建Springboot项目? 手把手带你创建Springboot项目,稳!

    导入下列依赖

    4、编写前端页面

            在resource目录下的static目录下创建表白墙的html页面,此处我将其命名为 messagewall.html

    如图红框位置:

    表白墙页面(messagewall.html):

    1. html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
    6. <title>留言板title>
    7. <style>
    8. .container {
    9. width: 350px;
    10. height: 300px;
    11. margin: 0 auto;
    12. /* border: 1px black solid; */
    13. text-align: center;
    14. }
    15. .grey {
    16. color: grey;
    17. }
    18. .container .row {
    19. width: 350px;
    20. height: 40px;
    21. display: flex;
    22. justify-content: space-between;
    23. align-items: center;
    24. }
    25. .container .row input {
    26. width: 260px;
    27. height: 30px;
    28. }
    29. #submit {
    30. width: 350px;
    31. height: 40px;
    32. background-color: orange;
    33. color: white;
    34. border: none;
    35. margin: 10px;
    36. border-radius: 5px;
    37. font-size: 20px;
    38. }
    39. style>
    40. head>
    41. <body>
    42. <div class="container">
    43. <h1>留言板h1>
    44. <p class="grey">输入后点击提交, 会将信息显示下方空白处p>
    45. <div class="row">
    46. <span>谁:span> <input type="text" name="" id="from">
    47. div>
    48. <div class="row">
    49. <span>对谁:span> <input type="text" name="" id="to">
    50. div>
    51. <div class="row">
    52. <span>说什么:span> <input type="text" name="" id="say">
    53. div>
    54. <input type="button" value="提交" id="submit" onclick="submit()">
    55. div>
    56. <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.4/jquery.min.js">script>
    57. <script>
    58. // 页面加载时,请求后端,获取留言列表
    59. $.ajax({
    60. url: "/message/getMessageInfo",
    61. type: "get",
    62. success:function (message){
    63. for(var m of message){
    64. // 2. 拼接节点的 html
    65. var divE = "
      "+ m.from +"对" + m.to + "说:" + m.message+"
      "
      ;
    66. //3. 把节点添加到页面上
    67. $(".container").append(divE);
    68. }
    69. }
    70. });
    71. function submit(){
    72. //1. 获取留言的内容
    73. var from = $('#from').val();
    74. var to = $('#to').val();
    75. var say = $('#say').val();
    76. if (from== '' || to == '' || say == '') {
    77. return;
    78. }
    79. // 发送请求
    80. $.ajax({
    81. url: "/message/publish",
    82. type: "post",
    83. data: {
    84. "from": from,
    85. "to": to,
    86. "message": say
    87. },
    88. success: function(result){
    89. if(result){
    90. // 添加成功
    91. // 2. 拼接节点的 html
    92. var divE = "
      "+from +"对" + to + "说:" + say+"
      "
      ;
    93. //3. 把节点添加到页面上
    94. $(".container").append(divE);
    95. //4. 清空输入框的值
    96. $('#from').val("");
    97. $('#to').val("");
    98. $('#say').val("");
    99. }else{
    100. // 添加失败
    101. alert("留言发布成功")
    102. }
    103. }
    104. });
    105. }
    106. script>
    107. body>
    108. html>

    代码逻辑:

    1.         从input输入框内获取from,to,message三个参数,向后端发送post请求并将参数发送过去,后端接收参数并保存,前端页面也将这段数据直接展示在页面上
    2.         每次刷新页面,前端向后端发送get请求,后端响应回封装成List集合的所有数据,前端遍历集合,并将其展示在前端页面上

    5、配置并连接数据库(数据库相关工作)

    5.1、数据库分析及建库建表语句

    建库建表语句:

    库在此处就不指定了,放你自己常用的库下或者新建都行

    1. DROP TABLE IF EXISTS message_info;
    2. CREATE TABLE `message_info` (
    3. `id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
    4. `from` VARCHAR ( 127 ) NOT NULL,
    5. `to` VARCHAR ( 127 ) NOT NULL,
    6. `message` VARCHAR ( 256 ) NOT NULL,
    7. `delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
    8. `create_time` DATETIME DEFAULT now(),
    9. `update_time` DATETIME DEFAULT now() ON UPDATE now(),
    10. PRIMARY KEY ( `id` )
    11. ) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

    数据库分析:

    你也可以输入   desc userinfo;    命令来在你的客户端查看建表的各字段信息,和我上面这张图表达的意思一样

    解析如下:

    • id  整数类型的自增字段  作为主键
    • from  字符串类型   用于存储消息发送者信息
    • to    字符串类型     用于存储消息接收者信息
    • message   字符串类型   用于存储两者间发送的消息内容
    • delete_flag   小整数类型(0或1) 此处默认设置为0    用于标记信息是否被删除(0表示正常,1表示删除)
    • create_time    DATETIME 类型  用于存储消息创建时间    默认为当前时间
    • update_time   DATETIME 类型  用于存储消息更新时间    默认为当前时间,并且会在更新时自动更新为当前时间

    字符集采用utf8mb4,自增主键是id

    默认为...即若不主动设置,则为默认值

    5.2、连接数据库(yml文件)并配置相关配置

    你的application文件后缀大概率是  .properties  ,将其后缀改为 .yml   即可

    yml配置如下: 

    1. spring:
    2. datasource:
    3. url: jdbc:mysql://127.0.0.1:3306/message?characterEncoding=utf8&useSSL=false
    4. username: root
    5. password: 123456
    6. driver-class-name: com.mysql.cj.jdbc.Driver
    7. mybatis:
    8. configuration: # 配置打印 MyBatis 日志
    9. log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
    10. map-underscore-to-camel-case: true #配置驼峰⾃动转换
    11. # 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件 classpath指的是resources
    12. mapper-locations: classpath:mapper/**Mapper.xml

    mybatis下面不懂得还好,我都给注释了!如果是前几行不懂的话,那完蛋,你最好先稍微学习下mybatis,前六行的作用是连接数据库

    第三行的ip地址、端口号、数据库库名以及第四行和第五行的信息根据自己实际情况修改!!! 

    6、编写后端代码

    yml配置在上面

    6.1、数据库实体类(Model)

    为实现分层架构,定义Model层,其实Model层也算Dao层的一部分,然后将对应的实体类命名为MessageInfo

    这其中的各个属性对应数据库中的各个字段

    以下是全部代码:

    1. import lombok.Data;
    2. import java.util.Date;
    3. @Data // 组合注解,集成了@Getter @Setter @ToString 等注解
    4. public class MessageInfo {
    5. private Integer id;
    6. private String from;
    7. private String to;
    8. private String message;
    9. private Date createTime;
    10. private Date updateTime;
    11. }

    6.2、Controller —— 控制层

    注意,Controller层、Serbice层、Dao层密切相关,所以Controller代码里就行Service和Dao层的部分,不要慌,我的代码是绝对完整正确的!,跟着我的讲解走并且理解就行!

    接收前端发送的请求,对请求进行处理,并响应数据

    全部代码如下:

    1. import com.example.messagewall_mybatis.model.MessageInfo;
    2. import com.example.messagewall_mybatis.service.MessageService;
    3. import lombok.extern.slf4j.Slf4j;
    4. import org.springframework.beans.factory.annotation.Autowired;
    5. import org.springframework.util.StringUtils;
    6. import org.springframework.web.bind.annotation.RequestMapping;
    7. import org.springframework.web.bind.annotation.RestController;
    8. import java.util.List;
    9. @Slf4j
    10. @RequestMapping("/message")
    11. @RestController
    12. public class MessageController {
    13. @Autowired
    14. private MessageService messageService;
    15. @RequestMapping("/publish")
    16. public Boolean publishMessage(MessageInfo messageInfo){
    17. log.info("发表留言");
    18. // 进行参数的校验
    19. if(!StringUtils.hasLength(messageInfo.getFrom())
    20. || !StringUtils.hasLength(messageInfo.getTo())
    21. || !StringUtils.hasLength(messageInfo.getMessage())){
    22. return false;
    23. }
    24. // 添加留言
    25. messageService.addMessage(messageInfo);
    26. return true;
    27. }
    28. @RequestMapping("/getMessageInfo")
    29. public List getMessageInfo(){
    30. return messageService.getMessageInfo();
    31. }
    32. }

    下面这个注入了MessageService,该类里面都是对数据库进行增删改查的方法!

    方法publishMessage处理逻辑:

    1. 接收前端传来的数据,自动将其封装为MessageInfo类型
    2. 进行参数的校验,判断三个参数是否有空,若有空,返回false(失败)
    3. 若参数正确,则添加留言,调用Service中的addMessage方法,将留言添加到数据库中,并返回true

    方法getMessageInfo处理逻辑:

    • 调用Service中的getMessageInfo方法,从数据库中查询数据,并返回给前端

    6.3、Service —— 业务逻辑层

    处理具体的业务逻辑

    全部代码如下:

    1. import com.example.messagewall_mybatis.mapper.MessageMapper;
    2. import com.example.messagewall_mybatis.model.MessageInfo;
    3. import org.springframework.beans.factory.annotation.Autowired;
    4. import org.springframework.stereotype.Service;
    5. import java.util.List;
    6. @Service
    7. public class MessageService {
    8. @Autowired
    9. private MessageMapper messageMapper;
    10. public void addMessage(MessageInfo messageInfo) {
    11. messageMapper.insertMessage(messageInfo);
    12. }
    13. public List getMessageInfo() {
    14. return messageMapper.selectAllMessage();
    15. }
    16. }

    下面这个注入了MessageService,该接口里面都是实现对数据库进行增删改查的具体操作!

    两个方法的定义逻辑:

    • Service层属于声明和调用,对数据库的具体操作在Mapper中
    • 可以看到,addMessage和getMessageInfo实际都是再次调用Mapper中的操作
    • addMessage 表示向数据库中添加信息
    • getMessageInfo 表示从数据库中查询所有的信息

    6.4、Dao(此处命名为Mapper) —— 持久层

    负责数据访问操作,包括数据的增、删、改、查

    全部代码如下:

    1. import com.example.messagewall_mybatis.model.MessageInfo;
    2. import org.apache.ibatis.annotations.Insert;
    3. import org.apache.ibatis.annotations.Mapper;
    4. import org.apache.ibatis.annotations.Select;
    5. import java.util.List;
    6. @Mapper
    7. public interface MessageMapper {
    8. @Insert("insert into message_info(`from`,`to`,`message`) values (#{from},#{to},#{message})")
    9. public void insertMessage(MessageInfo messageInfo);
    10. @Select("select * from message_info where delete_flag=0")
    11. List selectAllMessage();
    12. }

    此处因为SQL语句过于简单,所以我选择了使用注解,当然,你使用XML当然也可以,按你的写作风格就行,但若你是初学者又刚好了解注解,那么按着我的来更方便一些!

    方法insertMessage处理逻辑:

    定义Insert注解,将messageInfo实体类中的from、to、message三个属性的值赋值给mossage_info表中对应的三个属性

    方法selectAllMessage处理逻辑:

    定义Select注解,查询message_info表中未被逻辑删除的数据,并将其放到集合中

    至此,本项目就讲解完成了!

    前路漫漫,数年磨剑只求一朝天下知!希望本篇博客给您进了些绵薄之力,感谢您的阅读!

    7、全部代码(超级全,含建库建表语句)含搭建教程:

    友情提示:若您是初学者,建议慢慢阅读博客,跟着步骤一步一步搭建项目,代码也是绝对正确的!

    但同时也为时间紧迫者以及掌握了一些知识的人提供了快捷方案,直接把所有代码放在下面,并配上了简单的搭建教程(详细的在上面)

    依赖

    建表:

    库自己指定就行

    1. DROP TABLE IF EXISTS message_info;
    2. CREATE TABLE `message_info` (
    3. `id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
    4. `from` VARCHAR ( 127 ) NOT NULL,
    5. `to` VARCHAR ( 127 ) NOT NULL,
    6. `message` VARCHAR ( 256 ) NOT NULL,
    7. `delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
    8. `create_time` DATETIME DEFAULT now(),
    9. `update_time` DATETIME DEFAULT now() ON UPDATE now(),
    10. PRIMARY KEY ( `id` )
    11. ) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

    yml配置文件书写

    1. spring:
    2. datasource:
    3. url: jdbc:mysql://127.0.0.1:3306/message?characterEncoding=utf8&useSSL=false
    4. username: root
    5. password: 123456
    6. driver-class-name: com.mysql.cj.jdbc.Driver
    7. mybatis:
    8. configuration: # 配置打印 MyBatis 日志
    9. log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
    10. map-underscore-to-camel-case: true #配置驼峰⾃动转换
    11. # 配置 mybatis xml 的⽂件路径,在 resources/mapper 创建所有表的 xml ⽂件 classpath指的是resources
    12. mapper-locations: classpath:mapper/**Mapper.xml

    实体类:

    1. import lombok.Data;
    2. import java.util.Date;
    3. @Data // 组合注解,集成了@Getter @Setter @ToString 等注解
    4. public class MessageInfo {
    5. private Integer id;
    6. private String from;
    7. private String to;
    8. private String message;
    9. private Date createTime;
    10. private Date updateTime;
    11. }

    Controller:

    1. import com.example.messagewall_mybatis.model.MessageInfo;
    2. import com.example.messagewall_mybatis.service.MessageService;
    3. import lombok.extern.slf4j.Slf4j;
    4. import org.springframework.beans.factory.annotation.Autowired;
    5. import org.springframework.util.StringUtils;
    6. import org.springframework.web.bind.annotation.RequestMapping;
    7. import org.springframework.web.bind.annotation.RestController;
    8. import java.util.List;
    9. @Slf4j
    10. @RequestMapping("/message")
    11. @RestController
    12. public class MessageController {
    13. @Autowired
    14. private MessageService messageService;
    15. @RequestMapping("/publish")
    16. public Boolean publishMessage(MessageInfo messageInfo){
    17. log.info("发表留言");
    18. // 进行参数的校验
    19. if(!StringUtils.hasLength(messageInfo.getFrom())
    20. || !StringUtils.hasLength(messageInfo.getTo())
    21. || !StringUtils.hasLength(messageInfo.getMessage())){
    22. return false;
    23. }
    24. // 添加留言
    25. messageService.addMessage(messageInfo);
    26. return true;
    27. }
    28. @RequestMapping("/getMessageInfo")
    29. public List getMessageInfo(){
    30. return messageService.getMessageInfo();
    31. }
    32. }

    Service:

    1. import com.example.messagewall_mybatis.mapper.MessageMapper;
    2. import com.example.messagewall_mybatis.model.MessageInfo;
    3. import org.springframework.beans.factory.annotation.Autowired;
    4. import org.springframework.stereotype.Service;
    5. import java.util.List;
    6. @Service
    7. public class MessageService {
    8. @Autowired
    9. private MessageMapper messageMapper;
    10. public void addMessage(MessageInfo messageInfo) {
    11. messageMapper.insertMessage(messageInfo);
    12. }
    13. public List getMessageInfo() {
    14. return messageMapper.selectAllMessage();
    15. }
    16. }

    Dao(此处是Mapper)

    1. import com.example.messagewall_mybatis.model.MessageInfo;
    2. import org.apache.ibatis.annotations.Insert;
    3. import org.apache.ibatis.annotations.Mapper;
    4. import org.apache.ibatis.annotations.Select;
    5. import java.util.List;
    6. @Mapper
    7. public interface MessageMapper {
    8. @Insert("insert into message_info(`from`,`to`,`message`) values (#{from},#{to},#{message})")
    9. public void insertMessage(MessageInfo messageInfo);
    10. @Select("select * from message_info where delete_flag=0")
    11. List selectAllMessage();
    12. }

    前端:

    1. html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
    6. <title>留言板title>
    7. <style>
    8. .container {
    9. width: 350px;
    10. height: 300px;
    11. margin: 0 auto;
    12. /* border: 1px black solid; */
    13. text-align: center;
    14. }
    15. .grey {
    16. color: grey;
    17. }
    18. .container .row {
    19. width: 350px;
    20. height: 40px;
    21. display: flex;
    22. justify-content: space-between;
    23. align-items: center;
    24. }
    25. .container .row input {
    26. width: 260px;
    27. height: 30px;
    28. }
    29. #submit {
    30. width: 350px;
    31. height: 40px;
    32. background-color: orange;
    33. color: white;
    34. border: none;
    35. margin: 10px;
    36. border-radius: 5px;
    37. font-size: 20px;
    38. }
    39. style>
    40. head>
    41. <body>
    42. <div class="container">
    43. <h1>留言板h1>
    44. <p class="grey">输入后点击提交, 会将信息显示下方空白处p>
    45. <div class="row">
    46. <span>谁:span> <input type="text" name="" id="from">
    47. div>
    48. <div class="row">
    49. <span>对谁:span> <input type="text" name="" id="to">
    50. div>
    51. <div class="row">
    52. <span>说什么:span> <input type="text" name="" id="say">
    53. div>
    54. <input type="button" value="提交" id="submit" onclick="submit()">
    55. div>
    56. <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.4/jquery.min.js">script>
    57. <script>
    58. // 页面加载时,请求后端,获取留言列表
    59. $.ajax({
    60. url: "/message/getMessageInfo",
    61. type: "get",
    62. success:function (message){
    63. for(var m of message){
    64. // 2. 拼接节点的 html
    65. var divE = "
      "+ m.from +"对" + m.to + "说:" + m.message+"
      "
      ;
    66. //3. 把节点添加到页面上
    67. $(".container").append(divE);
    68. }
    69. }
    70. });
    71. function submit(){
    72. //1. 获取留言的内容
    73. var from = $('#from').val();
    74. var to = $('#to').val();
    75. var say = $('#say').val();
    76. if (from== '' || to == '' || say == '') {
    77. return;
    78. }
    79. // 发送请求
    80. $.ajax({
    81. url: "/message/publish",
    82. type: "post",
    83. data: {
    84. "from": from,
    85. "to": to,
    86. "message": say
    87. },
    88. success: function(result){
    89. if(result){
    90. // 添加成功
    91. // 2. 拼接节点的 html
    92. var divE = "
      "+from +"对" + to + "说:" + say+"
      "
      ;
    93. //3. 把节点添加到页面上
    94. $(".container").append(divE);
    95. //4. 清空输入框的值
    96. $('#from').val("");
    97. $('#to').val("");
    98. $('#say').val("");
    99. }else{
    100. // 添加失败
    101. alert("留言发布成功")
    102. }
    103. }
    104. });
    105. }
    106. script>
    107. body>
    108. html>

    🧸欢迎您于百忙之中阅读这篇博客,📜希望这篇博客给您带来了一些帮助,祝您生活愉快!

  • 相关阅读:
    Android选项卡TabHost
    模板_整数二分
    【论文阅读】BGE Landmark Embedding: 一种用于大语言模型长上下文检索增强的嵌入方法
    Could not load library libcudnn_cnn_train.so.8, 解决类似问题的思路与方法
    2022牛客多校联赛第七场 题解
    IP摄像头实现远程目标检测
    关于libxml的使用
    FlinkSQL-UDF自定义数据源
    2022年下半年软件设计师下午真题及答案解析
    python xml 解析
  • 原文地址:https://blog.csdn.net/dream_ready/article/details/134470416