From a4c9dc1e68d79dee0e6d165090fd913cc846c697 Mon Sep 17 00:00:00 2001
From: click33 <2393584716@qq.com>
Date: Fri, 21 Feb 2025 02:32:57 +0800
Subject: [PATCH] =?UTF-8?q?feat:=20=E6=96=B0=E5=A2=9E=20SaTokenPlugin=20?=
=?UTF-8?q?=E6=8E=A5=E5=8F=A3=EF=BC=8C=E9=87=8D=E6=9E=84=E6=8F=92=E4=BB=B6?=
=?UTF-8?q?=E4=BD=93=E7=B3=BB?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../dev33/satoken/plugin/SaTokenPlugin.java | 31 +++++++
.../satoken/plugin/SaTokenPluginLoader.java | 45 +++++++++
sa-token-demo/sa-token-demo-test/pom.xml | 9 +-
sa-token-plugin/pom.xml | 2 +
sa-token-plugin/sa-token-jackson/pom.xml | 41 +++++++++
.../dao}/SaJsonTemplateForJackson.java | 49 ++++++++--
.../plugin/impl/SaTokenPluginForJackson.java | 35 +++++++
.../cn.dev33.satoken.plugin.SaTokenPlugin | 1 +
.../satoken/dao/SaTokenDaoRedisJackson.java | 91 +------------------
.../cn.dev33.satoken.plugin.SaTokenPlugin | 0
.../dev33/satoken/spring/SaBeanRegister.java | 22 -----
11 files changed, 209 insertions(+), 117 deletions(-)
create mode 100644 sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java
create mode 100644 sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java
create mode 100644 sa-token-plugin/sa-token-jackson/pom.xml
rename {sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json => sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao}/SaJsonTemplateForJackson.java (59%)
create mode 100644 sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java
create mode 100644 sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin
create mode 100644 sa-token-plugin/sa-token-redis-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin
diff --git a/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java
new file mode 100644
index 00000000..64efc44b
--- /dev/null
+++ b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java
@@ -0,0 +1,31 @@
+/*
+ * Copyright 2020-2099 sa-token.cc
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package cn.dev33.satoken.plugin;
+
+/**
+ * Sa-Token 插件总接口
+ *
+ * @author click33
+ * @since 1.41.0
+ */
+public interface SaTokenPlugin {
+
+ /**
+ * 安装插件
+ */
+ void setup();
+
+}
diff --git a/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java
new file mode 100644
index 00000000..72063a76
--- /dev/null
+++ b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java
@@ -0,0 +1,45 @@
+/*
+ * Copyright 2020-2099 sa-token.cc
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package cn.dev33.satoken.plugin;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.ServiceLoader;
+
+/**
+ * Sa-Token 插件加载器,管理所有插件的加载
+ *
+ * @author click33
+ * @since 1.41.0
+ */
+public class SaTokenPluginLoader {
+
+ public static List pluginList;
+
+ /**
+ * 初始化插件
+ */
+ public static void init() {
+ List list = new ArrayList<>();
+ ServiceLoader plugins = ServiceLoader.load(SaTokenPlugin.class);
+ for (SaTokenPlugin plugin : plugins) {
+ plugin.setup();
+ list.add(plugin);
+ }
+ pluginList = list;
+ }
+
+}
diff --git a/sa-token-demo/sa-token-demo-test/pom.xml b/sa-token-demo/sa-token-demo-test/pom.xml
index f3983dd0..bf6e1a6c 100644
--- a/sa-token-demo/sa-token-demo-test/pom.xml
+++ b/sa-token-demo/sa-token-demo-test/pom.xml
@@ -53,7 +53,14 @@
sa-token-redis
${sa-token.version}
-->
-
+
+
+
+ cn.dev33
+ sa-token-jackson
+ ${sa-token.version}
+
+
cn.dev33
diff --git a/sa-token-plugin/pom.xml b/sa-token-plugin/pom.xml
index a3684dd3..c1a4aab6 100644
--- a/sa-token-plugin/pom.xml
+++ b/sa-token-plugin/pom.xml
@@ -18,6 +18,8 @@
+ sa-token-jackson
+
sa-token-redis
sa-token-redis-jackson
sa-token-redis-fastjson
diff --git a/sa-token-plugin/sa-token-jackson/pom.xml b/sa-token-plugin/sa-token-jackson/pom.xml
new file mode 100644
index 00000000..291293ae
--- /dev/null
+++ b/sa-token-plugin/sa-token-jackson/pom.xml
@@ -0,0 +1,41 @@
+
+
+ 4.0.0
+
+
+ cn.dev33
+ sa-token-plugin
+ ${revision}
+ ../pom.xml
+
+ jar
+
+ sa-token-jackson
+ sa-token-jackson
+ sa-token-jackson
+
+
+
+
+ cn.dev33
+ sa-token-core
+
+
+
+ com.fasterxml.jackson.core
+ jackson-databind
+ true
+
+
+
+ com.fasterxml.jackson.datatype
+ jackson-datatype-jsr310
+ true
+
+
+
+
+
+
diff --git a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json/SaJsonTemplateForJackson.java b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao/SaJsonTemplateForJackson.java
similarity index 59%
rename from sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json/SaJsonTemplateForJackson.java
rename to sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao/SaJsonTemplateForJackson.java
index c5394fd0..7a3125ac 100644
--- a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json/SaJsonTemplateForJackson.java
+++ b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao/SaJsonTemplateForJackson.java
@@ -13,29 +13,47 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
-package cn.dev33.satoken.spring.json;
+package cn.dev33.satoken.dao;
-import cn.dev33.satoken.error.SaSpringBootErrorCode;
import cn.dev33.satoken.exception.SaJsonConvertException;
import cn.dev33.satoken.json.SaJsonTemplate;
import cn.dev33.satoken.util.SaFoxUtil;
import com.fasterxml.jackson.annotation.JsonTypeInfo;
import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.jsontype.BasicPolymorphicTypeValidator;
import com.fasterxml.jackson.databind.jsontype.PolymorphicTypeValidator;
+import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
+import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
+import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
+import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer;
+import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
+import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
+import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer;
+import java.time.LocalDate;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.time.format.DateTimeFormatter;
import java.util.Map;
/**
- * JSON 转换器, Jackson 版实现
+ * JSON 转换器, Jackson 版实现
*
* @author click33
* @since 1.34.0
*/
public class SaJsonTemplateForJackson implements SaJsonTemplate {
+ public static final String DATE_TIME_PATTERN = "yyyy-MM-dd HH:mm:ss";
+ public static final String DATE_PATTERN = "yyyy-MM-dd";
+ public static final String TIME_PATTERN = "HH:mm:ss";
+ public static final DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern(DATE_TIME_PATTERN);
+ public static final DateTimeFormatter DATE_FORMATTER = DateTimeFormatter.ofPattern(DATE_PATTERN);
+ public static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern(TIME_PATTERN);
+
/**
* 底层 Mapper 对象
*/
@@ -63,6 +81,25 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate {
// 2、使空 bean 在序列化时也能记录类型信息,而不是只序列化成 {}
objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
+ // 3、配置 [ 忽略未知字段 ]
+ this.objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
+
+ // 4、配置 [ 时间类型转换 ]
+ JavaTimeModule timeModule = new JavaTimeModule();
+ // LocalDateTime序列化与反序列化
+ timeModule.addSerializer(new LocalDateTimeSerializer(DATE_TIME_FORMATTER));
+ timeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DATE_TIME_FORMATTER));
+ // LocalDate序列化与反序列化
+ timeModule.addSerializer(new LocalDateSerializer(DATE_FORMATTER));
+ timeModule.addDeserializer(LocalDate.class, new LocalDateDeserializer(DATE_FORMATTER));
+ // LocalTime序列化与反序列化
+ timeModule.addSerializer(new LocalTimeSerializer(TIME_FORMATTER));
+ timeModule.addDeserializer(LocalTime.class, new LocalTimeDeserializer(TIME_FORMATTER));
+ //
+ this.objectMapper.registerModule(timeModule);
+
+
+
}
/**
@@ -73,7 +110,7 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate {
try {
return objectMapper.writeValueAsString(obj);
} catch (JsonProcessingException e) {
- throw new SaJsonConvertException(e).setCode(SaSpringBootErrorCode.CODE_20103);
+ throw new SaJsonConvertException(e);
}
}
@@ -89,7 +126,7 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate {
Object value = objectMapper.readValue(jsonStr, Object.class);
return value;
} catch (JsonProcessingException e) {
- throw new SaJsonConvertException(e).setCode(SaSpringBootErrorCode.CODE_20106);
+ throw new SaJsonConvertException(e);
}
}
@@ -106,7 +143,7 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate {
Map map = objectMapper.readValue(jsonStr, Map.class);
return map;
} catch (JsonProcessingException e) {
- throw new SaJsonConvertException(e).setCode(SaSpringBootErrorCode.CODE_20104);
+ throw new SaJsonConvertException(e);
}
}
diff --git a/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java
new file mode 100644
index 00000000..a189e8d3
--- /dev/null
+++ b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java
@@ -0,0 +1,35 @@
+/*
+ * Copyright 2020-2099 sa-token.cc
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package cn.dev33.satoken.plugin.impl;
+
+import cn.dev33.satoken.SaManager;
+import cn.dev33.satoken.dao.SaJsonTemplateForJackson;
+import cn.dev33.satoken.plugin.SaTokenPlugin;
+
+/**
+ * SaToken 插件:JSON 转换器安装
+ *
+ * @author click33
+ * @since 1.41.0
+ */
+public class SaTokenPluginForJackson implements SaTokenPlugin {
+
+ @Override
+ public void setup() {
+ SaManager.setSaJsonTemplate(new SaJsonTemplateForJackson());
+ }
+
+}
\ No newline at end of file
diff --git a/sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin b/sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin
new file mode 100644
index 00000000..488ec78a
--- /dev/null
+++ b/sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin
@@ -0,0 +1 @@
+cn.dev33.satoken.plugin.impl.SaTokenPluginForJackson
\ No newline at end of file
diff --git a/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java b/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java
index 60084954..642ce65c 100644
--- a/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java
+++ b/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java
@@ -17,28 +17,11 @@ package cn.dev33.satoken.dao;
import cn.dev33.satoken.strategy.SaStrategy;
import cn.dev33.satoken.util.SaFoxUtil;
-import com.fasterxml.jackson.databind.DeserializationFeature;
-import com.fasterxml.jackson.databind.ObjectMapper;
-import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
-import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
-import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
-import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer;
-import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
-import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
-import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.connection.RedisConnectionFactory;
-import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
-import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
-import org.springframework.data.redis.serializer.StringRedisSerializer;
import org.springframework.stereotype.Component;
-import java.lang.reflect.Field;
-import java.time.LocalDate;
-import java.time.LocalDateTime;
-import java.time.LocalTime;
-import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
@@ -53,35 +36,11 @@ import java.util.concurrent.TimeUnit;
@Component
public class SaTokenDaoRedisJackson implements SaTokenDao {
- public static final String DATE_TIME_PATTERN = "yyyy-MM-dd HH:mm:ss";
- public static final String DATE_PATTERN = "yyyy-MM-dd";
- public static final String TIME_PATTERN = "HH:mm:ss";
- public static final DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern(DATE_TIME_PATTERN);
- public static final DateTimeFormatter DATE_FORMATTER = DateTimeFormatter.ofPattern(DATE_PATTERN);
- public static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern(TIME_PATTERN);
-
- /**
- * ObjectMapper 对象 (以 public 作用域暴露出此对象,方便开发者二次更改配置)
- *
- * 例如:
- *
- * SaTokenDaoRedisJackson redisJackson = (SaTokenDaoRedisJackson) SaManager.getSaTokenDao();
- * redisJackson.objectMapper.xxx = xxx;
- *
- *
- */
- public ObjectMapper objectMapper;
-
/**
* String 读写专用
*/
public StringRedisTemplate stringRedisTemplate;
- /**
- * Object 读写专用
- */
- public RedisTemplate objectRedisTemplate;
-
/**
* 标记:是否已初始化成功
*/
@@ -93,59 +52,15 @@ public class SaTokenDaoRedisJackson implements SaTokenDao {
if(this.isInit) {
return;
}
-
- // 指定相应的序列化方案
- StringRedisSerializer keySerializer = new StringRedisSerializer();
- GenericJackson2JsonRedisSerializer valueSerializer = new GenericJackson2JsonRedisSerializer();
- // 通过反射获取Mapper对象, 增加一些配置, 增强兼容性
- try {
- Field field = GenericJackson2JsonRedisSerializer.class.getDeclaredField("mapper");
- field.setAccessible(true);
- this.objectMapper = (ObjectMapper) field.get(valueSerializer);
-
- // 配置[忽略未知字段]
- this.objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
-
- // 配置[时间类型转换]
- JavaTimeModule timeModule = new JavaTimeModule();
-
- // LocalDateTime序列化与反序列化
- timeModule.addSerializer(new LocalDateTimeSerializer(DATE_TIME_FORMATTER));
- timeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DATE_TIME_FORMATTER));
-
- // LocalDate序列化与反序列化
- timeModule.addSerializer(new LocalDateSerializer(DATE_FORMATTER));
- timeModule.addDeserializer(LocalDate.class, new LocalDateDeserializer(DATE_FORMATTER));
-
- // LocalTime序列化与反序列化
- timeModule.addSerializer(new LocalTimeSerializer(TIME_FORMATTER));
- timeModule.addDeserializer(LocalTime.class, new LocalTimeDeserializer(TIME_FORMATTER));
-
- this.objectMapper.registerModule(timeModule);
-
- // 重写 SaSession 生成策略
- SaStrategy.instance.createSession = (sessionId) -> new SaSessionForJacksonCustomized(sessionId);
- } catch (Exception e) {
- System.err.println(e.getMessage());
- }
// 构建StringRedisTemplate
StringRedisTemplate stringTemplate = new StringRedisTemplate();
stringTemplate.setConnectionFactory(connectionFactory);
stringTemplate.afterPropertiesSet();
-
- // 构建RedisTemplate
- RedisTemplate template = new RedisTemplate<>();
- template.setConnectionFactory(connectionFactory);
- template.setKeySerializer(keySerializer);
- template.setHashKeySerializer(keySerializer);
- template.setValueSerializer(valueSerializer);
- template.setHashValueSerializer(valueSerializer);
- template.afterPropertiesSet();
-
- // 开始初始化相关组件
this.stringRedisTemplate = stringTemplate;
- this.objectRedisTemplate = template;
+
+ // 重写 SaSession 生成策略
+ SaStrategy.instance.createSession = (sessionId) -> new SaSessionForJacksonCustomized(sessionId);
// 打上标记,表示已经初始化成功,后续无需再重新初始化
this.isInit = true;
diff --git a/sa-token-plugin/sa-token-redis-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin b/sa-token-plugin/sa-token-redis-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin
new file mode 100644
index 00000000..e69de29b
diff --git a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java b/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java
index 4a91bb59..ab8e1e91 100644
--- a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java
+++ b/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java
@@ -16,10 +16,7 @@
package cn.dev33.satoken.spring;
import cn.dev33.satoken.config.SaTokenConfig;
-import cn.dev33.satoken.json.SaJsonTemplate;
-import cn.dev33.satoken.json.SaJsonTemplateDefaultImpl;
import cn.dev33.satoken.spring.context.path.ApplicationContextPathLoading;
-import cn.dev33.satoken.spring.json.SaJsonTemplateForJackson;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
@@ -41,25 +38,6 @@ public class SaBeanRegister {
public SaTokenConfig getSaTokenConfig() {
return new SaTokenConfig();
}
-
- /**
- * 获取 json 转换器 Bean (Jackson版)
- *
- * @return json 转换器 Bean (Jackson版)
- */
- @Bean
- public SaJsonTemplate getSaJsonTemplateForJackson() {
- try {
- // 部分开发者会在 springboot 项目中排除 jackson 依赖,所以这里做一个判断:
- // 1、如果项目中存在 jackson 依赖,则使用 jackson 的 json 转换器
- // 2、如果项目中不存在 jackson 依赖,则使用默认的 json 转换器
- // to:防止因为 jackson 依赖问题导致项目无法启动
- Class.forName("com.fasterxml.jackson.databind.ObjectMapper");
- return new SaJsonTemplateForJackson();
- } catch (ClassNotFoundException e) {
- return new SaJsonTemplateDefaultImpl();
- }
- }
/**
* 应用上下文路径加载器