前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >java:解决URL.setURLStreamHandlerFactory只能被jvm调用一次的问题

java:解决URL.setURLStreamHandlerFactory只能被jvm调用一次的问题

作者头像
10km
发布2018-01-03 11:25:27
1.4K0
发布2018-01-03 11:25:27
举报
文章被收录于专栏:10km的专栏

如果你能找到这篇博客,你肯定是为实现URL协议扩展时自定义协议的StreamHandlerFactory注册问题而头痛。 一般而言,URL 的格式是: protocol://[authority]hostname:port/resource?queryString 常见协议头(protocol)有http,https,file。对应不同的协议,java都有提供默认URLStreamHandler对象来解析这些协议,如下图,这些位于rt.jar包中每一个package都对应一种协议,package下都有一个继承自URLStreamHandler的Handler类用于对应协议解析

这里写图片描述
这里写图片描述

如果要实现自己的协议,就需要自己写一个URLStreamHandler,如何写URLStreamHandler与具体项目需求相关,不是本文要讨论的重点。当我们想让自己写的URLStreamHandler生效,就需要将它注册到URL中,这篇文章《Java URL协议扩展实现》详细描述了两种机制,来实现URL协议扩展。 第一种方法就是用URL.setURLStreamHandlerFactory方法将自己的URLStreamHandlerFactory注册到URL类中。我打算采用的就是这种方式,因为这种方式相比jvm参数方式更加可控。 然而,根据URL.setURLStreamHandlerFactory方法的说明以及其代码可知,这个方法具有独占性,在JVM运行时只能被调用一次。(现在看来,这应该算是java的一个设计缺陷) 一般情况下,我们不一定能保证在自己调用URL.setURLStreamHandlerFactory时是第一次,所以调用很有可能失败。 怎么解决这个问题呢?Apache Commons Sandbox提供了一个解决方法,就是commons-jnet,它基本原理就是使用java reflect技术,强行改变URL中的私有成员变量factory(类型为URLStreamHandlerFactory)来保setURLStreamHandlerFactory能被成功调用,并且不破坏原有的factory。 common-jnet代码非常少,只有4个类,没有提供jar包,只是提供源码,从svn上checkout出来加入自己的项目代码就可以使用了

svn checkout http://svn.apache.org/repos/asf/commons/sandbox/jnet/trunk commons-jnet

具体的使用方式,common-jnet的官网上说明得非常明白也非常简单。 http://commons.apache.org/sandbox/commons-jnet/

找到common-jnet之前就发现org.eclipse.osgi中的EquinoxFactoryManager就是用相同的办法解决这个问题的,只是其中的代码混在一起不好摘出来。

参见 EquinoxFactoryManager.installURLStreamHandlerFactory方法和 EquinoxFactoryManager.forceURLStreamHandlerFactory方法的源码

参考资料:

《Java URL协议扩展实现》 apache.sandbox.commons-jnet

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2017年01月11日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档