spring+springmvc+Interceptor+jwt+redis实现sso单点登录

在分布式环境中,如何支持PC、APP(ios、android)等多端的会话共享,这也是所有公司都需要的解决方案,用传统的session方式来解决,我想已经out了,我们是否可以找一个通用的方案,比如用传统cas来实现多系统之间的sso单点登录或使用oauth的第三方登录方案? 今天给大家简单讲解一下使用spring拦截器Interceptor机制、jwt认证方式、redis分布式缓存实现sso单点登录

}

publicvoid setUid(String uid) {

this.uid = uid;

}

public String getToken() {

return token;

}

publicvoid setToken(String token) {

this.token = token;

}

publiclong getRefTime() {

return refTime;

}

publicvoid setRefTime(long refTime) {

this.refTime = refTime;

}

}

public class RedisLogin implements Serializable{

/**

*

*/

private static final long serialVersionUID = 8116817810829835862L;

/**

* 用户id

*/

private String uid;

/**

* jwt生成的token信息

*/

private String token;

/**

* 登录或刷新应用的时间

*/

private long refTime;

public RedisLogin(){

}

public RedisLogin(String uid, String token, long refTime){

this.uid = uid;

this.token = token;

this.refTime = refTime;

}

public String getUid() {

return uid;

}

public void setUid(String uid) {

this.uid = uid;

}

public String getToken() {

return token;

}

public void setToken(String token) {

this.token = token;

}

public long getRefTime() {

return refTime;

}

public void setRefTime(long refTime) {

this.refTime = refTime;

}

}

6. 编写LoginInterceptor.java拦截器

Java代码

publicclass LoginInterceptorimplements HandlerInterceptor{

publicboolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)

