造一个方形轮子文章目录:造一个方形的轮子
01、解决遗留问题
最后的最后,来解决一下静态文件的问题吧,这个问题在9里提出来的,但做的时候才发现其实从加进来DisplayServlet进来开始就已经出问题了,只是我一直没有用到,所以没发现,具体的问题是因为我把DisplayServlet绑定到了"/"路径上,导致所以的请求都被DisplayServlet接收了,包括静态文件,然后按它的逻辑跑一遍肯定是找不到对应的Controller的,最后就返回了404。
解决问题的梳理:
1、指定tomcat的webapp目录为静态资源目录
2、使用DefaultServlet处理静态资源
3、打包后(静态资源在jar包里)处理静态资源访问问题
4、分别验证开发阶段及打包运行阶段静态资源/接口是否可用
指定不用静态目录重新设置一下就可以,之前一直使用的是随机生成的临时目录,而且还没有删除,时间长了会有很多垃圾目录,看了SpringBoot的源码,原来文件还可以在JVM销毁的时候删除,果然Java里还有很多可以学习的地方,直接上代码:
ClassesPathUtil.java:
public class ClassesPathUtil {
private static final Logger log = LoggerFactory.getLogger(ClassesPathUtil.class);
/**
* 项目目录(.../classes)
*/
private String projectPath;
/**
* 静态资源目录(.../classes/public)
*/
private String publicPath;
public ClassesPathUtil(Class clzz){
String basePath = clzz.getResource("").getPath();
log.info("basePath+++++{}", basePath);
// ..../classes
if(basePath.indexOf("classes")>0) {
projectPath = basePath.substring(0, basePath.indexOf("classes")+7);
publicPath = basePath.substring(0, basePath.indexOf("target"));
publicPath = publicPath+"target/classes/public";
} else {
projectPath = basePath.substring(0, basePath.indexOf("!")+1);
publicPath = SquareApplication.TEMP_TOMCAT_DIR;
}
}
}
主要改了设置publicPath目录的地方,因为开发阶段设置的是编译后的public目录,而打包运行的时候直接指定临时目录,哪jar包里的静态资源如何访问吗?接着往下看:
SquareApplication.java:
public class SquareApplication {
//....
private static String SUFFIXS = "htm,html,css,js,jpg,jpeg,png,gif,json,mp4,mp3,avi,rm";
public static String TEMP_TOMCAT_DIR = "/tmp/square/tomcat";
public static void run(Class clzz, String[] args) {
try {
//...
tomcat = new Tomcat();
tomcat.setPort(TOMCAT_PORT);
// 设置Tomcat工作目录
File f = new File(TEMP_TOMCAT_DIR);
if(!f.exists()){
f.mkdirs();
}
f.deleteOnExit();
tomcat.setBaseDir(f.getPath());
Context context = tomcat.addWebapp(CONTEXT_PATH, classesPathUtil.getPublicPath());
// jar包中静态文件特殊处理
if(classesPathUtil.getProjectPath().indexOf("!")>0) {
String jar = "jar:"+classesPathUtil.getProjectPath()+"/";
URL url = new URL(jar);
context.setResources(new StandardRoot(context));
context.getResources().createWebResourceSet(WebResourceRoot.ResourceSetType.RESOURCE_JAR, "/", url, "/public");
}
// 添加DsipatcherServlet
Wrapper wrapper = Tomcat.addServlet(context, "DispatcherServlet", new DispatcherServlet());
wrapper.addMapping("/");
// 设置静态资源走默认Servlet处理
Tomcat.addServlet(context, "DefaultServlet", new DefaultServlet());
String[] suffixs = SUFFIXS.split(",");
for (String suffix : suffixs) {
context.addServletMappingDecoded("*."+suffix, "DefaultServlet");
}
// 执行这句才能支持JDNI查找
tomcat.enableNaming();
tomcat.getConnector().setURIEncoding(ENCODING);
tomcat.start();
log.info("Tomcat started on port(s): {} with context path '{}'", TOMCAT_PORT, CONTEXT_PATH);
log.info("Started Application in {} ms." , (System.currentTimeMillis() - startTime));
// 保持服务器进程
tomcat.getServer().await();
} catch (Exception e) {
if(e instanceof SquareException){
log.error(((SquareException) e).getMsg());
}
log.error("Application startup failed...", e);
}
}
}
代码都有注释逻辑不复杂,Tomcat目录设置还是使用临时目录,同时调用File的deleteOnExit()方法使目录在JVM销毁时删除临时目录
然后是设置webapp目录,根据上边的ClassesPathUtil代码可知,如果是开发阶段那么设置的是*classes/public目录,如果是打完jar包运行的话设置的是临时目录
再下边是判断如果当前在jar包中运行则将jar包的静态资源添加到容器的WebResource中,以达到在访问jar包中静态资源的目的
最后将默认的DefaultServlet注册到容器中,同时设置需要DefaultServlet处理的静态资源后缀,这样就解决了我们上边提出的问题
测试效果(square-demo项目):
访问静态资源:
浏览器请求:http://localhost:8765/square-demo/square.html
返回页面:success!!!
访问接口:
浏览器请求:http://localhost:8765/square-demo/id?id=1
返回页面:success! name is : ixx
02、后记
从六月份开始第一篇到现在已经过去了三个月了,从更新频率上能看的出来后边明显更新变慢了,多种原因吧,总算是完成了这个系列,虽然有很多的不足,以后如果有机会再做修改吧,至少写这个的最初上的我达到了,让自己多思考每天用的轮子到底是怎么转的。
虽说叫方形的轮子可写完才觉得真的这玩意连别说方形了,连轮子都算不上,一个能称为轮子的框架真的有太多需要考虑的点了,而且每一个点都值得深入研究学习一下,这也是后边做业务之余要去学习的一个方向吧。
03、遗留问题
有人看吗 = =!
本篇代码地址: https://github.com/iuv/square/tree/square11
演示项目地址: https://github.com/iuv/square-demo
本文作者: ixx
本文链接: http://jianpage.com/2019/09/20/square11
版权声明: 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。转载请注明出处!