小编典典
找到了:
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获取URL-class中静态字段的保留权,使其可访问,获取其当前值并将其更改为 null
。然后,它调用URL.setStreamHandlerFactory( null
)(现在可以完成而没有错误)以使此设置为“正式”,即为该函数提供机会进行它可能想要进行的其他任何清理。然后它返回先前注册的工厂的类名,仅供参考。如果发生任何错误,它将吞下该异常(我知道,不好的主意…)并返回
null 。
注意: 这种方法可能比使用内部sun类(就可移植性而言)更具风险,因为它依赖于URL类的特定内部结构(即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(...)不是完全私有的方法,而是仅打包私有的。因此,修改其调用签名可能会破坏同一程序包中的其他代码。另外,它的名字实际上并没有留下太多返回我们想要的东西的余地。因此,我希望URL-class
的不同/未来实现不太可能(尽管仍然不是不可能)与此处所作的假设不兼容。
2020-11-01