throws Exception {

PrintWriter writer =null;

HandlerMethod method =null;

try {

method = (HandlerMethod) handler;

}catch (Exception e) {

writer = response.getWriter();

ResponseVO responseVO = ResponseCode.buildEnumResponseVO(ResponseCode.REQUEST_URL_NOT_SERVICE,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

IsLogin isLogin = method.getMethodAnnotation(IsLogin.class);

if(null == isLogin){

returntrue;

}

response.setCharacterEncoding("utf-8");

String token = request.getHeader("token");

String uid = request.getHeader("uid");

//token不存在

if(StringUtils.isEmpty(token)) {

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TOKEN_NOT_NULL,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

if(StringUtils.isEmpty(uid)){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_NULL,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

Login login = JWT.unsign(token, Login.class);

//解密token后的loginId与用户传来的loginId判断是否一致

if(null == login || !StringUtils.equals(login.getUid(), uid)){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

//验证登录时间

RedisLogin redisLogin = (RedisLogin)JedisUtils.getObject(uid);

if(null == redisLogin){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.RESPONSE_CODE_UNLOGIN_ERROR,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

if(!StringUtils.equals(token, redisLogin.getToken())){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

//系统时间>有效期(说明已经超过有效期)

if (System.currentTimeMillis() > redisLogin.getRefTime()) {

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TIME_EXP,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

//重新刷新有效期

redisLogin =new RedisLogin(uid, token, System.currentTimeMillis() + 60L* 1000L* 30L);

JedisUtils.setObject(uid , redisLogin,360000000);

returntrue;

}

publicvoid postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,

ModelAndView modelAndView)throws Exception {

}

publicvoid afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)

throws Exception {

}

privatevoid responseMessage(HttpServletResponse response, PrintWriter out, ResponseVO responseVO) {

response.setContentType("application/json; charset=utf-8");

JSONObject result =new JSONObject();

result.put("result", responseVO);

out.print(result);

out.flush();

out.close();

}

}

[java]view plaincopyprint?

publicclass LoginInterceptorimplements HandlerInterceptor{

publicboolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)

throws Exception {

PrintWriter writer =null;

HandlerMethod method =null;

try {

method = (HandlerMethod) handler;

}catch (Exception e) {

writer = response.getWriter();

ResponseVO responseVO = ResponseCode.buildEnumResponseVO(ResponseCode.REQUEST_URL_NOT_SERVICE,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

IsLogin isLogin = method.getMethodAnnotation(IsLogin.class);

if(null == isLogin){

returntrue;

}

response.setCharacterEncoding("utf-8");

String token = request.getHeader("token");

String uid = request.getHeader("uid");

//token不存在

if(StringUtils.isEmpty(token)) {

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TOKEN_NOT_NULL,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

if(StringUtils.isEmpty(uid)){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_NULL,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

Login login = JWT.unsign(token, Login.class);

//解密token后的loginId与用户传来的loginId判断是否一致

if(null == login || !StringUtils.equals(login.getUid(), uid)){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

//验证登录时间

RedisLogin redisLogin = (RedisLogin)JedisUtils.getObject(uid);

if(null == redisLogin){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.RESPONSE_CODE_UNLOGIN_ERROR,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

if(!StringUtils.equals(token, redisLogin.getToken())){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

//系统时间>有效期(说明已经超过有效期)

if (System.currentTimeMillis() > redisLogin.getRefTime()) {

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TIME_EXP,false);

responseMessage(response, writer, responseVO);

returnfalse;

}

//重新刷新有效期

redisLogin =new RedisLogin(uid, token, System.currentTimeMillis() + 60L* 1000L* 30L);

JedisUtils.setObject(uid , redisLogin,360000000);

returntrue;

}

publicvoid postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,

ModelAndView modelAndView)throws Exception {

}

publicvoid afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)

throws Exception {

}

privatevoid responseMessage(HttpServletResponse response, PrintWriter out, ResponseVO responseVO) {

response.setContentType("application/json; charset=utf-8");

JSONObject result =new JSONObject();

result.put("result", responseVO);

out.print(result);

out.flush();

out.close();

}

}

public class LoginInterceptor implements HandlerInterceptor{

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)

throws Exception {

PrintWriter writer = null;

HandlerMethod method = null;

try {

method = (HandlerMethod) handler;

} catch (Exception e) {

writer = response.getWriter();

ResponseVO responseVO = ResponseCode.buildEnumResponseVO(ResponseCode.REQUEST_URL_NOT_SERVICE, false);

responseMessage(response, writer, responseVO);

return false;

}

IsLogin isLogin = method.getMethodAnnotation(IsLogin.class);

if(null == isLogin){

return true;

}

response.setCharacterEncoding("utf-8");

String token = request.getHeader("token");

String uid = request.getHeader("uid");

//token不存在

if(StringUtils.isEmpty(token)) {

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TOKEN_NOT_NULL, false);

responseMessage(response, writer, responseVO);

return false;

}

if(StringUtils.isEmpty(uid)){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_NULL, false);

responseMessage(response, writer, responseVO);

return false;

}

Login login = JWT.unsign(token, Login.class);

//解密token后的loginId与用户传来的loginId判断是否一致

if(null == login || !StringUtils.equals(login.getUid(), uid)){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED, false);

responseMessage(response, writer, responseVO);

return false;

}

//验证登录时间

RedisLogin redisLogin = (RedisLogin)JedisUtils.getObject(uid);

if(null == redisLogin){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.RESPONSE_CODE_UNLOGIN_ERROR, false);

responseMessage(response, writer, responseVO);

return false;

}

if(!StringUtils.equals(token, redisLogin.getToken())){

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED, false);

responseMessage(response, writer, responseVO);

return false;

}

//系统时间>有效期(说明已经超过有效期)

if (System.currentTimeMillis() > redisLogin.getRefTime()) {

writer = response.getWriter();

ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TIME_EXP, false);

responseMessage(response, writer, responseVO);

return false;

}

//重新刷新有效期

redisLogin = new RedisLogin(uid, token, System.currentTimeMillis() + 60L* 1000L* 30L);

JedisUtils.setObject(uid , redisLogin, 360000000);

return true;

}

public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,

ModelAndView modelAndView) throws Exception {

}

public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)

throws Exception {

}

private void responseMessage(HttpServletResponse response, PrintWriter out, ResponseVO responseVO) {

response.setContentType("application/json; charset=utf-8");

JSONObject result = new JSONObject();

result.put("result", responseVO);

out.print(result);

out.flush();

out.close();

}

}

7. 定义异常的LoginResponseCode

Java代码

publicenum LoginResponseCode {

USERID_NOT_NULL(3001,"用户id不能为空."),

LOGIN_TOKEN_NOT_NULL(3002,"登录token不能为空."),

USERID_NOT_UNAUTHORIZED(3003,"用户token或ID验证不通过"),

RESPONSE_CODE_UNLOGIN_ERROR(421,"未登录异常"),

LOGIN_TIME_EXP(3004,"登录时间超长,请重新登录");

// 成员变量

privateint code;//状态码

private String message;//返回消息

// 构造方法

private LoginResponseCode(int code,String message) {

this.code = code;

this.message = message;

}

publicint getCode() {

return code;

}

publicvoid setCode(int code) {

this.code = code;

}

public String getMessage() {

return message;

}

publicvoid setMessage(String message) {

this.message = message;

}

publicstatic ResponseVO buildEnumResponseVO(LoginResponseCode responseCode, Object data) {

returnnew ResponseVO(responseCode.getCode(),responseCode.getMessage(),data);

}

publicstatic Map buildReturnMap(LoginResponseCode responseCode, Object data) {

Map map =new HashMap();

map.put("code", responseCode.getCode());

map.put("message", responseCode.getMessage());

map.put("data", data);

return map;

}

}

