头条博文 (SSM)(下)

简介: 头条博文 (SSM)(下)

五、实现后端的 Service 层,Contoller 层



Blog 对应的是博客表,关于 Blog 的类,都是操作博客表。


客户端 => BlogController => BlogService => BlogMapper => xml 文件 => 数据库


User 对应的是用户表,关于 User 的类,都是操作用户表。


客户端 => UserController => UserService => UserMapper => xml 文件 => 数据库


14d8a95e214549f5a20bb34f04a2dd64.png


六、编辑前后端代码,实现页面



1. 博客注册页


(1) 作用:博客注册页是用于实现用户注册的页面,它不直接出现在用户面前,而是通过点击登录页面的【注册账号】这个链接,来进行跳转。


(2) 约定 POST 请求 的路径:" /register"


(3) 前端代码:通过 form 表单进行 HTTP 请求的提交,在提交的过程中,需要带上 【username】 、【password1】、【password2】这三个参数,以便于服务器端进行验证。


(4) 单独准备一个 UserVerify 实体类,用于接收前端的这三个参数。


@Data
public class UserVerify {
    private String username;
    private String password1;
    private String password2;
}


(5) 服务器端代码:创建一个 register 方法放在 UserController 类中,来实现 HTTP 响应,并往数据库的 user 表中插入新用户。


首先我们应该做的,就是进行判空操作。


/**
 * 1. 注册用户
 */
@ResponseBody
@RequestMapping("/register")
public String register(UserVerify userVerify) {
    // 判空操作
    if (!StringUtils.hasLength(userVerify.getUsername())) {
        return "<h3> 您未输入用户名 </h3>" + "<a href='javascript:history.go(-1);'>返回</a>";
    }
    if (!StringUtils.hasLength(userVerify.getPassword1())) {
        return "<h3> 您未输入密码 </h3>" + "<a href='javascript:history.go(-1);'>返回</a>";
    }
    ......
    ......
    ......
}


(6) 注意事项:

① 注册的用户名与数据库中的用户名不能重合。

② 重复输入一次密码,以此保障注册的密码不会失败。

③ 注册完成后,应该给用户一个提示,是否需要进行跳转登录页。


展示效果:


f81f613e505840298f4972bd915d5751.png


2. 博客列表页


(1) 作用:博客列表页主要用来展示所有博客的摘要


(2) 约定 GET 请求 的路径:" /blog_list"


(3) 前端代码:通过 ajax 来构造请求,思想:先按照纯前端代码创建相应的节点,之后再挂在 DOM 树上。


(4) 服务器端代码:创建一个 displayBlogs 方法放在 BlogController 类中,来实现 HTTP 响应,只要用户登录成功后,就会展示博客列表页。


(5) 由于在博客列表页,我们只是展示所有博客的摘要,所以在服务层中,通过字符串截取的方式,来控制页面为用户展示的博客字数。这样一来,控制层在返回数据的时候,只是一个简化的 json 数据了。


@Service
public class BlogService {
    @Resource
    private BlogMapper blogMapper;
    // 1. 查询所有博客
    // 在这里的逻辑,我们将展示的摘要进行了简化
    public List<Blog> getAllBlogs(){
        List<Blog> blogs1 = blogMapper.getAllBlogs();
        List<Blog> blogs2 = new ArrayList<>();
        for (int i = 0; i < blogs1.toArray().length; i++) {
            Blog blog = new Blog();
            blog.setBlogID(blogs1.get(i).getBlogID());
            blog.setTitle(blogs1.get(i).getTitle());
            // 由于这里显示的是博客内容的摘要,所以,
            // 我们约定: 当字符的数量大于 10个的时候,我们通过截取字符串的形式,来放入 blog 对象中
            String content = blogs1.get(i).getContent();
            if (content.length() > 10) {
                content = content.substring(0, 10) + ".......";
            }
            blog.setContent( content );
            blog.setPostTime(blogs1.get(i).getPostTime());
            blog.setUserID(blogs1.get(i).getUserID());
            blogs2.add(blog);
        }
        return blogs2;
    }
}


