本篇博客是 JavaWeb 應用服務器端在不同環境下獲取文件路徑的全面總結。
獲取文件路徑后主要應用的場景,讀取 JavaWeb 自定義配置文件、在特定路徑下生成各種類型的文件提供下載......
想必看官也是來找方法的,先看上面目錄有沒有能解決你問題的,如果有就點擊進到對應小節,如果沒有速度關閉,看搜索引擎列出來的下一條記錄吧。
1.Servlet 的init 方法中獲取
getServletContext().getRealPath("/");
output :E:\workspace\tree\ (tree是我web項目的根目錄)
2.任意的class中獲取
this.getClass().getClassLoader().getResource("/").getPath(); Thread.currentThread().getContextClassLoader().getResource("/").getPath();
output : E:\workspace\tree\WEB-INF\classes\
3.HttpServletRequest 中獲取
request.getServletContext().getRealPath("/");
output:E:\Workspaces\tree\WebContent\
weblogic 下 request.getServletContext() 路徑獲取為空,其原因是 servlet-api.jar 本地和 weblogic 上有差別。
換用 request.getSession().getServletContext().getRealPath("/") 可以解決問題,但這樣的話你需要判斷容器。
最好是取類加載路徑也就是第二種方式,web 相對路徑,然后獲取你想要的資源。
4.jsp取得當前目錄的路徑
request.getRealPath("");
path=C:\jboss-4.0.5.GA\server\default\tmp\deploy\tmp14544test-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,也是初學者必須跨過的門檻,有沒有靈光一閃的感覺,我們按照它的類查找范圍給它取名為類路徑類裝載器。