7

我有一个库需要在我的一个项目中使用,不幸的是它注册了自己的URLStreamHandler库来处理 http- URLs。有没有办法获得对 Java 的默认 http- 和 https- 的引用URLStreamHandlers,所以我可以在 的构造函数中指定其中一个URL来打开标准的 http 连接而不使用库覆盖的协议?

4

3 回答 3

14

找到了:

sun.net.www.protocol.http.Handler

有了这个,我现在可以做:

URL url = new URL(null, "http://...", new sun.net.www.protocol.http.Handler());
HttpURLConnection cxn = (HttpURLConnection) url.openConnection();

我得到了一个普通的 Java HttpURLConnection,而不是库提供的那个。


更新:

我发现了另一种更通用的方法:删除库的URLStreamHandlerFactory!

这有点棘手,因为从技术上讲,URL 类不允许您多次设置工厂或使用官方函数调用清除它,但是通过一点反射魔法,我们无论如何都可以做到:

public static String unsetURLStreamHandlerFactory() {
    try {
        Field f = URL.class.getDeclaredField("factory");
        f.setAccessible(true);
        Object curFac = f.get(null);
        f.set(null, null);
        URL.setURLStreamHandlerFactory(null);
        return curFac.getClass().getName();
    } catch (Exception e) {
        return null;
    }
}

factory此方法获取 -class中的静态字段URL,使其可访问,获取其当前值并将其更改为null。之后,它调用URL.setStreamHandlerFactory( null )(现在完成没有错误)使这个设置“正式”,即给函数一个机会来做它可能想做的任何其他清理。然后返回之前注册的工厂的类名,仅供参考。如果出现任何问题,它会吞下异常(我知道,坏主意......)并返回null

供参考:这是URL.java 的相关源代码

注意:这种方法可能比使用内部 sun-classes 风险更大(就可移植性而言),因为它依赖于 URL-class 的特定内部结构(即factory-field 的存在和确切功能),但是它的优点是我不需要遍历所有代码来查找所有 URL 构造函数并添加处理程序参数......此外,它可能会破坏依赖于其注册处理程序的库的某些功能。幸运的是,这两个问题(可移植性和部分损坏的库功能)都不是与我的案例相关的问题。


更新:#2

当我们使用反射时:这是获取默认处理程序引用的最安全的方法:

public static URLStreamHandler getURLStreamHandler(String protocol) {
    try {
        Method method = URL.class.getDeclaredMethod("getURLStreamHandler", String.class);
        method.setAccessible(true);
        return (URLStreamHandler) method.invoke(null, protocol);        
    } catch (Exception e) {
        return null;
    }
}

然后您只需将其称为:

URLStreamHandler hander = getURLStreamHandler("http");

注意:这个调用需要在库注册它之前URLStreamHandlerFactory发生,否则你最终会引用它们的处理程序。

为什么我会认为这是最安全的方法?因为URL.getURLStreamHandler(...)不是完全私有的方法,而只是包私有的。因此,修改其调用签名可能会破坏同一包中的其他代码。此外,它的名字并没有留下太多空间来返回我们正在寻找的东西之外的任何东西。因此,我认为 -class 的不同/未来实现不太可能(尽管仍然不是不可能)URL与此处所做的假设不兼容。

于 2013-08-02T14:04:16.583 回答
1

这是 Markus 为 Android 重写的解决方案:

public static URLStreamHandler getURLStreamHandler(String protocolIdentifier) {
    try {
        URL url = new URL(protocolIdentifier);
        Field handlerField = URL.class.getDeclaredField("handler");
        handlerField.setAccessible(true);
        return (URLStreamHandler)handlerField.get(url);
     } catch (Exception e) {
        return null;
    }
}

android中不存在getURLStreamHandler方法,所以需要换一种方式。protocolIdentifier 是协议名称加上冒号。您需要传入一个值,以便用于为您想要的协议 URLStreamHandler 实例化一个 URL 实例。如果您想要标准的“jar:”协议处理程序,您必须输入如下内容:“jar:file:!/”。“文件”可以替换为“http”或“https”,因为它们都为您提供相同的处理程序实例。

android支持的标准Handler有http、https、file、jar、ftp。

于 2019-08-01T23:24:36.273 回答
0

你可以注册自己的URLStreamHandlerFactorywirhURL.setURLStreamHandlerFactory(new URLStreamHandlerFactory());

public class URLStreamHandlerFactory implements java.net.URLStreamHandlerFactory {
    public URLStreamHandlerFactory() {}

    public URLStreamHandler createURLStreamHandler(String protocol) {
        if(protocol.equals("http")) {
            return new sun.net.www.protocol.http.Handler();
        } else if(protocol.equals("https")) {
            return new sun.net.www.protocol.https.Handler();
        }
        return null;
    }
}

所以你可以使用标准处理程序

编辑:找到这段代码

于 2013-08-02T14:11:47.597 回答