抓包结果:( HTTP 响应的正文 )


1f7385647d8f4281b26592273168d8a8.png


(6) 由于页面的内容过长,这就可能导致溢出页面,或者说,内容溢出版心,此时,我们就可以通过设置 CSS 文件,并引入 HTML,来弥补这一缺点。


如下代码:当内容溢出页面的时候,就会自动设置滑动栏,这很实用。


overflow: auto;


展示效果:


image.png


3. 博客内容页


(1) 作用:博客内容页用来展示某个用户的某篇文章


(2) 约定 GET 请求 的路径:" /blog_content "


(3) 前端代码:通过 ajax 来构造请求,思想:先按照纯前端代码创建相应的节点,之后再挂在 DOM 树上。


(4) 服务器端代码:创建一个 displayUserBlog 方法放在 BlogController 类中,来实现 HTTP 响应。


(5) 博客列表页和博客内容页之间的配合:只要用户在博客列表页点击【查看全文】按钮的时候,就会跳转到内容页。


此功能在博客列表页中,通过 blogID 参数来实现指定跳转。在 ajax 中,我们利用 JS-WebAPI 来直接拼接节点。


// 跳转博客内容页的链接
let linkA = document.createElement('a');
linkA.innerHTML = '查看全文 &gt;&gt';
linkA.href = 'blog_content.html?blogID=' + blog.blogID;


展示效果:


8f163ce9399c404d9dfa56423ca8d0e7.png


4. 博客登录页


(1) 作用:博客登录页是用于实现用户登录的页面,它可以判断用户名和密码各自是否正确。


(2) 约定 POST 请求 的路径:" /login"

我们打开写死的博客登录页面,点击【登录】,浏览器自然就发送了 POST 请求,因为我们将【登录】放在了 form 表单下,通过 input 标签实现的。


(3) 前端代码:通过 form 表单进行 HTTP 请求的提交,在提交的过程中,需要带上 【username】 和【password】这两个参数,以便于服务器端进行验证。


(4) 服务器端代码:创建一个 login 方法放在 UserController 类中,来实现 HTTP 响应。只要用户登录成功后,就会跳转到博客列表页。


(5) 博客登录页这里,并不需要展示什么,所以,此页面也无需通过 ajax 的方式进行构造 HTTP 请求,直接通过 form 表单的形式提交请求,这会让整个登录流程变得更加简单。此外,这里的 if 语句进行判断,我们应该考虑到所有的意外情况与不合理的情况


(6) 登录页面的时候,我们可以利用 session 机制。


① 若登录成功,就将 session 会话创建出来,并将当前登录的用户,以 Java 对象的方式放入会话 session 中,后面再次使用博客系统的时候,服务器就能够通过会话中的对象,知晓当前登录者是哪个用户。


② 若登录失败,就不将 session 会话创建出来。我们也可以反过来思考:若 session 会话未被创建出来,那就意味着登录失败。


③ session 机制就和之前的 ServletContext 机制差不多,我们可以将其想象成一个冰箱,随拿随放。


image.png


(1) 判定登录与注销操作


鉴于上面的思想,我们不仅可以用当前的 session 会话机制判断博客列表页,也可以用它来判断博客内容页,博客发布页的用户登录情况。此外,通过会话机制,也能够应用于注销操作。


所以,我们对上面的代码改进一下,封装一个 Check 类,让上面所说的页面都能够通过这个 Check 类来判断当前用户是否登录了。


public class Check {
    public static User CheckLogin(HttpServletRequest req) {
        HttpSession httpSession = req.getSession(false);
        if (httpSession == null) {
            // 会话未创建出来,意味着用户未登录
            return null;
        }
        User loginUser = (User) httpSession.getAttribute("user");
        if (loginUser == null) {
            // 就算会话创建出来了,但是里面没有对象,也意味着用户未登录
            return null;
        }
        return loginUser;
    }
}


注意 if 语句的顺序,为什么先要判定 httpSession 存在与否呢?这是为了防止空指针异常。【 若 httpSession 为 null,那么,httpSession.getAttribute(“user”) 这行代码就会出现空指针异常 】


