一直懒于做笔记,今天突然想好好把项目记录下来,希望秋招时能收到满意的offer(实习计划已经凉透了)。不过发现记笔记也太耗时了,也不知道自己能坚持多久。先把之前做的都梳理一下吧。
做了一个秒杀系统(系统算不上,勉强称为功能吧),毕竟不在大厂实习过也没有高并发的经历(何止大厂,小厂都没有,扎心了),想体验一下秒杀的场景和高并发时发生的问题(呵,谁没事想体验,还不是被面试逼的)。
这里直接给entity的代码了,数据库和entity的字段一一对应。
public class MiaoshaUser {
private Long id;//手机号
private String nickname;
private String password;
private String salt;
private String head;//头像
private Date registerDate;
private Date lastLoginDate;
private Integer loginCount;
//Getter and Setter
//...
}
给出Dao层代码:
@Repository//这里一定要有这个注解,否则会报错,为什么?
@Mapper
public interface MiaoshaUserDao {
@Select("select * from user where id = #{id}")
public MiaoshaUser getById(@Param("id") long id);
}
用户输入密码到请求中,做了一次MD5加密,此时salt是固定值;从请求到数据库中又做了一次MD5加密,这个MD5是随机生成并存在数据库中的。
public class MD5Util {
//这里直接添加maven依赖、导包、用了库函数
public static String md5(String src) {
return DigestUtils.md5Hex(src);
}
//第一次加密的固定salt
private static final String salt = "1a2b3c4d";
//从客户输入到网上,第一次md5
public static String inputPassToFormPass(String inputPass) {
String str = "" + salt.charAt(0) + salt.charAt(2) + inputPass + salt.charAt(5) + salt.charAt(4);//这里排列组合可以任意,记住即可
return md5(str);
}
//从请求到数据库,第二次md5
public static String formPassToDBPass(String formPass, String salt) {
String str = "" + salt.charAt(0) + salt.charAt(2) + formPass + salt.charAt(5) + salt.charAt(4);
return md5(str);
}
//把以上两次结合起来
public static String inputPassToDbPass(String inputPass, String saltDB) {
String formPass = inputPassToFormPass(inputPass);
String dbPass = formPassToDBPass(formPass, saltDB);
return dbPass;
}
public static void main(String[] args) {//测试一下
System.out.println(inputPassToFormPass("12222222222"));
System.out.println(formPassToDBPass(inputPassToFormPass("123456"), "1a2b3c4d"));
System.out.println(inputPassToDbPass("12222222222", "123456"));
}
}
Redis不介绍了,这里主要记一下其在项目中是如何用的。业务中将token的键值对存放在redis中。每次登陆,都更新用户的token和redis的过期时间。
1. Redis的Key生成
Redis的key生成也是一个比较关键的部分,这里采用接口+抽象类+具体实现的形式将Redis的key生成出来。
先写一个接口。
public interface KeyPrefix {
public int expireSeconds();//过期时间
public String getPrefix();
}
抽象类。
public abstract class BasePrefix implements KeyPrefix {
private int expireSeconds;
private String prefix;
public BasePrefix(String prefix) {//0代表永不过期
this(0, prefix);
}
public BasePrefix(int expireSeconds, String prefix) {
this.expireSeconds = expireSeconds;
this.prefix = prefix;
}
public int expireSeconds() {//默认0代表永不过期
return expireSeconds;
}
public String getPrefix() {
String className = getClass().getSimpleName();//得到类的简写名称
return className + ":" + prefix;
}
}
实现类。
public class MiaoshaUserKey extends BasePrefix {
public static final int TOKEN_EXPIRE = 3600 * 24 * 2; //假设2天
private MiaoshaUserKey(int expireSeconds, String prefix) {
super(expireSeconds, prefix);
}
public static MiaoshaUserKey token = new MiaoshaUserKey(TOKEN_EXPIRE, "tk");
}
2. Redis的配置和RedisService
先在properties中添加配置。
#redis
spring.redis.host=127.0.0.1
spring.redis.port=6379
spring.redis.lettuce.pool.max-active=10
spring.redis.jedis.pool.max-idle=10
spring.redis.timeout=3000ms
spring.redis.jedis.pool.max-wait=3000ms
写一个RedisService的配置类/包装类。
@Service
public class RedisService implements InitializingBean {
private JedisPool jedisPool;
@Override
public void afterPropertiesSet() throws Exception { //生成bean时执行的方法
jedisPool = new JedisPool();
}
/**
* 获取单个对象
*/
public <T> T get(KeyPrefix prefix, String key, Class<T> clazz) {
Jedis jedis = null;
try {
jedis = jedisPool.getResource();
//生成真正的key
String realKey = prefix.getPrefix() + key;
String str = jedis.get(realKey);
T t = stringToBean(str, clazz);
return t;
} finally {
returnToPool(jedis);
}
}
/**
* 设置对象
*/
public <T> boolean set(KeyPrefix prefix, String key, T value) {
Jedis jedis = null;
try {
jedis = jedisPool.getResource();
String str = beanToString(value);
if (str == null || str.length() <= 0) {
return false;
}
//生成真正的key
String realKey = prefix.getPrefix() + key;
int seconds = prefix.expireSeconds();
if (seconds <= 0) {
jedis.set(realKey, str);
} else {
jedis.setex(realKey, seconds, str);
}
return true;
} finally {
returnToPool(jedis);
}
}
/**
* 判断key是否存在
*/
public <T> boolean exists(KeyPrefix prefix, String key) {
Jedis jedis = null;
try {
jedis = jedisPool.getResource();
//生成真正的key
String realKey = prefix.getPrefix() + key;
return jedis.exists(realKey);
} finally {
returnToPool(jedis);
}
}
/**
* 增加值
*/
public <T> Long incr(KeyPrefix prefix, String key) {
Jedis jedis = null;
try {
jedis = jedisPool.getResource();
//生成真正的key
String realKey = prefix.getPrefix() + key;
return jedis.incr(realKey);
} finally {
returnToPool(jedis);
}
}
/**
* 减少值
*/
public <T> Long decr(KeyPrefix prefix, String key) {
Jedis jedis = null;
try {
jedis = jedisPool.getResource();
//生成真正的key
String realKey = prefix.getPrefix() + key;
return jedis.decr(realKey);
} finally {
returnToPool(jedis);
}
}
private <T> String beanToString(T value) {
if (value == null) {
return null;
}
Class<?> clazz = value.getClass();
if (clazz == int.class || clazz == Integer.class) {
return "" + value;
} else if (clazz == String.class) {
return (String) value;
} else if (clazz == long.class || clazz == Long.class) {
return "" + value;
} else {
return JSON.toJSONString(value);
}
}
@SuppressWarnings("unchecked")
private <T> T stringToBean(String str, Class<T> clazz) {
if (str == null || str.length() <= 0 || clazz == null) {
return null;
}
if (clazz == int.class || clazz == Integer.class) {
return (T) Integer.valueOf(str);
} else if (clazz == String.class) {
return (T) str;
} else if (clazz == long.class || clazz == Long.class) {
return (T) Long.valueOf(str);
} else {
return JSON.toJavaObject(JSON.parseObject(str), clazz);
}
}
private void returnToPool(Jedis jedis) {
if (jedis != null) {
jedis.close();
}
}
}
3. Service层关于Session的内容
@Service
public class MiaoshaUserService {
public static final String COOKI_NAME_TOKEN = "token";
@Autowired
MiaoshaUserDao miaoshaUserDao;
@Autowired
RedisService redisService;
public MiaoshaUser getById(long id) {
return miaoshaUserDao.getById(id);
}
public MiaoshaUser getByToken(HttpServletResponse response, String token) {
if (StringUtils.isEmpty(token)) {
return null;
}
MiaoshaUser user = redisService.get(MiaoshaUserKey.token, token, MiaoshaUser.class);
//延长有效期
if (user != null) {
addCookie(response, token, user);
}
return user;
}
public String login(HttpServletResponse response, LoginVo loginVo) {
//一系列验证操作,先省略
//...
//验证成功后,最后生成cookie
String token = UUIDUtil.uuid();
addCookie(response, token, user);
return token;
}
private void addCookie(HttpServletResponse response, String token, MiaoshaUser user) {
redisService.set(MiaoshaUserKey.token, token, user);//把token信息写到缓存中,在redis中管理session
Cookie cookie = new Cookie(COOKI_NAME_TOKEN, token);//在cookie中放入名为“token” 值为token的字段
cookie.setMaxAge(MiaoshaUserKey.token.expireSeconds());//这里把cookie和token的Redis设为一致的有效期
cookie.setPath("/");
response.addCookie(cookie);
}
}
相关代码块:UUIDUtil和LoginVo(LoginVo在后面贴,涉及到下一个知识点)
public class UUIDUtil {
public static String uuid() {
return UUID.randomUUID().toString().replace("-", "");
}
}
这个模块的功能主要是避免过多重复复杂的判断,为了表现出代码的简洁性,把参数校验和业务代码抽离出来,并采用全局异常处理的形式统一处理异常。
先给出两个辅助类的代码,Result类和CodeMsg类,这两个类的作用分别是返回结果和错误代码标注。
public class Result<T> {
private int code;
private String msg;
private T data;
/**
* 成功时候的调用
*/
public static <T> Result<T> success(T data) {
return new Result<T>(data);
}
/**
* 失败时候的调用
*/
public static <T> Result<T> error(CodeMsg codeMsg) {
return new Result<T>(codeMsg);
}
//三种构造函数
private Result(T data) {
this.data = data;
}
private Result(int code, String msg) {
this.code = code;
this.msg = msg;
}
private Result(CodeMsg codeMsg) {
if (codeMsg != null) {
this.code = codeMsg.getCode();
this.msg = codeMsg.getMsg();
}
}
//Getter and Setter...
}
public class CodeMsg {
private int code;
private String msg;
//通用的错误码
public static CodeMsg SUCCESS = new CodeMsg(0, "success");
public static CodeMsg SERVER_ERROR = new CodeMsg(500100, "服务端异常");
public static CodeMsg BIND_ERROR = new CodeMsg(500101, "参数校验异常:%s");
//登录模块 5002XX
public static CodeMsg SESSION_ERROR = new CodeMsg(500210, "Session不存在或者已经失效");
public static CodeMsg PASSWORD_EMPTY = new CodeMsg(500211, "登录密码不能为空");
public static CodeMsg MOBILE_EMPTY = new CodeMsg(500212, "手机号不能为空");
public static CodeMsg MOBILE_ERROR = new CodeMsg(500213, "手机号格式错误");
public static CodeMsg MOBILE_NOT_EXIST = new CodeMsg(500214, "手机号不存在");
public static CodeMsg PASSWORD_ERROR = new CodeMsg(500215, "密码错误");
//商品模块 5003XX
//订单模块 5004XX
//秒杀模块 5005XX
public static CodeMsg MIAO_SHA_OVER = new CodeMsg(500500, "商品已经秒杀完毕");
public static CodeMsg REPEATE_MIAOSHA = new CodeMsg(500501, "不能重复秒杀");
//两种构造函数
private CodeMsg() {}
private CodeMsg(int code, String msg) {
this.code = code;
this.msg = msg;
}
//Getter and Setter...
public CodeMsg fillArgs(Object... args) {//变参数,用args填充message
int code = this.code;
String message = String.format(this.msg, args);
return new CodeMsg(code, message);
}
@Override
public String toString() {
return "CodeMsg [code=" + code + ", msg=" + msg + "]";
}
}
@Controller
@RequestMapping("/login")
public class LoginController {
private static Logger log = LoggerFactory.getLogger(LoginController.class);
@Autowired
MiaoshaUserService miaoshaUserService;
@Autowired
RedisService redisService;
//请求登陆页面的url,返回登陆页面
@RequestMapping("/to_login")
public String toLogin() {
return "login";
}
@RequestMapping("/do_login")
@ResponseBody //返回jason串
public Result<String> doLogin(HttpServletResponse response, @Valid LoginVo loginVo) {//注意这边有个valid,那么这个类里面需要设置一系列注解
log.info(loginVo.toString());
//登录
String token = miaoshaUserService.login(response, loginVo);
return Result.success(token);
}
}
注意到这里有个@Valid参数校验的注解,表示对loginVo这个对象进行参数校验。贴上LoginVo类的代码。
public class LoginVo {
@NotNull//表示不能为空
@IsMobile//自定义一个校验器
private String mobile;
@NotNull
@Length(min = 32)//表示长度最小为32位
private String password;
//Getter and Setter...
@Override
public String toString() {
return "LoginVo [mobile=" + mobile + ", password=" + password + "]";
}
}
这里两个字段上均有注解,表示mobile不能为空且要符合自定义注解@IsMobile的要求,password字段需要不能为空且长度最少32位。
下面看一下如何自定义@IsMobile注解。
@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER})
@Retention(RUNTIME)
@Documented
@Constraint(validatedBy = {IsMobileValidator.class}) //系统会调用这个校验器进行校验
public @interface IsMobile {
boolean required() default true;//默认参数必须有值(不能为空)
String message() default "手机号码格式错误";//如果验证不通过的信息
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
}
这里出现了IsMobileValidator.class,这是具体的校验方法,贴上代码。
public class IsMobileValidator implements ConstraintValidator<IsMobile, String> {
private boolean required = false;
//初始化
public void initialize(IsMobile constraintAnnotation) {
required = constraintAnnotation.required();
}
public boolean isValid(String value, ConstraintValidatorContext context) { //通过这个方法判断是否合法
if (required) {
return ValidatorUtil.isMobile(value);
} else {
if (StringUtils.isEmpty(value)) {
return true;
} else {
return ValidatorUtil.isMobile(value);
}
}
}
}
里面用到了辅助工具类ValidatorUtil,这是具体的方法,当然也可以不用这个类直接写在上面代码中,这里为了清晰把具体判别方法写成辅助类。
public class ValidatorUtil {
private static final Pattern mobile_pattern = Pattern.compile("1\\d{10}");//正则表达式,以1开头,后面跟了10个数字
public static boolean isMobile(String src) {
if (StringUtils.isEmpty(src)) {
return false;
}
Matcher m = mobile_pattern.matcher(src);//匹配
return m.matches();
}
}
参数校验部分就全部完成啦~
@Service
public class MiaoshaUserService {
public static final String COOKI_NAME_TOKEN = "token";
@Autowired
MiaoshaUserDao miaoshaUserDao;
@Autowired
RedisService redisService;
public MiaoshaUser getById(long id) {...}//上面实现过了,略
public MiaoshaUser getByToken(HttpServletResponse response, String token) {...}//已写,略。
public String login(HttpServletResponse response, LoginVo loginVo) {
if (loginVo == null) {
throw new GlobalException(CodeMsg.SERVER_ERROR);//直接抛,在handler中会集中处理
}
String mobile = loginVo.getMobile();
String formPass = loginVo.getPassword();
//判断手机号是否存在
MiaoshaUser user = getById(Long.parseLong(mobile));
if (user == null) {
throw new GlobalException(CodeMsg.MOBILE_NOT_EXIST);
}
//验证密码
String dbPass = user.getPassword();
String saltDB = user.getSalt();
String calcPass = MD5Util.formPassToDBPass(formPass, saltDB);
if (!calcPass.equals(dbPass)) {
throw new GlobalException(CodeMsg.PASSWORD_ERROR);
}
//生成cookie
String token = UUIDUtil.uuid();
addCookie(response, token, user);
return token;
}
private void addCookie(HttpServletResponse response, String token, MiaoshaUser user) {...} //略。
}
这里面统一抛了GlobalException,其实现为:
public class GlobalException extends RuntimeException {
private static final long serialVersionUID = 1L;
private CodeMsg cm;
public GlobalException(CodeMsg cm) {
super(cm.toString());
this.cm = cm;
}
public CodeMsg getCm() {
return cm;
}
}
这只是继承了RuntimeException,重新包装了一下,下面才是重点,捕捉全局异常~
@ControllerAdvice//@ControllerAdvice是一个@Component,用于定义@ExceptionHandler,@InitBinder和@ModelAttribute方法,适用于所有使用@RequestMapping方法。
@ResponseBody
public class GlobalExceptionHandler {
@ExceptionHandler(value = Exception.class)//拦截什么类型的异常,这里是拦截所有异常
public Result<String> exceptionHandler(HttpServletRequest request, Exception e) {
e.printStackTrace();
if (e instanceof GlobalException) {
GlobalException ex = (GlobalException) e;
return Result.error(ex.getCm());
} else if (e instanceof BindException) { //参数验证异常,上面定义的@Valid参数不合法时
BindException ex = (BindException) e;
List<ObjectError> errors = ex.getAllErrors();//也许有多个错误
ObjectError error = errors.get(0);//获取第一个错误
String msg = error.getDefaultMessage();
return Result.error(CodeMsg.BIND_ERROR.fillArgs(msg));
} else {
return Result.error(CodeMsg.SERVER_ERROR);
}
}
}
第一部分终于完成啦,写了一天了,哭泣,第一次用markdown,感觉有点好玩~ 明天再更~