概述
由于之前一直使用Struts2+Spring开发项目,整合Freemarker时页面如果想直接调用静态方法,可以使用<#assign a= stack.findValue(‘@com.test.package.class@method’)>的方式获取静态方法调用的返回值,现在使用SpringMVC+Freemarker来开发项目时,由于惯性思维的缘故,也想在页面上直接调用静态方法,来获取静态方法的返回值。
- 现有方案
在网上找了一部分的相关资料,只找到类似的方案:,仔细阅读这个方案,大致的思路是会在项目resources下新建一个staticClass.properties文件,在文件内配置所有需要使用的静态类,然后新建一个FreemarkerStaticModels类继承HashMap,用于将所有的静态类配置封装成<静态类名,TemplateHashModel >的形式,然后在Spring配置文件中的对FreeMarkerViewResolver的attributeMap配置成这个FreemarkerStaticModels对象,完成静态方法的配置,页面上可以直接使用静态类名+静态方法名的方法来访问静态方法了。 - 该方法的明显不足
1、如果每次我需要新建一个静态类,就需要在配置文件中新增一个配置,感觉特别麻烦;
2、如果我在配置文件中配置的静态类包名不正确,会导致该静态方法加载异常;
3、如果静态类特别多,这个文件会添加很多配置,这样看起来也不是特别舒服,查找起来也不方便。
新的思路
所以就在原有的基础上进行了修改,大致思路是考虑扫描相关的静态类所处的包,然后将所有的静态类在FreemarkerStaticModels里面进行封装,而不去进行手动的配置,这样大大减少了手动配置的麻烦,同时也简化了开发,当然也会有一些问题,下面也会进行讨论。
具体实现
- 编写FreemarkerStaticModels类
该类是对上面FreemarkerStaticModels类的改造,通过包扫描方式来加载静态资源,这个类里面使用了懒加载模式,包扫描,后置处理器等一些常用的操作,具体实现如下:
import java.io.IOException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.List;
import org.apache.commons.lang3.StringUtils;
import org.apache.log4j.Logger;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.core.io.support.ResourcePatternResolver;
import org.springframework.core.type.classreading.CachingMetadataReaderFactory;
import org.springframework.core.type.classreading.MetadataReader;
import org.springframework.core.type.classreading.MetadataReaderFactory;
import org.springframework.ui.ModelMap;
import org.springframework.util.ClassUtils;
import org.springframework.util.SystemPropertyUtils;
import freemarker.ext.beans.BeansWrapper;
import freemarker.template.TemplateHashModel;
import freemarker.template.TemplateModelException;
public class FreemarkerMap extends ModelMap implements BeanFactoryPostProcessor{
private static final long serialVersionUID = -4675940717727748450L;
private List<String> locations;
private static final String DEFAULT_RESOURCE_PATTERN = "**/*.class";
private static Logger log = Logger.getLogger(FreemarkerMap.class);
public List<String> getLocations() {
return locations;
}
public void setLocations(List<String> locations) {
this.locations = locations;
}
private FreemarkerMap(){}
private static volatile FreemarkerMap instance;
/**
* 懒加载模式
*/
public static FreemarkerMap getInstance(){
if(instance == null){
synchronized (FreemarkerMap.class) {
if(instance == null){
instance = new FreemarkerMap();
}
}
}
return instance;
}
/**
* 后置处理器重新postProcessBeanFactory方法,加载静态类的配置
*/
public void postProcessBeanFactory(
ConfigurableListableBeanFactory beanFactory) throws BeansException {
loadCheckClassMethods(locations);
}
/**
* 根据扫描包的配置
* 加载需要检查的方法
*/
private static void loadCheckClassMethods(List<String> scanPackages) {
ResourcePatternResolver resourcePatternResolver = new PathMatchingResourcePatternResolver();
MetadataReaderFactory metadataReaderFactory = new CachingMetadataReaderFactory(resourcePatternResolver);
for (String basePackage : scanPackages) {
if (StringUtils.isBlank(basePackage)) {
continue;
}
String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX +
ClassUtils.convertClassNameToResourcePath(SystemPropertyUtils.resolvePlaceholders(basePackage)) + "/" + DEFAULT_RESOURCE_PATTERN ;
try {
Resource[] resources = resourcePatternResolver.getResources(packageSearchPath);
for (Resource resource : resources) {
loadClassMethod(metadataReaderFactory, resource);
}
} catch (Exception e) {
log.error("初始化SensitiveWordInterceptor失败", e);
}
}
}
/**
* 加载资源,判断里面的方法
*
* @param metadataReaderFactory spring中用来读取resource为class的工具
* @param resource 这里的资源就是一个Class
* @throws IOException
*/
private static void loadClassMethod(MetadataReaderFactory metadataReaderFactory, Resource resource) throws IOException {
try {
if (resource.isReadable()) {
MetadataReader metadataReader = metadataReaderFactory.getMetadataReader(resource);
if (metadataReader != null) {
String className = metadataReader.getClassMetadata().getClassName();
try {
tryCacheMethod(className);
} catch (ClassNotFoundException e) {
log.error("检查" + className + "是否含有需要信息失败", e);
}
}
}
} catch (Exception e) {
log.error("判断类中的方法实现需要检测xxx失败", e);
}
}
/**
* 把action下面的所有method遍历一次,标记他们是否需要进行xxx验证
* 如果需要,放入cache中
*
* @param fullClassName
* @throws TemplateModelException
*/
private static void tryCacheMethod(String fullClassName) throws ClassNotFoundException, TemplateModelException {
Class<?> clz = Class.forName(fullClassName);
Method[] methods = clz.getDeclaredMethods();
for (Method method : methods) {
int mod = method.getModifiers();
if (Modifier.isStatic(mod)&&Modifier.isPublic(mod)) {
BeansWrapper beansWrapper = BeansWrapper.getDefaultInstance();
TemplateHashModel model = beansWrapper.getStaticModels();
log.debug("已加载"+clz.getName());
instance.put(clz.getSimpleName(), (TemplateHashModel)model.get(clz.getName()));
break;
}
}
}
}
该方法会扫描locations指定路径包及子包下的所有类,同时也支持模糊包匹配,如果该类下面有静态方法,则会将该类加入FreemarkerMap中,如果没有则会跳过,扫描包时会做限制扫描指定路径下的包,而不会全局扫描,防止因为项目太大,扫描时会比较慢,导致项目启动变慢,同时该方法也支持枚举内静态方法的调用。
- Spring配置FreemarkerMap
locations指定扫描包的路径:
<bean id="freemarkerMap" class="com.test.freemarker.FreemarkerMap" factory-method="getInstance">
<property name="locations">
<list>
<value>com.test.common.*.b</value>
<value>com.test.util.*.a</value>
</list>
</property>
</bean>
- 配置FreeMarkerViewResolver
attributeMap指定freemarkerMap的引用
<!-- 要求视图使用FreeMarker模板,指定controller层返回的页面在webapp目录下进行访问,且为html页面-->
<bean class="org.springframework.web.servlet.view.freemarker.FreeMarkerViewResolver">
<property name="prefix">
<value>/</value>
</property>
<property name="suffix">
<value>.html</value>
</property>
<!-- 此处需要声明为utf-8编码,否则即使页面是utf-8编码,中文还是不能正常显示 -->
<property name="contentType" value="text/html;charset=UTF-8"></property>
<property name="attributesMap" ref="freemarkerMap"/>
</bean>
- 页面读取方式
页面直接使用类似于${(StaticUtil.get().name)!”}的读取方式来获取就行了,由于get()方法可以返回一个对象,freemarker可以直接.属性来获取对应属性的值,非常方便。
不足与改进
- 不足
其实这种实现方案自然不是最优的,肯定也会存在一些不足之处:
1、如果一个包下只有一个静态类,要扫描整个包感觉会很笨;
2、如果项目很庞大,包扫描时很有可能会有遗漏,除非有非常明确的包及分层结构;
3、如果项目很庞大,包扫描可能也会导致项目启动变得很慢。 - 优化
可以考虑将原始的版本和目前版本进行整合,如果有的包下只有一个静态类,那么我把这个静态类放到配置文件中进行配置,如果明确某一个包下大部的类都是静态的,比如枚举包,那么可以直接将这个包通过包扫描的方式进行加载,整合之后也可以有效的解决两种方案上的不足。