这很好理解:当冰箱都没有的时候,我们怎么从冰箱拿东西呢?所以说,一定是先有冰箱了,我们才能从里面拿东西,才能往里面放东西。


image.png


注意事项:


这里需要注意一件事情,当用户没有登录的时候,就访问了博客列表页或博客内容页,应该马上让 HTML 页面再次重定向至登录页面。而在当前的 ajax 方式,我们让前端去处理重定向操作了。这里并不建议在后端进行重定向操作,为什么呢?


因为,使用 ajax 这种方式构造 HTTP 请求的时候,服务器端只负责在 HTTP 响应中写入数据,比方说:它应该写入状态码、正文 body、body 格式等等…而前端就应该根据拿到的 HTTP 响应的数据,决定实现什么样的前端页面与样式,所以说,在这里,所有展现给用户看的内容和格式,都应该由前端负责。 我们在后端设置一个 403 这样的权限状态码,就能够告知 ajax 的 error 函数,进行重定向了。


然而,如果我们通过服务器端的重定向操作来实现也不是不可以,但会出现一定的问题,例如:重定向之后的页面,展示的效果可能不是一个 HTML 页面,而是一些未经处理的文本数据。也就是说,我们在方法上已经加上了一个 " @ResponseBody " 注解,当登录不成功的时候,我们需要返回一些数据。 实际上,后端返回 json 数据的场景居于大多数,真正处理用户视觉效果的,全部是前端。


① 正确的前后端判定代码


服务器端自定义写入状态码:

403 这个状态码表示的是访问被拒绝,页面通常需要用户具有一定的权限才能访问。


(403 Forbidden)


// 验证用户有没有登录
User loginUser = Check.CheckLogin(request);
if (loginUser == null) {
    System.out.println("当前用户未登录");
    // 若登录不成功,设置状态码为 403,前端的回调 error 函数就能够直接重定向页面了
    response.setStatus(403);
    return null;
}


image.png


前端重定向操作:


error: function() {
    // 此处表示前端形式的重定向, 相当于后端的 resp.sendRedirect()方法
    location.assign('blog_login.html');
}


(2) 登录用户与文章作者的数据信息


当用户登录成功后,首先跳转到的是博客列表页,那么,博客列表页就应该显示当前用户的一些信息:头像、昵称、文章总数…接着,若用户点击【查看全文】后,就可以跳转到某一篇博客全文,此时,页面显示的应该是作者信息。


鉴于此,


(1) 将博客列表页展示当前登录者的用户信息

(2) 将博客内容页展示文章作者的用户信息


这里就不再展示代码了,它的思想其实和上面的博客内容页的 ajax 思想是相同的,先选中 DOM 树的节点,再对其标签的内容进行更改。同样地,后端应该利用 blogID 这个参数和 session 会话机制,来确定谁是作者,谁是当前登录的用户。


展示效果:


image.png


5. 博客编辑页


(1) 作用:博客编辑页是实现让用户用来撰写博客的,它可以在浏览器上进行提交,而后,服务器经过一些处理,让博客的一些数据放入数据库中。


(2) 约定 POST 请求 的路径:" /blog_writing"

我们打开写死的博客编辑页面,点击【发布文章】,浏览器自然就发送了 POST 请求,因为我们将【发布文章】放在了 form 表单下,通过 input 标签实现的。


(3) 前端代码:通过 form 表单进行 HTTP 请求的提交,在提交的过程中,需要带上 【title】 和【content】这两个参数,以便于服务器端进行验证。

然而,这里的代码较为少见,因为当前是根据 jQuery 提供的依赖,才会有这个编辑页面的展示效果,所以,写死的 HTML页面同时需要配合 JS 代码,并且需要基于 【editor.md】的一些写法规则


(4) 服务器端代码:创建一个 blogWriting 方法放在 BlogController 类中,来实现 HTTP 响应,并传入博客到数据库中。由于前端只传来 " 标题 " 和 " 正文 " 这两个参数,所以,userID 需要我们通过获取登录者的信息才能拿到。


