(4)java的和spring的资源管理器的应用

1.java访问资源的方式

比如现在有一个资源文件application.properties,现在要得到该配置文件的流,该文件放在resouces目录下,文件内容如下

spring.application.name = spring-resources

通过java方式的加载资源,打印输出流

/**
 * @Project: spring
 * @description:  java 加载资源的方式
 * @author: sunkang
 * @create: 2018-09-23 22:16
 * @ModificationHistory who      when       What
 **/
public class FileLoadDemo {

    public static void main(String[] args) throws IOException {
        //1.用类加载器来实现,不过这个是直接加载编译好的classpath路劲的
        InputStream inputStream= FileLoadDemo.class.getClassLoader().getResourceAsStream("application.properties");
        System.out.println(inputStream);

        //2.通过绝对路劲的方式来加载
        File file = new File("");
        //file.getAbsolutePath() 得到的是user.dir,工作路径,跟下面表示方法一样
        System.out.println(System.getProperty("user.dir"));
        //spring-resources为一个模块,所以这里需要加上模块的路径
        File resouceFile = new File("spring-resources/src/main/resources/application.properties");
        InputStream ins=  new BufferedInputStream(new FileInputStream(resouceFile));
        System.out.println(ins);

        //3通过nio加载
        InputStream nos= Files.newInputStream(Paths.get("spring-resources/src/main/resources/application.properties"));
        System.out.println(nos);

        //4 通过URL的方式
        URL fileURL = file.toURI().toURL();
        URLConnection urlConnection = fileURL.openConnection();
        InputStream inputStreamFromURL = urlConnection.getInputStream();
        System.out.println(inputStreamFromURL);
    }
}

这里面着重讲解第四种方式
通过URL来定义资源的位置,该资源可以是文件,jar包,或者网络资源。通过不同前缀名来代表每一种资源的协议,具体请求而的时候由不同的处理协议的handler进行处理
举个例子:


/**
 * java 访问资源可以通过URL这个对象来访问各种资源,实际上URL为了得到inpustream需要以下的过程
*        URL -> URLConnection -> URLStreamHandler -> InputStream
*       这里用了委派模式,获取inputStream会先要获取URLConnection,而URLConnection是由URLStreamHandler创建而来
*       下面秒速了java支持的集中协议,可以rt.jar下在sun.net.www.protocol找到支持的协议模式
*      URL url = new URL("https://www.baidu.com"); // https 协议
*       URL ftpURL = new URL("ftp://ftp.baidu.com"); // ftp 协议
*       URL jar = new URL("jar://jar.baidu.com"); // jar 协议
*        file URLStreamHandler   = sun.net.www.protocol.file.Handler
*        http URLStreamHandler  =  sun.net.www.protocol.http.Handler
*        https URLStreamHandler  = sun.net.www.protocol.https.Handler
*        jar URLStreamHandler  = sun.net.www.protocol.jar.Handler
*        ftp URLStreamHandler  = sun.net.www.protocol.ftp.Handler
*        模式 URLStreamHandler =  sun.net.www.protocol.${protocol}.Handler
 *
 */
public class FileDemo {
    public static void main(String[] args) throws Exception {
        File file = new File("spring-resources/src/main/resources/application.properties");
        URL fileURL = file.toURI().toURL();
        URLConnection urlConnection = fileURL.openConnection();
        InputStream inputStreamFromURL = urlConnection.getInputStream();
        //spring-core  核心包的工具类,把流的内容转成字符串
        String content = StreamUtils.copyToString(inputStreamFromURL, Charset.forName("UTF-8"));
        System.out.println(inputStreamFromURL);
    }
}

从下面的这个图可以看出URL和其他接口的关系

image.png

在URL源码部分中的getURLStreamHandler的方法中,存在如下的代码段

if (handler == null) {
                String packagePrefixList = null;

                packagePrefixList
                    = java.security.AccessController.doPrivileged(
                    new sun.security.action.GetPropertyAction(
                        protocolPathProp,""));
                if (packagePrefixList != "") {
                    packagePrefixList += "|";
                }

                // REMIND: decide whether to allow the "null" class prefix
                // or not.
                packagePrefixList += "sun.net.www.protocol";

                StringTokenizer packagePrefixIter =
                    new StringTokenizer(packagePrefixList, "|");

                while (handler == null &&
                       packagePrefixIter.hasMoreTokens()) {

                    String packagePrefix =
                      packagePrefixIter.nextToken().trim();
                    try {
                        String clsName = packagePrefix + "." + protocol +
                          ".Handler";
                        Class<?> cls = null;
                        try {
                            cls = Class.forName(clsName);
                        } catch (ClassNotFoundException e) {
                            ClassLoader cl = ClassLoader.getSystemClassLoader();
                            if (cl != null) {
                                cls = cl.loadClass(clsName);
                            }
                        }
                        if (cls != null) {
                            handler  =
                              (URLStreamHandler)cls.newInstance();
                        }
                    } catch (Exception e) {
                        // any number of exceptions can get thrown here
                    }
                }
            }

