在Java中,您可以使用相同的API但使用不同的URL协议加载所有类型的资源:
file:///tmp.txt
http://127.0.0.1:8080/a.properties
jar:http://www.foo.com/bar/baz.jar!/COM/foo/Quux.class
这很好地将资源的实际加载与需要资源的应用程序分离,并且因为URL只是一个String,所以资源加载也很容易配置。
是否有使用当前类加载器加载资源的协议? 这类似于Jar协议,除了我不需要知道资源来自哪个jar文件或类文件夹。
当然,我可以使用Class.getResourceAsStream("a.xml")
来做到这一点,但这需要我使用不同的API,因此需要更改现有代码。我希望能够在所有我可以通过更新属性文件指定资源URL的地方使用它。
答案 0 :(得分:340)
首先,你至少需要一个URLStreamHandler。这实际上将打开与给定URL的连接。请注意,这简称为Handler
;这允许您指定java -Djava.protocol.handler.pkgs=org.my.protocols
并使用“简单”包名称作为支持的协议(在本例中为“classpath”)自动获取它。
new URL("classpath:org/my/package/resource.extension").openConnection();
package org.my.protocols.classpath;
import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLStreamHandler;
/** A {@link URLStreamHandler} that handles resources on the classpath. */
public class Handler extends URLStreamHandler {
/** The classloader to find resources from. */
private final ClassLoader classLoader;
public Handler() {
this.classLoader = getClass().getClassLoader();
}
public Handler(ClassLoader classLoader) {
this.classLoader = classLoader;
}
@Override
protected URLConnection openConnection(URL u) throws IOException {
final URL resourceUrl = classLoader.getResource(u.getPath());
return resourceUrl.openConnection();
}
}
如果您控制代码,则可以执行
new URL(null, "classpath:some/package/resource.extension", new org.my.protocols.classpath.Handler(ClassLoader.getSystemClassLoader()))
这将使用您的处理程序打开连接。
但是再次,这不太令人满意,因为你不需要一个URL来做到这一点 - 你想这样做,因为你不能(或不想)控制的某些lib需要url ...
最终的选择是注册一个URLStreamHandlerFactory
来处理整个jvm的所有网址:
package my.org.url;
import java.net.URLStreamHandler;
import java.net.URLStreamHandlerFactory;
import java.util.HashMap;
import java.util.Map;
class ConfigurableStreamHandlerFactory implements URLStreamHandlerFactory {
private final Map<String, URLStreamHandler> protocolHandlers;
public ConfigurableStreamHandlerFactory(String protocol, URLStreamHandler urlHandler) {
protocolHandlers = new HashMap<String, URLStreamHandler>();
addHandler(protocol, urlHandler);
}
public void addHandler(String protocol, URLStreamHandler urlHandler) {
protocolHandlers.put(protocol, urlHandler);
}
public URLStreamHandler createURLStreamHandler(String protocol) {
return protocolHandlers.get(protocol);
}
}
要注册处理程序,请使用您配置的工厂调用URL.setURLStreamHandlerFactory()
。然后像第一个例子一样new URL("classpath:org/my/package/resource.extension")
然后离开。
请注意,此方法每个JVM只能调用一次,并且请注意Tomcat将使用此方法注册JNDI处理程序(AFAIK)。试试码头(我会);在最坏的情况下,你可以先使用这个方法,然后它必须解决你的问题!
我将此版本发布到公共域,并询问您是否希望修改它在某处启动OSS项目并在此处评论详细信息。更好的实施方式是让URLStreamHandlerFactory
使用ThreadLocal
来为每个URLStreamHandler
存储Thread.currentThread().getContextClassLoader()
。我甚至会给你我的修改和测试课程。
答案 1 :(得分:91)
URL url = getClass().getClassLoader().getResource("someresource.xxx");
应该这样做。
答案 2 :(得分:12)
我认为这值得自己回答 - 如果你使用的是Spring,那么你已经拥有了
Resource firstResource =
context.getResource("http://www.google.fi/");
Resource anotherResource =
context.getResource("classpath:some/resource/path/myTemplate.txt");
如spring documentation中所述,并在skaffman的评论中指出。
答案 3 :(得分:10)
您还可以在启动期间以编程方式设置属性:
final String key = "java.protocol.handler.pkgs";
String newValue = "org.my.protocols";
if (System.getProperty(key) != null) {
final String previousValue = System.getProperty(key);
newValue += "|" + previousValue;
}
System.setProperty(key, newValue);
使用此课程:
package org.my.protocols.classpath;
import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLStreamHandler;
public class Handler extends URLStreamHandler {
@Override
protected URLConnection openConnection(final URL u) throws IOException {
final URL resourceUrl = ClassLoader.getSystemClassLoader().getResource(u.getPath());
return resourceUrl.openConnection();
}
}
因此,你可以采用最少侵入性的方式来做到这一点。 :) java.net.URL将始终使用系统属性中的当前值。
答案 4 :(得分:5)
(与Azder's answer类似,但机会略有不同。)
我不相信有来自类路径的内容的预定义协议处理程序。 (所谓的classpath:
协议)。
但是,Java确实允许您添加自己的协议。这是通过提供具体实施java.net.URLStreamHandler
和java.net.URLConnection
来完成的。
本文介绍如何实现自定义流处理程序: http://java.sun.com/developer/onlineTraining/protocolhandlers/
答案 5 :(得分:5)
我创建了一个类,它有助于减少设置自定义处理程序时的错误,并利用系统属性,因此首先调用方法或不在正确的容器中没有问题。如果你弄错了,还有一个例外类:
CustomURLScheme.java:
/*
* The CustomURLScheme class has a static method for adding cutom protocol
* handlers without getting bogged down with other class loaders and having to
* call setURLStreamHandlerFactory before the next guy...
*/
package com.cybernostics.lib.net.customurl;
import java.net.URLStreamHandler;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* Allows you to add your own URL handler without running into problems
* of race conditions with setURLStream handler.
*
* To add your custom protocol eg myprot://blahblah:
*
* 1) Create a new protocol package which ends in myprot eg com.myfirm.protocols.myprot
* 2) Create a subclass of URLStreamHandler called Handler in this package
* 3) Before you use the protocol, call CustomURLScheme.add(com.myfirm.protocols.myprot.Handler.class);
* @author jasonw
*/
public class CustomURLScheme
{
// this is the package name required to implelent a Handler class
private static Pattern packagePattern = Pattern.compile( "(.+\\.protocols)\\.[^\\.]+" );
/**
* Call this method with your handlerclass
* @param handlerClass
* @throws Exception
*/
public static void add( Class<? extends URLStreamHandler> handlerClass ) throws Exception
{
if ( handlerClass.getSimpleName().equals( "Handler" ) )
{
String pkgName = handlerClass.getPackage().getName();
Matcher m = packagePattern.matcher( pkgName );
if ( m.matches() )
{
String protocolPackage = m.group( 1 );
add( protocolPackage );
}
else
{
throw new CustomURLHandlerException( "Your Handler class package must end in 'protocols.yourprotocolname' eg com.somefirm.blah.protocols.yourprotocol" );
}
}
else
{
throw new CustomURLHandlerException( "Your handler class must be called 'Handler'" );
}
}
private static void add( String handlerPackage )
{
// this property controls where java looks for
// stream handlers - always uses current value.
final String key = "java.protocol.handler.pkgs";
String newValue = handlerPackage;
if ( System.getProperty( key ) != null )
{
final String previousValue = System.getProperty( key );
newValue += "|" + previousValue;
}
System.setProperty( key, newValue );
}
}
CustomURLHandlerException.java:
/*
* Exception if you get things mixed up creating a custom url protocol
*/
package com.cybernostics.lib.net.customurl;
/**
*
* @author jasonw
*/
public class CustomURLHandlerException extends Exception
{
public CustomURLHandlerException(String msg )
{
super( msg );
}
}
答案 6 :(得分:4)
由@Stephen https://stackoverflow.com/a/1769454/980442启发 和http://docstore.mik.ua/orelly/java/exp/ch09_06.htm
使用
new URL("classpath:org/my/package/resource.extension").openConnection()
只需将此类创建到sun.net.www.protocol.classpath
包中,然后将其运行到Oracle JVM实现中,就像魅力一样。
package sun.net.www.protocol.classpath;
import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLStreamHandler;
public class Handler extends URLStreamHandler {
@Override
protected URLConnection openConnection(URL u) throws IOException {
return Thread.currentThread().getContextClassLoader().getResource(u.getPath()).openConnection();
}
}
如果您正在使用其他JVM实现,请设置java.protocol.handler.pkgs=sun.net.www.protocol
系统属性。
答案 7 :(得分:3)
注册URLStreamHandlers的解决方案当然是最正确的,但有时需要最简单的解决方案。所以,我使用以下方法:
/**
* Opens a local file or remote resource represented by given path.
* Supports protocols:
* <ul>
* <li>"file": file:///path/to/file/in/filesystem</li>
* <li>"http" or "https": http://host/path/to/resource - gzipped resources are supported also</li>
* <li>"classpath": classpath:path/to/resource</li>
* </ul>
*
* @param path An URI-formatted path that points to resource to be loaded
* @return Appropriate implementation of {@link InputStream}
* @throws IOException in any case is stream cannot be opened
*/
public static InputStream getInputStreamFromPath(String path) throws IOException {
InputStream is;
String protocol = path.replaceFirst("^(\\w+):.+$", "$1").toLowerCase();
switch (protocol) {
case "http":
case "https":
HttpURLConnection connection = (HttpURLConnection) new URL(path).openConnection();
int code = connection.getResponseCode();
if (code >= 400) throw new IOException("Server returned error code #" + code);
is = connection.getInputStream();
String contentEncoding = connection.getContentEncoding();
if (contentEncoding != null && contentEncoding.equalsIgnoreCase("gzip"))
is = new GZIPInputStream(is);
break;
case "file":
is = new URL(path).openStream();
break;
case "classpath":
is = Thread.currentThread().getContextClassLoader().getResourceAsStream(path.replaceFirst("^\\w+:", ""));
break;
default:
throw new IOException("Missed or unsupported protocol in path '" + path + "'");
}
return is;
}
答案 8 :(得分:2)
我不知道是否已有,但你可以轻松自己做。
不同的协议示例在我看来就像一个外观模式。当每种情况有不同的实现时,您有一个通用接口。
您可以使用相同的原则,创建一个ResourceLoader类,它从您的属性文件中获取字符串,并检查我们的自定义协议
myprotocol:a.xml
myprotocol:file:///tmp.txt
myprotocol:http://127.0.0.1:8080/a.properties
myprotocol:jar:http://www.foo.com/bar/baz.jar!/COM/foo/Quux.class
剥离myprotocol:从字符串的开头开始,然后决定加载资源的方式,然后给你资源。
答案 9 :(得分:2)
Dilums's answer的扩展名:
如果不改变代码,您可能需要像Dilum推荐的那样追求URL相关接口的自定义实现。为了简化您的工作,我建议您查看Spring Framework's Resources的来源。虽然代码不是流处理程序的形式,但它的设计完全符合您的要求,并且符合ASL 2.0许可,使其足够友好,可以在您的代码中重复使用并获得应有的信用。 / p>
答案 10 :(得分:1)
如果您的类路径上有tomcat,就这么简单:
TomcatURLStreamHandlerFactory.register();
这将为“ war”和“ classpath”协议注册处理程序。
答案 11 :(得分:1)
从Java 9+开始,您可以定义一个新的URLStreamHandlerProvider
。 URL
类使用服务加载程序框架在运行时加载它。
创建提供者:
package org.example;
import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLStreamHandler;
import java.net.spi.URLStreamHandlerProvider;
public class ClasspathURLStreamHandlerProvider extends URLStreamHandlerProvider {
@Override
public URLStreamHandler createURLStreamHandler(String protocol) {
if ("classpath".equals(protocol)) {
return new URLStreamHandler() {
@Override
protected URLConnection openConnection(URL u) throws IOException {
return ClassLoader.getSystemClassLoader().getResource(u.getPath()).openConnection();
}
};
}
return null;
}
}
在java.net.spi.URLStreamHandlerProvider
目录中创建一个名为META-INF/services
的文件,内容如下:
org.example.ClasspathURLStreamHandlerProvider
现在,URL类在看到类似以下内容时将使用提供程序:
URL url = new URL("classpath:myfile.txt");
答案 12 :(得分:0)
我尽量避免使用URL
类,而是依赖URI
。因此,对于需要URL
我需要执行Spring资源的事情,比如查找Spring,我会执行以下操作:
public static URL toURL(URI u, ClassLoader loader) throws MalformedURLException {
if ("classpath".equals(u.getScheme())) {
String path = u.getPath();
if (path.startsWith("/")){
path = path.substring("/".length());
}
return loader.getResource(path);
}
else if (u.getScheme() == null && u.getPath() != null) {
//Assume that its a file.
return new File(u.getPath()).toURI().toURL();
}
else {
return u.toURL();
}
}
要创建URI,您可以使用URI.create(..)
。这种方式也更好,因为您控制将执行资源查找的ClassLoader
。
我注意到其他一些答案试图将URL解析为String来检测方案。我认为最好传递URI并使用它来解析。
答案 13 :(得分:0)
在Spring Boot应用程序中,我使用以下内容获取文件URL
Thread.currentThread().getContextClassLoader().getResource("PromotionalOfferIdServiceV2.wsdl")