问题描述:
我在eclipse中调试没有任何问题,现在打成jar包后运行出现错误。原因就是我在程序中使用了一些图片(在工程目录下的gaiji目录下),打成jar包后它好像是在包中找这些资源,我现在要作成这个样子:就是和jar文件同一目录下有一个目录gaiji,执行这个jar调用gaiji目录下的那些图片,这个路径该怎么设置呢。我现在的路径是用URL imgURL = getClass().getResource("/gaiji" + gaiji1.FILENAME)ImageIcon icon = new ImageIcon(imgURL)Image img = icon.getImage()
解析:
在java中,经常要定位某些文件的位置,为了能让程序与物理位置无关,就要使用相对路径。但java中使用相对路径总会遇到一些很麻烦的问题,就是到底相对于哪个参照物的问题。因为我们平时使用相对路径总是相对当前工作目录而言的,但有时需求并非如此。比如,要在一个开发包中使用相对路径,却不知道开发包被其他程序调用时的所在路径,而且特别是在web应用中,很难确定某个文件在整个应用中的相对路径。
所以使用相对路径最好的办法就是让路径相对的参照物是我的开发包或我的应用本身的东西,最好的就是用我开发包中的类的class文件。只要知道了某个class文件的绝对路径,就可以以它为参照物,使用相对路径来定位其他任何文件了。
为了实现这个想法,我写了这个Path类,这个类提供了两个静态公共方法,一个用来定位类的class文件的位置,另一个以某个类为参照物来定位一个相对路径。使用这两个方法,我们可以完全不必理会应用的当前工作路径,随心所欲的根据自己的位置来寻找任何文件。比如在编写某个功能性开发包时,就可以完全不用管调用这个开发包的应用的路径情况,而仅仅根据开发包本身的位置来定位文件,这样很好的实现了封装性,将文件的路径处理完全封闭在了开发包自身之内。
以下是Path类的源代码:
* 创建日期 2004-11-22
*
* 更改所生成文件模板为
* 窗口 >首选项 >Java >代码生成 >代码和注释
*/
package mytools
import java.io.File
import java.io.IOException
import java.MalformedURLException
import java.URL
import java.security.CodeSource
import java.security.ProtectionDomain
/**
* @author 由月
*
* 这个类提供了一些根据类的class文件位置来定位的方法。
*/
public class Path {
/**
* 获取一个类的class文件所在的绝对路径。 这个类可以是JDK自身的类,也可以是用户自定义的类,或者是第三方开发包里的类。
* 只要是在本程序中可以被加载的类,都可以定位到它的class文件的绝对路径。
*
* @param cls
* 一个对象的Class属性
* @return 这个类的class文件位置的绝对路径。 如果没有这个类的定义,则返回null。
*/
public static String getPathFromClass(Class cls) throws IOException {
String path = null
if (cls == null) {
throw new NullPointerException()
}
URL url = getClassLocationURL(cls)
if (url != null) {
path = url.getPath()
if ("jar".equalsIgnoreCase(url.getProtocol())) {
try {
path = new URL(path).getPath()
} catch (MalformedURLException e) {
}
int location = path.indexOf("!/")
if (location != -1) {
path = path.substring(0, location)
}
}
File file = new File(path)
path = file.getCanonicalPath()
}
return path
}
/**
* 这个方法可以通过与某个类的class文件的相对路径来获取文件或目录的绝对路径。 通常在程序中很难定位某个相对路径,特别是在B/S应用中。
* 通过这个方法,我们可以根据我们程序自身的类文件的位置来定位某个相对路径。
* 比如:某个txt文件相对于程序的Test类文件的路径是../../resource/test.txt,
* 那么使用本方法Path.getFullPathRelateClass("../../resource/test.txt",Test.class)
* 得到的结果是txt文件的在系统中的绝对路径。
*
* @param relatedPath
* 相对路径
* @param cls
* 用来定位的类
* @return 相对路径所对应的绝对路径
* @throws IOException
* 因为本方法将查询文件系统,所以可能抛出IO异常
*/
public static String getFullPathRelateClass(String relatedPath, Class cls)
throws IOException {
String path = null
if (relatedPath == null) {
throw new NullPointerException()
}
String clsPath = getPathFromClass(cls)
File clsFile = new File(clsPath)
String tempPath = clsFile.getParent() + File.separator + relatedPath
File file = new File(tempPath)
path = file.getCanonicalPath()
return path
}
/**
* 获取类的class文件位置的URL。这个方法是本类最基础的方法,供其它方法调用。
*/
private static URL getClassLocationURL(final Class cls) {
if (cls == null)
throw new IllegalArgumentException("null input: cls")
URL result = null
final String clsAsResource = cls.getName().replace('.', '/').concat(
".class")
final ProtectionDomain pd = cls.getProtectionDomain()
java.lang.Class contract does not specify
if 'pd' can ever be null
it is not the case for Sun's implementations,
but guard against null
just in case:
if (pd != null) {
final CodeSource cs = pd.getCodeSource()
'cs' can be null depending on
the classloader behavior:
if (cs != null)
result = cs.getLocation()
if (result != null) {
Convert a code source location into
a full class file location
for some mon cases:
if ("file".equals(result.getProtocol())) {
try {
if (result.toExternalForm().endsWith(".jar")
|| result.toExternalForm().endsWith(".zip"))
result = new URL("jar:".concat(
result.toExternalForm()).concat("!/")
.concat(clsAsResource))
else if (new File(result.getFile()).isDirectory())
result = new URL(result, clsAsResource)
} catch (MalformedURLException ignore) {
}
}
}
}
if (result == null) {
Try to find 'cls' definition as a resource
this is not
document.d to be legal, but Sun's
implementations seem to allow this:
final ClassLoader clsLoader = cls.getClassLoader()
result = clsLoader != null ? clsLoader.getResource(clsAsResource)
: ClassLoader.getSystemResource(clsAsResource)
}
return result
}
public static void main(String[] args) {
try {
System.out.println(getPathFromClass(Path.class))
System.out.println(getFullPathRelateClass("../..", Path.class))
} catch (Exception e) {
e.printStackTrace()
}
}
}
程序运行结果:
C:\java\>java Path
C:\java\\Path.class
C:\
在jsp和class文件中调用的相对路径不同。 在jsp里,根目录是WebRoot 在class文件中,根目录是WebRoot/WEB-INF/classes 当然你也可以用System.getProperty("user.dir")获取你工程的绝对路径。另:在Jsp,Servlet,Java中详细获得路径的方法!
1.jsp中取得路径:
以工程名为TEST为例:
(1)得到包含工程名的当前页面全路径:request.getRequestURI()
结果:/TEST/test.jsp
(2)得到工程名:request.getContextPath()
结果:/TEST
(3)得到当前页面所在目录下全名称:request.getServletPath()
结果:如果页面在jsp目录下 /TEST/jsp/test.jsp
(4)得到页面所在服务器的全路径:application.getRealPath("页面.jsp")
结果:D:\resin\webapps\TEST\test.jsp
(5)得到页面所在服务器的绝对路径:absPath=new java.io.File(application.getRealPath(request.getRequestURI())).getParent()
结果:D:\resin\webapps\TEST
2.在类中取得路径:
(1)类的绝对路径:Class.class.getClass().getResource("/").getPath()
结果:/D:/TEST/WebRoot/WEB-INF/classes/pack/
(2)得到工程的路径:System.getProperty("user.dir")
结果:D:\TEST
3.在Servlet中取得路径:
(1)得到工程目录:request.getSession().getServletContext().getRealPath("") 参数可具体到包名。
结果:E:\Tomcat\webapps\TEST
(2)得到IE地址栏地址:request.getRequestURL()
结果:http://localhost:8080/TEST/test
(3)得到相对地址:request.getRequestURI()
结果:/TEST/test
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)