本篇博客是 JavaWeb 應用服務器端在不同環境下獲取文件路徑的全面總結。
獲取文件路徑後主要應用的場景,讀取 JavaWeb 自定義配置文件、在特定路徑下生成各種類型的文件提供下載......
想必看官也是來找方法的,先看上面目錄有沒有能解決你問題的,如果有就點擊進到對應小節,如果沒有速度關閉,看搜索引擎列出來的下一條記錄吧。
getServletContext().getRealPath("/");
output :E:\workspace\tree\ (tree是我web項目的根目錄)
this.getClass().getClassLoader().getResource("/").getPath(); Thread.currentThread().getContextClassLoader().getResource("/").getPath();
output : E:\workspace\tree\WEB-INF\classes\
request.getServletContext().getRealPath("/");
output:E:\Workspaces\tree\WebContent\
request.getRealPath("");
得到jbossWEB發布臨時目錄 warUrl=.../tmp/deploy/tmp14544test-exp.war/
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());
得到當前文件的磁盤絕對路徑
new File(".").getAbsolutePath();
new File(".").getPath();
new File(application.getRealPath(request.getRequestURI())).getParent()
得到jboss運行目錄 path=C:\jboss-4.0.5.GA\bin\
服務器端的相對地址指的是相對於你的web應用的地址。這個地址是在服務器端解析的(不同於html和javascript中的相對地址,他們是由客戶端浏覽器解析的)
也就是說這時候在jsp和servlet中的相對地址應該是相對於你的web應用,即相對於http: //192.168.0.1/webapp/的。
eg:
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");%>
所有的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應用中的目錄。
另外,應該盡量避免使用類似".","./","../../"等類似的相對該文件位置的相對路徑,這樣當文件移動時,很容易出問題。
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"));
WebApplication的系統文件根目錄是你的weblogic安裝所在根目錄。
例如:如果你的weblogic安裝在c:beaweblogic700.....
那麼,你的文件根路徑就是c:
所以,有兩種方式能夠讓你訪問你的服務器端的文件。
使用絕對路徑:比如將你的參數文件放在c:yourconfigyourconf.properties。
new FileInputStream("yourconfig/yourconf.properties");
使用相對路徑:相對路徑的根目錄就是你的webapplication的根路徑,即WEB-INF的上一級目錄,將你的參數文件放在yourwebappyourconfigyourconf.properties,
new FileInputStream("./yourconfig/yourconf.properties");
b.Tomcat
在類中輸出System.getProperty("user.dir");顯示的是%Tomcat_Home%/bin
c.Resin
不是你的JSP放的相對路徑,是JSP引擎執行這個JSP編譯成SERVLET的路徑為根.比如用新建文件法測試File f = new File("a.htm");
這個a.htm在resin的安裝目錄下
Java中取當前項目的根目錄的方法
//因為類名為"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;
//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); }
在任意class中使用到了getClassLoader():取得該Class對象的類裝載器,對於classLoader 的認識還是在後續工作中接觸後不斷建立起來。
裝載類的過程非常簡單:查找類所在位置,並將找到的Java類的字節碼裝入內存,生成對應的Class對象。
Java的類裝載器專門用來實現這樣的過程,JVM並不止有一個類裝載器,事實上,如果你願意的話,你可以讓JVM擁有無數個類裝載器.
類裝載器自身也是一個類,它也需要被裝載到內存中來,那麼這些類裝載器由誰來裝載呢,總得有個根吧?
確實存在這樣的根,它就是神龍見首不見尾的Bootstrap ClassLoader. 為什麼說它神龍見首不見尾呢?
因為你根本無法在Java代碼中抓住哪怕是它的一點點的尾巴,
盡管你能時時刻刻體會到它的存在,因為java的運行環境所需要的所有類庫,都由它來裝載,而它本身是C++寫的程序,可以獨立運行,可以說是JVM的運行起點,顫抖吧,騷年!
在Bootstrap完成它的任務後,會生成一個AppClassLoader(實際上之前系統還會使用擴展類裝載器ExtClassLoader,它用於裝載Java運行環境擴展包中的類)。
這個類裝載器才是我們經常使用的,可以調用ClassLoader.getSystemClassLoader() 來獲得。
我們假定程序中沒有使用類裝載器相關操作設定或者自定義新的類裝載器,那麼我們編寫的所有java類通通會由它來裝載,值得尊敬吧。
AppClassLoader查找類的區域就是耳熟能詳的Classpath,也是初學者必須跨過的門檻,有沒有靈光一閃的感覺,我們按照它的類查找范圍給它取名為類路徑類裝載器。