无论是后台管理系统还是用户端,登录场景是必备的。那么掌握高效的写法是非常有利于后期的维护的。使用ThinkPHP6.0可以使用两种非常高效的写法。
一种是使用功能类继承的方式,一种是使用中间件拦截来实现。今天,小超越工作室就先分享第一种,功能类继承的方法,也就是extends Base方法。
这种方式的好处就只要功能继承校验的base即可自动判断不符合条件,自动跳转到对应的页面去,不需要再额外写代码 一劳永逸。
登录逻辑:
1、访问登录页-》未登录-》页面维持不变
2、访问登录页-》已登录-》跳转到指定页面
3、指定页面-》未登录-》跳转到登录页面
4、指定页面-》已登录-》页面维持不变
5、访问退出页面->跳转到登录页面
我们需要创建四个控制器,分别是AdminBase.php/Login.php/Loginout.php/admin.php
以下是各自的代码及注释
登录控制器 login.php
/*
* @Author: 小超越
* @LastEditors: 小超越工作室
*/
namespace app\controller;
use think\facade\Request;
use think\facade\View;
use app\model\User;
use think\facade\Cache;
use app\controller\AdminBase;
class Login extends AdminBase
{
/**
* @description: 重写初始化,防止跳转
*/
public function initialize(){
if($this->isLogin()){
$this->redirect(url('../admin/index'));
}
}
public function index(){
return view::fetch();
}
/**
* @description: 用户登录操作
* @param $username 用户名称
* @param $password 用户密码 测试用无需md5加密
* @return array
*/
public function getInto(){
$username=request::param('username');
$password=request::param('password');
if(!$username||!$password){
return apiout(config('status.error'),'用户名或密码不得为空');
}
$user=new User();
$userInfo=$user->getDataByName($username);
if(!$userInfo['id']){
return apiout(config('status.error'),'用户名或密码不正确');
}
$md5pass=$userInfo['md5pw'];
if($md5pass!=md5($password)){
return apiout(config('status.error'),'用户名或密码不正确');;
}
$userid=$userInfo['id'];
$menuid=$userInfo['menu_id'];
$username=$userInfo['user_name'];
$token=getToken($userid);//获取token
$userInfo= cache::store('redis')->set(config('redis.user.admin_token').$token,['id'=>$userid,'username'=>$username,'perid'=>$menuid],10800);
Session('user_id', $userid);//兼容旧版
//更新登录数据
$data=['token'=>$token,'last_login_time'=>time()];
try {
$cou=$user->upDateById($userid,$data);
} catch (\Exception $e) {
return apiout(1,'内部错误');
}
cookie(config('redis.user.admin_token'),$token);
return apiout(config('status.success'),'登录成功',['token'=>$token]);
}
}
认证判断控制器 adminbase.php
/*
* @Author: 小超越
* @LastEditors: 小超越工作室
* @desc 后台登录认证
*/
namespace app\controller;
use app\BaseController;
use think\exception\HttpResponseException;
use think\facade\Cache;
class AdminBase extends BaseController
{
public $adminToken=null;
public function initialize(){
parent::initialize();
//判断是否登录
if(empty($this->isLogin())){
return $this->redirect(url('../login/index'),302);
}
}
/**
* @description: 判断是否登录
* @return bool
*/
public function isLogin(){
$token=cookie(config('redis.user.admin_token'));
$this->adminToken=cache::store('redis')->get(config('redis.user.admin_token').$token);
if(empty($this->adminToken)){
return false;
}
return true;
}
public function redirect(...$args){
//因为是被继承的,直接return跳转无效,故使用这个方法
throw new HttpResponseException(redirect(...$args));
}
}
退出控制器loginout.php
/*
* @Author: 小超越
* @LastEditors: 小超越工作室
*/
namespace app\controller;
use app\controller\AdminBase;
use think\facade\Cache;
class Loginout extends AdminBase
{
public function index(){
$token=cookie(config('redis.user.admin_token'));
if(empty($token)){
//return apiout(1,'已退出,无需重复提交');
return redirect(url('../login/index'));
}
cache::store('redis')->delete($token);
cookie(config('redis.user.admin_token'),null);
Session('user_id', null);//兼容旧版
//返回参数由前端作为跳转
// return apiout(config('status.success'),'成功');
//直接由后端作为跳转
return redirect(url('../login/index'));
}
}
后台首页admin.php控制器
namespace app\controller;
class ce extends AdminBase
{
public function index()
{
echo '后台首页';
}
}
因为是实战代码,部分功能使用函数封装,但是不影响整体的架构,将就着看吧~
不过这种方式有个缺点,就是只能继承一个base,那么非常是不利于我们之前分享的《Thinkphp等MVC框架非常优秀的基础架构分层思想》 这种架构的,所以我们还需要优化一下,下篇为大家带来使用中间件的方法!