[java]view plaincopyprint?

publicenum LoginResponseCode {

USERID_NOT_NULL(3001,"用户id不能为空."),

LOGIN_TOKEN_NOT_NULL(3002,"登录token不能为空."),

USERID_NOT_UNAUTHORIZED(3003,"用户token或ID验证不通过"),

RESPONSE_CODE_UNLOGIN_ERROR(421,"未登录异常"),

LOGIN_TIME_EXP(3004,"登录时间超长,请重新登录");

// 成员变量

privateint code;//状态码

private String message;//返回消息

// 构造方法

private LoginResponseCode(int code,String message) {

this.code = code;

this.message = message;

}

publicint getCode() {

return code;

}

publicvoid setCode(int code) {

this.code = code;

}

public String getMessage() {

return message;

}

publicvoid setMessage(String message) {

this.message = message;

}

publicstatic ResponseVO buildEnumResponseVO(LoginResponseCode responseCode, Object data) {

returnnew ResponseVO(responseCode.getCode(),responseCode.getMessage(),data);

}

publicstatic Map buildReturnMap(LoginResponseCode responseCode, Object data) {

Map map =new HashMap();

map.put("code", responseCode.getCode());

map.put("message", responseCode.getMessage());

map.put("data", data);

return map;

}

}

public enum LoginResponseCode {

USERID_NOT_NULL(3001,"用户id不能为空."),

LOGIN_TOKEN_NOT_NULL(3002,"登录token不能为空."),

USERID_NOT_UNAUTHORIZED(3003, "用户token或ID验证不通过"),

RESPONSE_CODE_UNLOGIN_ERROR(421, "未登录异常"),

LOGIN_TIME_EXP(3004, "登录时间超长,请重新登录");

// 成员变量

private int code; //状态码

private String message; //返回消息

// 构造方法

private LoginResponseCode(int code,String message) {

this.code = code;

this.message = message;

}

public int getCode() {

return code;

}

public void setCode(int code) {

this.code = code;

}

public String getMessage() {

return message;

}

public void setMessage(String message) {

this.message = message;

}

public static ResponseVO buildEnumResponseVO(LoginResponseCode responseCode, Object data) {

return new ResponseVO(responseCode.getCode(),responseCode.getMessage(),data);

}

public static Map buildReturnMap(LoginResponseCode responseCode, Object data) {

Map map = new HashMap();

map.put("code", responseCode.getCode());

map.put("message", responseCode.getMessage());

map.put("data", data);

return map;

}

}

8. 编写统一sso单点登录接口:

Java代码

@RequestMapping(value ="/login", method = RequestMethod.POST)

public Map login(@RequestBody JSONObject json){

String loginName = json.optString("loginName");

String password = json.optString("password");

//校验用户名不能为空

if(StringUtils.isEmpty(loginName)){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_NAME_IS_NOT_EMPTY,null);

}

//校验用户密码不能为空

if(StringUtils.isEmpty(password)){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_PWD_CAN_NOT_BE_EMPTY,null);

}

//根据用户名查询数据库用户信息

User user = systemService.getBaseUserByLoginName(loginName);

//用户名或密码不正确

if(null == user){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);

}

boolean isValidate = systemService.validatePassword(password, user.getPassword());

if(!isValidate){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);

}

if(isValidate){

//HttpSession session =request.getSession(false);

Login login =new Login(user.getId(), user.getLoginName(), user.getPassword());

//给用户jwt加密生成token

String token = JWT.sign(login, 60L* 1000L* 30L);

Map result =new HashMap();

result.put("loginToken", token);

result.put("userId", user.getId());

result.put("user", user);

//保存用户信息到session

//session.setAttribute(user.getId() + "@@" + token, user);

//重建用户信息

this.rebuildLoginUser(user.getId(), token);

return ResponseCode.buildReturnMap(ResponseCode.RESPONSE_CODE_LOGIN_SUCCESS, result);

}

return ResponseCode.buildReturnMap(ResponseCode.USER_LOGIN_PWD_ERROR,false);

}

[java]view plaincopyprint?

@RequestMapping(value ="/login", method = RequestMethod.POST)