(5) 博客编辑页这里,和博客登录页是一样的思路,并不需要展示什么,所以,此页面并不需要基于 ajax 构造请求。此外,这里的 if 语句进行判断,我们应该考虑到所有的意外情况与不合理的情况。


(6) 由于编辑博客的时候,它是依据 markdown 的语法规则,可以让一些字体变成我们想要的格式,例如:一级标题,二级标题,加粗,删除线等等…

我们当前的 CSDN 就是拥有这样的规则。


而在之前的博客列表页显示博客的时候,它是一种素的、原始的文字。就算经过博客发布了,但展示给用户看的时候,并没有经过博客编辑器处理,所以,同样地,我们为博客列表页引入 【editor.md】这样的依赖,并通过 JS 代码,让文字变成处理后的结果。


实现思想:

markdown 的原始内容,放在上面的 div 中,我们可以将这个 div 中的内容通过 markdownToHTML 函数进行替换。


展示效果:


image.png


展示效果:


1956d5fcf3db4a0b9289d9f0c875c10b.png


删除博客功能


删除博客,通过前端的 a 标签构造 DELETE 请求,之后在服务器端进行处理。


思想:在博客内容页,若当前登录用户和博客作者是同一个人,即可以删除;若不是同一个人,则不能删除,只能观看。


七、优化项目



对项目的一些功能进行统一处理,如:统一用户的验证问题、统一的异常处理。


image.png


1. 使用 Spring 拦截器来进行用户验证


自定义一个 LoginInterceptor 登录拦截器,在 preHandle 方法中,若返回 true,表示用户已经登录,则前端可以继续访问其他页面;若返回 false,表示用户还未登录,即拦截成功,此时我们应该重定向至登录页面。


@Component
public class LoginInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        HttpSession httpSession = request.getSession(false);
        if (httpSession != null && httpSession.getAttribute("user") != null) {
            // 表示用户已经登录
            return true;
        }
        // 代码走到这里,说明用户还未登录
        response.sendRedirect("blog_login.html");
        return false;
    }
}


添加拦截器至整个 Spring Boot 项目之中,并设置拦截规则,我们期望不对注册页面、登录页面、以及一些静态文件进行拦截。


@Configuration
public class AddConfiguration implements WebMvcConfigurer {
    @Resource
    private LoginInterceptor loginInterceptor;
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(loginInterceptor).
                addPathPatterns("/**"). // 拦截所有 URL 路径
                excludePathPatterns("/login"). // 排除登录路径
                excludePathPatterns("/blog_login.html"). // 排除登录的静态页面
                excludePathPatterns("/**/*.js"). // 排除所有的 js 文件
                excludePathPatterns("/**/*.css").
                excludePathPatterns("/**/*.md").
                excludePathPatterns("/register").
                excludePathPatterns("/blog_register.html").
                excludePathPatterns("/**/*.png"). // 排除所有的 png 图片
                excludePathPatterns("/**/*.jpg");
    }
}


2. 统一异常处理


自定义一个返回的对象,供前端拿到数据,并约定若 state 为 -1,那么就重定向至一个写死的异常页面,供用户观看。


后端 (统一异常处理):


@RestControllerAdvice
public class MyExceptionAdvice {
    @ExceptionHandler(Exception.class)
    public HashMap<String, String> ExceptionAdvice(Exception e, HttpServletResponse response) throws IOException {
        // 构造返回对象
        HashMap<String, String> result = new HashMap<>();
        result.put("state", "-1");
        result.put("data", e.getMessage());
        System.out.println("发现异常: ");
        e.printStackTrace(); // 供后端程序员发现异常位置
        return result;
    }
}


前端 (exception.html):


<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>页面丢失</title>
</head>
<body>
    <h3>服务器端发现异常,正在联系后台紧急处理,请耐心等待...</h3>
    <a href='blog_list.html'>点击返回主页</a>
</body>
</html>


展示结果:


image.png


测试



测试、调试过程其实也花了我很长时间。


后端:


Spring Boot 项目提供了一个非常优秀的单元测试,它可以不污染数据库,达到方法级别的测试,这很方便程序员对于项目的某个地方进行调试。


在本次项目中,我利用单元测试,主要测试了数据库的 CURD 操作。有好几次我发现浏览器控制台总是出现 " 500 " 状态码的情况,找了后端,看了很多英文报错也找不到原因,后来逐一排查,才发现是 SQL 语句报错。当时就是利用了 SpringBootTest 找到的问题,很方便,也不影响后端代码。


前端:


前端主要通过浏览器的控制台发现问题,它的调试和我们平时使用 IDEA 终端调试代码很相似,点到哪都可以停顿。


此外,利用 Fiddler 抓包,postman 构造 HTTP 请求,也能够很好地模拟出与项目相关的前后端交互过程。这样也能够不影响代码的编辑。


总代码



1e2e4a2c87f44dababdd85c6fa460c9e.png

目录
相关文章
|
XML SQL 前端开发
头条博文 (SSM)(上)
头条博文 (SSM)(上)
134 0
头条博文 (SSM)(上)
|
4月前
|
Java 数据库连接 Maven
手把手教你如何搭建SSM框架、图书商城系统案例
这篇文章是关于如何搭建SSM框架以及实现一个图书商城系统的详细教程,包括了项目的配置文件整合、依赖管理、项目结构和运行效果展示,并提供了GitHub源码链接。
手把手教你如何搭建SSM框架、图书商城系统案例
|
6月前
|
搜索推荐 JavaScript Java
计算机Java项目|基于SSM的个性化商铺系统
计算机Java项目|基于SSM的个性化商铺系统
|
3月前
|
Java 应用服务中间件 数据库连接
ssm项目整合,简单的用户管理系统
文章介绍了一个使用SSM框架(Spring、SpringMVC、MyBatis)构建的简单用户管理系统的整合过程,包括项目搭建、数据库配置、各层代码实现以及视图展示。
ssm项目整合,简单的用户管理系统
|
6月前
|
前端开发 JavaScript Java
计算机Java项目|SSM智能仓储系统
计算机Java项目|SSM智能仓储系统
|
3月前
|
XML Java 数据库连接
如何搭建SSM框架、图书商城系统
这是一份详尽的《Spring + SpringMVC + Mybatis 整合指南》,作者耗时良久整理出约五万字的内容,现已经全部笔记公开。此文档详细地介绍了如何搭建与整合SSM框架,具体步骤包括创建Maven项目、添加web骨架、配置pom文件以及整合Spring、SpringMVC和Mybatis等。无论是对初学者还是有一定基础的开发者来说,都是很好的学习资源。此外,作者还提供了项目源码的GitHub链接,方便读者实践。虽然当前主流推荐学习SpringBoot,但了解SSM框架仍然是不可或缺的基础。
50 0
|
4月前
|
SQL Java 应用服务中间件
使用SSM搭建图书商城管理系统(完整过程介绍、售后服务哈哈哈)
这篇文章是关于如何使用SSM框架搭建图书商城管理系统的教程,包括完整过程介绍、常见问题解答和售后服务,提供了项目地址、运行环境配置、效果图展示以及运行代码的步骤。
使用SSM搭建图书商城管理系统(完整过程介绍、售后服务哈哈哈)
|
5月前
|
存储 关系型数据库 测试技术
基于ssm+vue的校园驿站管理系统+(源码+部署说明+演示视频+源码介绍)(2)
基于ssm+vue的校园驿站管理系统+(源码+部署说明+演示视频+源码介绍)
83 1
|
6月前
|
前端开发
杨校老师之基于SSM开发的校园点餐配送系统
杨校老师之基于SSM开发的校园点餐配送系统
67 0
杨校老师之基于SSM开发的校园点餐配送系统
|
6月前
|
小程序 前端开发 测试技术
微信小程序|ssm基于微信小程序的高校课堂教学管理系统
微信小程序|ssm基于微信小程序的高校课堂教学管理系统
下一篇
DataWorks