本篇博客是 JavaWeb 应用服务器端在不同环境下获取文件路径的全面总结。
获取文件路径后主要应用的场景,读取 JavaWeb 自定义配置文件、在特定路径下生成各种类型的文件提供下载......
想必看官也是来找方法的,先看上面目录有没有能解决你问题的,如果有就点击进到对应小节,如果没有速度关闭,看搜索引擎列出来的下一条记录吧。
1.Servlet 的init 方法中获取
getServletContext().getRealPath("/");
output :E:workspace ree (tree是我web项目的根目录)
2.任意的class中获取
this.getClass().getClassLoader().getResource("/").getPath(); Thread.currentThread().getContextClassLoader().getResource("/").getPath();
output : E:workspace reeWEB-INFclasses
3.HttpServletRequest 中获取
request.getServletContext().getRealPath("/");
output:E:Workspaces reeWebContent
weblogic 下 request.getServletContext() 路径获取为空,其原因是 servlet-api.jar 本地和 weblogic 上有差别。
换用 request.getSession().getServletContext().getRealPath("/") 可以解决问题,但这样的话你需要判断容器。
最好是取类加载路径也就是第二种方式,web 相对路径,然后获取你想要的资源。
4.jsp取得当前目录的路径
request.getRealPath("");
path=C:jboss-4.0.5.GAserverdefault mpdeploy mp14544test-exp.war
(String)request.getContextPath();
得到项目(test)应用所在的真实的路径 path=/test
request.getRequestURI();
得到应用所在的真实的路径 path=/test/admin/admindex.jsp
request.getRealPath(request.getServletPath());
得到当前文件的磁盘绝对路径
5.服务器端的地址
服务器端的相对地址指的是相对于你的web应用的地址。这个地址是在服务器端解析的(不同于 html 和 javascript 中的相对地址,他们是由客户端浏览器解析的)
也就是说这时候在 jsp 和 servlet 中的相对地址应该是相对于你的web应用,即相对于http: //192.168.0.1/webapp/的。
Forward:servlet 中的 request.getRequestDispatcher(address); 这个 address 是在服务器端解析。
你要 forward 到 a.jsp 应该这么写:request.getRequestDispatcher(“/user/a.jsp”)
这个 / 相对于当前的 web 应用 webapp,其绝对地址就是:http://192.168.0.1/webapp/user/a.jsp。
sendRedirect:在 jsp 中 <%response.sendRedirect("/rtccp/user/a.jsp");%>
6.客户端的地址
所有的 html 页面中的相对地址都是相对于服务器根目录(http://192.168.0.1/)的。
而不是(跟目录下的该Web应用的目录) http://192.168.0.1/webapp/的。
Html 中的 form 表单的 action 属性的地址应该是相对于服务器根目录 (http://192.168.0.1/) 的。
如果提交到 a.jsp 为:action="/webapp/user/a.jsp" 或 action="<%=request.getContextPath()% >"/user/a.jsp;
提交到 servlet 为 actiom="/webapp/handleservlet" Javascript 也是在客户端解析的,所以其相对路径和 form 表单一样。
因此,一般情况下,在 jsp/html 页面等引用的 css,javascript.action等属性前面最好都加上 <%=request.getContextPath()%>,确保所引用的文件属于 Web 应用。
另外,应该尽量避免使用类似 ".","./","../../" 等类似的相对该文件位置的相对路径,这样当文件移动时,很容易出问题。
7.Servlet中获得当前应用的相对路径和绝对路径
a.根目录所对应的绝对路径:request.getServletPath();
b.文件的绝对路径 :request.getSession().getServletContext().getRealPath(request.getRequestURI())
c.当前web应用的绝对路径 :servletConfig.getServletContext().getRealPath("/");
javax.servlet.http.HttpSession.getServletContext()
javax.servlet.jsp.PageContext.getServletContext()
javax.servlet.ServletConfig.getServletContext()
d..Java 的Class中获得相对路径,绝对路径的方法
根据java.io.File的Doc文挡,可知:默认情况下new File("/")代表的目录为:System.getProperty("user.dir")。
System.out.println(Thread.currentThread().getContextClassLoader().getResource("")); System.out.println(Test.class.getClassLoader().getResource("")); System.out.println(ClassLoader.getSystemResource("")); System.out.println(Test.class.getResource("")); System.out.println(Test.class.getResource("/")); System.out.println(new File("/").getAbsolutePath()); System.out.println(System.getProperty("user.dir"));
8.Web服务器中的Java类获得当前路径
WebApplication的系统文件根目录是你的weblogic安装所在根目录。
例如:如果你的weblogic安装在c:beaweblogic700.....
那么,你的文件根路径就是 c: 所以,有两种方式能够让你访问你的服务器端的文件。
使用绝对路径:比如将你的参数文件放在 c:yourconfigyourconf.properties。
new FileInputStream("yourconfig/yourconf.properties");
使用相对路径:相对路径的根目录就是你的webapplication的根路径,即 WEB-INF 的上一级目录,将你的参数文件放在 yourwebappyourconfigyourconf.properties。
new FileInputStream("./yourconfig/yourconf.properties");
tomcat:在类中输出System.getProperty("user.dir");显示的是%Tomcat_Home%/bin
取当前项目的根目录下资源的方法:
//因为类名为"Application",因此" Application.class"一定能找到 String result = Application.class.getResource("Application.class").toString(); int index = result.indexOf("WEB-INF"); if(index == -1)...{ index = result.indexOf("bin"); } result = result.substring(0,index); if(result.startsWith("jar"))...{ // 当class文件在jar文件中时,返回"jar:file:/F:/ ..."样的路径 result = result.substring(10); }else if(result.startsWith("file"))...{ // 当class文件在class文件中时,返回"file:/F:/ ..."样的路径 result = result.substring(6); } if(result.endsWith("/"))result = result.substring(0,result.length()-1);//不包含最后的"/" return result;
9.JavaScript获取项目根目录
//js获取项目根路径,如: http://localhost:8083/uimcardprj function getRootPath(){ //获取当前网址,如: http://localhost:8083/uimcardprj/share/meun.jsp var curWwwPath=window.document.location.href; //获取主机地址之后的目录,如: uimcardprj/share/meun.jsp var pathName=window.document.location.pathname; var pos=curWwwPath.indexOf(pathName); //获取主机地址,如: http://localhost:8083 var localhostPaht=curWwwPath.substring(0,pos); //获取带"/"的项目名,如:/uimcardprj var projectName=pathName.substring(0,pathName.substr(1).indexOf('/')+1); return(localhostPaht+projectName); }
10.对ClassLoader的一点理解
在任意class中使用到了getClassLoader():取得该Class对象的类装载器,对于classLoader 的认识还是在后续工作中接触后不断建立起来。
装载类的过程非常简单:查找类所在位置,并将找到的Java类的字节码装入内存,生成对应的Class对象。
Java的类装载器专门用来实现这样的过程,JVM并不止有一个类装载器,事实上,如果你愿意的话,你可以让JVM拥有无数个类装载器.
类装载器自身也是一个类,它也需要被装载到内存中来,那么这些类装载器由谁来装载呢,总得有个根吧?
确实存在这样的根,它就是神龙见首不见尾的Bootstrap ClassLoader. 为什么说它神龙见首不见尾呢?
因为你根本无法在Java代码中抓住哪怕是它的一点点的尾巴,
尽管你能时时刻刻体会到它的存在,因为java的运行环境所需要的所有类库,都由它来装载,而它本身是C++写的程序,可以独立运行,可以说是JVM的运行起点,颤抖吧,骚年!
在Bootstrap完成它的任务后,会生成一个AppClassLoader(实际上之前系统还会使用扩展类装载器ExtClassLoader,它用于装载Java运行环境扩展包中的类)。
这个类装载器才是我们经常使用的,可以调用ClassLoader.getSystemClassLoader() 来获得。
我们假定程序中没有使用类装载器相关操作设定或者自定义新的类装载器,那么我们编写的所有java类通通会由它来装载,值得尊敬吧。
AppClassLoader查找类的区域就是耳熟能详的Classpath,也是初学者必须跨过的门槛,有没有灵光一闪的感觉,我们按照它的类查找范围给它取名为类路径类装载器。