public Map login(@RequestBody JSONObject json){

String loginName = json.optString("loginName");

String password = json.optString("password");

//校验用户名不能为空

if(StringUtils.isEmpty(loginName)){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_NAME_IS_NOT_EMPTY,null);

}

//校验用户密码不能为空

if(StringUtils.isEmpty(password)){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_PWD_CAN_NOT_BE_EMPTY,null);

}

//根据用户名查询数据库用户信息

User user = systemService.getBaseUserByLoginName(loginName);

//用户名或密码不正确

if(null == user){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);

}

boolean isValidate = systemService.validatePassword(password, user.getPassword());

if(!isValidate){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);

}

if(isValidate){

//HttpSession session =request.getSession(false);

Login login =new Login(user.getId(), user.getLoginName(), user.getPassword());

//给用户jwt加密生成token

String token = JWT.sign(login, 60L* 1000L* 30L);

Map result =new HashMap();

result.put("loginToken", token);

result.put("userId", user.getId());

result.put("user", user);

//保存用户信息到session

//session.setAttribute(user.getId() + "@@" + token, user);

//重建用户信息

this.rebuildLoginUser(user.getId(), token);

return ResponseCode.buildReturnMap(ResponseCode.RESPONSE_CODE_LOGIN_SUCCESS, result);

}

return ResponseCode.buildReturnMap(ResponseCode.USER_LOGIN_PWD_ERROR,false);

}

@RequestMapping(value = "/login", method = RequestMethod.POST)

public Map login(@RequestBody JSONObject json){

String loginName = json.optString("loginName");

String password = json.optString("password");

//校验用户名不能为空

if(StringUtils.isEmpty(loginName)){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_NAME_IS_NOT_EMPTY, null);

}

//校验用户密码不能为空

if(StringUtils.isEmpty(password)){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_PWD_CAN_NOT_BE_EMPTY, null);

}

//根据用户名查询数据库用户信息

User user = systemService.getBaseUserByLoginName(loginName);

//用户名或密码不正确

if(null == user){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS, false);

}

boolean isValidate = systemService.validatePassword(password, user.getPassword());

if(!isValidate){

return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS, false);

}

if(isValidate){

//HttpSession session =request.getSession(false);

Login login = new Login(user.getId(), user.getLoginName(), user.getPassword());

//给用户jwt加密生成token

String token = JWT.sign(login, 60L* 1000L* 30L);

Map result =new HashMap();

result.put("loginToken", token);

result.put("userId", user.getId());

result.put("user", user);

//保存用户信息到session

//session.setAttribute(user.getId() + "@@" + token, user);

//重建用户信息

this.rebuildLoginUser(user.getId(), token);

return ResponseCode.buildReturnMap(ResponseCode.RESPONSE_CODE_LOGIN_SUCCESS, result);

}

return ResponseCode.buildReturnMap(ResponseCode.USER_LOGIN_PWD_ERROR, false);

}

9. 测试sso单点登录:

返回结果集:

Java代码

{

"message":"用户登录成功",

"data": {

"loginToken":"eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJleHAiOjE1MDkzODA1OTU0NTksInBheWxvYWQiOiJ7XCJ1aWRcIjpcIjExXCIsXCJsb2dpbk5hbWVcIjpcImFkbWluXCIsXCJwYXNzd29yZFwiOlwiZjU0NGQxM2QyY2EwNDU5ZGQ0ZTU1NzVjNmZkYWIzMzM0MzE1MWFlZjgwYmE5ZTNiN2U1ZjM2MzJcIn0ifQ.56L60WtxHXSu9vNs6XsWy5zbmc3kP_IWG1YpReK50DM",

"userId":"11",

"user": {

"QQ":"2147775633",

"id":"11",

"isNewRecord":false,

"remarks":"",

"createDate":"2017-08-08 08:08:08",

"updateDate":"2017-10-29 11:23:50",

"loginName":"admin",

"no":"00012",

"name":"admin",

"email":"2147775633@qq.com",

"phone":"400000000",

"mobile":"13888888888",

"userType":"",

"loginIp":"0:0:0:0:0:0:0:1",

"loginDate":"2017-10-30 10:48:06",

"loginFlag":"1",

"photo":"",

"idCard":"420888888888888888",

"oldLoginIp":"0:0:0:0:0:0:0:1",

"oldLoginDate":"2017-10-30 10:48:06",

"roleNames":"",

"admin":false

}

},

"code":200

}

愿意了解框架技术或者源码的朋友直接求求交流分享技术:3133806896

分布式的一些解决方案,有愿意了解的朋友可以找我们团队探讨

更多详细源码参考来源

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容