📝 Sa-Token 异常处:未能获取对应StpLogic,type=user
🧨 异常信息
cn.dev33.satoken.exception.SaTokenException: 未能获取对应StpLogic,type=user
抛出位置:
java
throw new SaTokenException("未能获取对应StpLogic,type="+ loginType)
.setCode(SaErrorCode.CODE_10002);
📌 问题原因分析
编号 | 原因 | 说明 |
---|---|---|
1 | type=user 拼写错误或未定义 |
使用字符串 "user" 作为类型标识时,若未正确注册,或拼写错误(如 user vs user1 ),会导致找不到对应的 StpLogic 实例。 |
2 | 自定义 StpLogic 未初始化 |
若通过 new StpLogic(TYPE) 定义了自定义逻辑,但未在 Spring 容器中初始化,或未主动调用相关方法,会导致 StpLogic 实例为 null 。 |
3 | 未注册到 Sa-Token 全局管理器 | 即使创建了 StpLogic 实例,也必须通过 SaManager.putStpLogic() 注册到 Sa-Token 的全局上下文中,否则无法通过 type=user 查找。 |
✅ 解决方案
方案 1:使用 @PostConstruct
初始化并注册 StpLogic
步骤:
-
在配置类(如
SatokenConfigure.java
)中取消注释以下代码:java@PostConstruct @Order(0) public void registerUserStpLogic() { // 初始化 stpLogic,确保后续能正确获取 StpUserUtil.getStpLogic(); // 注册到 Sa-Token 全局管理器 cn.dev33.satoken.SaManager.putStpLogic(StpUserUtil.stpLogic); }
-
关键点 :
@PostConstruct
确保方法在 Spring 容器初始化后执行。@Order(0)
保证该方法在其他初始化逻辑(如rewriteSaStrategy()
)之前运行。
方案 2:使用 @Component
自动初始化 StpUserUtil
步骤:
-
在
StpUserUtil.java
类上添加注解:java@Component public class StpUserUtil { ... }
-
作用 :
- Spring 会自动加载该类,触发静态属性的初始化(如
StpLogic stpLogic
)。 - 无需手动调用
getStpLogic()
或@PostConstruct
。
- Spring 会自动加载该类,触发静态属性的初始化(如
方案 3:手动调用初始化方法
适用于:不想修改配置或注解的情况。
步骤:
-
在 Spring Boot 启动类中手动调用:
java@SpringBootApplication public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); // 手动触发初始化 StpUserUtil.getStpLogic(); cn.dev33.satoken.SaManager.putStpLogic(StpUserUtil.stpLogic); } }
✅ 验证方法
- 登录测试 :
- 调用
/userLogin
接口登录,确保返回 Token。
- 调用
- 权限校验测试 :
- 调用
/checkUserLogin
接口,请求头中携带 Token。 - 预期结果 :返回
{"data": true, ...}
,表示验证通过。
- 调用
- 日志检查 :
- 检查控制台是否仍有
SaTokenException
抛出。
- 检查控制台是否仍有
🔄 常见错误排查
检查项 | 建议操作 |
---|---|
type=user 是否拼写错误? |
使用常量替代字符串,如 public static final String TYPE = "user"; |
StpUserUtil.stpLogic 是否为 null ? |
在启动日志中打印 StpUserUtil.stpLogic 的值,确认是否初始化成功。 |
是否注册到 SaManager ? |
在 SatokenConfigure.java 中检查 SaManager.putStpLogic(...) 是否执行。 |
Spring 是否加载了 StpUserUtil ? |
确保 @Component 被启用,或手动调用初始化方法。 |
📚 代码示例
StpUserUtil.java
java
@Component
public class StpUserUtil {
public static final String TYPE = "user";
public static StpLogic stpLogic = new StpLogic(TYPE) {
@Override
public String splicingKeyTokenName() {
return super.splicingKeyTokenName() + "-user"; // satoken-user
}
};
public static StpLogic getStpLogic() {
return stpLogic;
}
}
SatokenConfigure.java
java
@Configuration
public class SatokenConfigure {
@PostConstruct
@Order(0)
public void registerUserStpLogic() {
StpUserUtil.getStpLogic(); // 触发初始化
SaManager.putStpLogic(StpUserUtil.stpLogic); // 注册到 Sa-Token
}
}
🧩 补充建议
-
配置文件检查 :
-
在
application.yml
中确认 Sa-Token 的配置是否正确,例如:yamlsa-token: token-name: satoken timeout: 86400
-
-
组件扫描 :
- 确保
StpUserUtil
所在包被 Spring Boot 的组件扫描覆盖(@SpringBootApplication
包路径)。
- 确保
-
多租户场景 :
- 如果使用多
StpLogic
(如user
、admin
),需为每个类型单独注册。
- 如果使用多
✅ 总结
原因 | 解决方案 |
---|---|
type=user 拼写错误 |
使用常量替代字符串 |
StpLogic 未初始化 |
使用 @PostConstruct 或 @Component |
未注册到 SaManager |
调用 SaManager.putStpLogic(...) |