基本上可以了解到是根据 "sun.net.www.protocol" + 协议名称+ ".handler" 得到一个类的全限定名,然后通过Class.forName(clsName)来得到类,如果出错,则用引导类加载器来加载类,判断cls不为空,cls.newInstance()用反射创建一个类

在java的sun.misc.Launcher有用到URLStreamHandlerFactory的工厂,ExtClassLoader和AppClassLoader的有用到这个URLStreamHandlerFactory工厂,有兴趣的源码可以研究下

2.拓展java的classpath路劲加载协议

要模仿创建一个类名为 sun.net.www.protocol.classpath.hanlder的类,该类如下:

/**
 * Classpath 协议 Handler
 *
 */
public class Handler extends URLStreamHandler {
    private final String PROTOCOL_PREFIX = "classpath:/";
    @Override
    protected URLConnection openConnection(URL url) throws IOException {
        // 比如 url = "classpath:/META-INF/license.txt"
        // classpath  = META-INF/license.txt
        // 移除前缀 classpath:/
        // classpath:/META-INF/license.txt
        String urlString = url.toString();
        // META-INF/license.txt
        String classpath = urlString.substring(PROTOCOL_PREFIX.length());
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        URL classpathURL = classLoader.getResource(classpath);
        // 委派给 ClassLoader 实现
        return classpathURL.openConnection();
    }
}

测试类:

public class ClassPathUrlTest {
    public static void main(String[] args) throws IOException {
        URL url = new URL("classpath:/application.properties");
        URLConnection urlConnection = url.openConnection();

        InputStream inputStreamFromURL = urlConnection.getInputStream();

        String content = StreamUtils.copyToString(inputStreamFromURL,     Charset.forName("UTF-8"));

        System.out.println(content);
    }
}

测试结果如下:

spring.application.name = spring-resources

3.spring的加载方式 (DefaultResourceLoader)

/**
 * @Project: spring
 * @description:  spring 默认的加载资源的方式
 * @author: sunkang
 * @create: 2018-09-23 16:51
 * @ModificationHistory who      when       What
 **/
public class SprignResouceLoadTest {

    public static void main(String[] args) throws IOException {
//        ApplicationContext context = new ClassPathXmlApplicationContext();
//        context.getResource("application.properties")
        //ClassPathXmlApplicationContext继承了DefaultResourceLoader,所以实际上DefaultResourceLoader在处理
        //用classPath
        //默认的加载器
        ResourceLoader recourceLoder = new DefaultResourceLoader();
        Resource resource =  recourceLoder.getResource("application.properties");
        InputStream ins= resource.getInputStream();
        System.out.println(ins);

        //加载 classpath路径下的文件
        Resource classpathResource =  recourceLoder.getResource("classpath:application.properties");
        System.out.println(classpathResource.getInputStream());

        //通过file文件协议加载资源文件
        Resource fileResource =  recourceLoder.getResource("file:D:/Eclipse2018Data/personProject/spring/spring-resources/src/main/resources/application.properties");
        System.out.println(fileResource.getInputStream());

        //通过https协议加载资源文件
        Resource httpResource =  recourceLoder.getResource("https://start.spring.io/");
        System.out.println(fileResource.getInputStream());

    }
}

4.基于spring的拓展协议

/**
 * spring  拓展协议举例
 *
 */
public class ResourceDemo {
    public static void main(String[] args) throws IOException {
        // Resource
        // FileSystemResource
        // ClasspathResource
        DefaultResourceLoader resourceLoader = new DefaultResourceLoader();
        // 添加一个protocol = "cp" 处理
        resourceLoader.addProtocolResolver(new ProtocolResolver() {
            private static final String PROTOCOL_PREFIX = "cp:/";
            @Override
            public Resource resolve(String location, ResourceLoader resourceLoader) {
                if (location.startsWith(PROTOCOL_PREFIX)) {
                    // application.properties
                    String classpath = ResourceLoader.CLASSPATH_URL_PREFIX +
                            location.substring(PROTOCOL_PREFIX.length());
                    // cp:/application.properties -> classpath:application.properties
                    return resourceLoader.getResource(classpath);
                }
                return null;
            }
        });
        Resource resource =
                resourceLoader.getResource("cp:/application.properties");

        InputStream inputStream = resource.getInputStream();
        String content = StreamUtils.copyToString(inputStream, Charset.forName("UTF-8"));
        System.out.println(content);
    }
}

看DefaultResourceLoader的源码可以了解到,先由添加的resourceLoader.addProtocolResolver()的协议一个个遍历先解析,解析到了就返回,cp拓展的协议的实现是委派给了ClassPathResource去解析了

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容