仿牛客论坛项目之显示登录信息

前言:用户登录成功后会将信息存到浏览器中(具体时间由我们来设置),当用户再次登录时,需要从浏览器中获取信息到服务器中,然后将服务器中的内容显示到浏览器中,具体模板如下。
请添加图片描述
拦截器示例(拦截浏览器传过来的请求)
● 定义拦截器,实现HandlerInterceptor
● 配置拦截器,为它指定拦截、排除的路径
拦截器应用
● 在请求开始时查询用户登录
● 在本次请求中持有用户数据
● 在模板视图上显示用户数据
● 在请求结束时清理用户数据

1 创建util类 HostHolder

此处选择将 user 类通过线程安全类 ThreadLocal 进行封装,代替 session 封装,主要目的是减小服务器的压力,并且保证线程安全。

import com.zcq.community.entity.User;
import org.springframework.stereotype.Component;

/**
 * 持有用户的信息,代替session对象
 */
@Component
public class HostHolder {

    // 线程安全
    private ThreadLocal<User> users = new ThreadLocal<>();

    public void setUser(User user) {
        users.set(user);
    }

    public User getUser() {
        return users.get();
    }

    public void clear() {
        users.remove();
    }

}

2 编写拦截器,处理请求

import com.zcq.community.entity.LoginTicket;
import com.zcq.community.entity.User;
import com.zcq.community.service.UserService;
import com.zcq.community.util.CookieUtil;
import com.zcq.community.util.HostHolder;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Date;

@Component
public class LoginTicketInterceptor implements HandlerInterceptor {

    @Autowired
    private UserService userService;

    @Autowired
    private HostHolder hostHolder;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 从cookie中获取凭证
        String ticket = CookieUtil.getValue(request, "ticket");
        if(ticket != null) {
            // 查询凭证
            LoginTicket loginTicket = userService.findLoginTicket(ticket);
            // 检查凭证是否有效
            // 不为空 且 处于登录状态 且 超时时间晚于当前时间
            if(loginTicket != null && loginTicket.getStatus() == 0 && loginTicket.getExpired().after(new Date())) {
                // 根据凭证查询用户
                User user = userService.findUserById(loginTicket.getUserId());
                System.out.println(user.getId());
                System.out.println(user.getHeaderUrl());
                // 在本次请求中持有用户
                hostHolder.setUser(user);
            }
        }
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        User user = hostHolder.getUser();
        if(user != null && modelAndView != null) {
            modelAndView.addObject("loginUser", user);
        }
    }

    // 在整个请求结束之后将user对象清掉
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        hostHolder.clear();
    }

}

3 编写前端代码,进行显示

注意:此处需要做业务逻辑处理,比如:只有用户登录了才会在头部显示消息和用户头像等信息,用户没登录则显示注册、登录字样。

<div>
	<ul>
		<li>
			<a th:href="@{/index}">首页</a>
		</li>
		<li th:if="${loginUser!=null}">
			<a href="site/letter.html">消息<span>12</span></a>
		</li>
		<li th:if="${loginUser==null}">
			<a th:href="@{/register}">注册</a>
		</li>
		<li th:if="${loginUser==null}">
			<a th:href="@{/login}">登录</a>
		</li>
		<li th:if="${loginUser!=null}">
			<a href="#" id="navbarDropdown" role="button">
				<img th:src="${loginUser.headerUrl}"/> <!-- 头像显示 -->
			</a>
			<div>
				<a>个人主页</a>
				<a th:href="@{/user/setting}">账号设置</a>
				<a th:href="@{/logout}">退出登录</a>
				<div></div>
				<span th:utext="${loginUser.username}">nowcoder</span>
			</div>
		</li>
	</ul>
</div>

4 bug反思

这次编写代码的过程中遇到了一个bug,就是没有获取到用户的头像,看了网上的代码说可以将th:src="${loginUser.headerUrl}"中的 $ 改为 # ,改完之后代码确实不会报错了,但还是获取不到用户的头像信息,最近经过排查确认,在th:if="${loginUser!=null}"中的不等于我写成等于了,导致逻辑出错,无法获取到用户头像和用户名,改完后可以获取到用户的用户名,但头像还是无法获取到,最终调用头像路径的静态资源发现可以获取到。最终刷新页面获取到了,不愧是很神奇的bug。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值