ContextClassLoader深度讲解.md

举报
西魏陶渊明 发表于 2022/09/25 03:05:21 2022/09/25
【摘要】 Thread.currentThread().getContextClassLoader(); 从方法名字来看,应该是获取当前上下文的类加载器 那么问题来了,为什么要这样设计? 解决了什么样的设计问题? 解决了什么样的开发问题? 我们带着这些问题,听小编娓娓道来 解决委派双亲加载模式的缺点实现了JNDI等解决开发中,文件...

Thread.currentThread().getContextClassLoader();

  • 从方法名字来看,应该是获取当前上下文的类加载器

那么问题来了,为什么要这样设计? 解决了什么样的设计问题? 解决了什么样的开发问题? 我们带着这些问题,听小编娓娓道来

  • 解决委派双亲加载模式的缺点
  • 实现了JNDI等
  • 解决开发中,文件加载不到的异常

  
  1. Thread.currentThread().getContextClassLoader();
  2. this.getClass().getClassLoader();

类加载器之前一直迷惑,终于这个问题在一篇博客的回答中,找到了清晰易懂的解释

原文是这样的:

Thread context class loader存在的目的主要是为了解决parent delegation机制下无法干净的解决的问题。假如有下述委派链:
ClassLoader A -> System class loader -> Extension class loader -> Bootstrap class loader

那么委派链左边的ClassLoader就可以很自然的使用右边的ClassLoader所加载的类。

但如果情况要反过来,是右边的ClassLoader所加载的代码需要反过来去找委派链靠左边的ClassLoader去加载东西怎么办呢?没辙,parent delegation是单向的,没办法反过来从右边找左边.*

类加载器的委派双亲模式?

不明白的同学,赶紧补习一波.

就是说当我们this.getClass().getClassLoader();可以获取到所有已经加载过的文件,
但是System class loader -> Extension class loader -> Bootstrap class loader 就获取不到ClassLoader A 能加载到的信息,那么怎么办呢? 于是,Thread就把当前的类加载器,给保存下来了,其他加载器,需要的时候,就把当前线程的加载器,获取到.

那么什么场景下,会遇到这种情况那,当通常发生在有些JVM核心代码必须动态加载由应用程序开发人员提供的资源时eg:

  • JNDI

JNDI(Java Naming and Directory Interface,Java命名和目录接口)是SUN公司提供的一种标准的Java命名系统接口,JNDI提供统一的客户端API,通过不同的访问提供者接口

在系统中,要调用开发者的资源,此时就遇到了这种情况

  • JAX和rt.jar 因为是两个加载器加载的 那么BootStrap需要加载Ext的资源,怎么办? 这不是与委托机制相反了吗? 所以就不能只依赖委派双亲模式,那么怎么做

然后我们看一波,Thread源码的注释,提供了,获取上下文加载器方法Thread.currentThread().getContextClassLoader()

Thread


  
  1. /* The context ClassLoader for this thread */
  2. private ClassLoader contextClassLoader;

问题:

项目中需要加载应用配置,加载不到,需要用ClassPathResource

问题同上,父加载器要加载应用配置,因此需要调用上下文加载器

代码块分析


  
  1. //获取文件绝对地址,并不是jar里面的文件路径
  2. URL resource = BlmSignature.class.getClassLoader().getResource(keystoreFilePath);
  3. String path = resource.getPath();
  4. //当发布到线上只发布jar文件, 所以就会报异常,找不到
  5. FileInputStream keystoreinputStream=new FileInputStream(path);

  
  1. //正确的做法是,获取到jar包里面的文件,需要注意类加载是否能加载到的问题,
  2. //1. Spring工具
  3. keystorePath = "classpath:"+keystoreFilePath;
  4. ClassPathResource classPathResource = new ClassPathResource(keystorePath);
  5. InputStream keystoreinputStream = classPathResource.getInputStream();
  6. //使用类加载器加载classpath里面的
  7. //指定Thread.currentThread().getContextClassLoader();加载器
  8. SmileClassPathResource smileClassPathResource = new SmileClassPathResource(keystoreFilePath);
  9. InputStream keystoreinputStream=smileClassPathResource.getInputStream();

SmileClassPathResource源码


  
  1. import java.io.BufferedReader;
  2. import java.io.IOException;
  3. import java.io.InputStream;
  4. import java.io.InputStreamReader;
  5. /**
  6. * @Package: org.smileframework.tool.io
  7. * @Description: 加载配置文件
  8. * @author: liuxin
  9. * @date: 2017/12/19 上午9:23
  10. */
  11. public class SmileClassPathResource {
  12. private final String path;
  13. private ClassLoader classLoader;
  14. public SmileClassPathResource(String name) {
  15. this(name, getDefaultClassLoader());
  16. }
  17. public SmileClassPathResource(String name, ClassLoader classLoader) {
  18. this.path = name;
  19. this.classLoader = classLoader;
  20. }
  21. public static ClassLoader getDefaultClassLoader() {
  22. ClassLoader cl = null;
  23. try {
  24. cl = Thread.currentThread().getContextClassLoader();
  25. } catch (Throwable var3) {
  26. ;
  27. }
  28. if(cl == null) {
  29. cl = ClassUtils.class.getClassLoader();
  30. if(cl == null) {
  31. try {
  32. cl = ClassLoader.getSystemClassLoader();
  33. } catch (Throwable var2) {
  34. ;
  35. }
  36. }
  37. }
  38. return cl;
  39. }
  40. public InputStream getInputStream() {
  41. InputStream is;
  42. if (this.classLoader != null) {
  43. is = this.classLoader.getResourceAsStream(this.path);
  44. } else {//当还是加载不到,调用上层加载器
  45. is = ClassLoader.getSystemResourceAsStream(this.path);
  46. }
  47. if (is == null) {
  48. throw new RuntimeException(path + " cannot be opened because it does not exist");
  49. } else {
  50. return is;
  51. }
  52. }
  53. public String getResourceStreamAsString() {
  54. InputStream is = getInputStream();
  55. BufferedReader reader = new BufferedReader(new InputStreamReader(is));
  56. StringBuilder sb = new StringBuilder();
  57. String line = null;
  58. try {
  59. while ((line = reader.readLine()) != null) {
  60. sb.append(line + "\n");
  61. }
  62. } catch (IOException e) {
  63. e.printStackTrace();
  64. } finally {
  65. try {
  66. is.close();
  67. } catch (IOException e) {
  68. e.printStackTrace();
  69. }
  70. }
  71. return sb.toString();
  72. }
  73. public static void main(String[] args) {
  74. SmileClassPathResource classPathResource = new SmileClassPathResource("logback.xml");
  75. System.out.println(classPathResource.getResourceStreamAsString());
  76. }
  77. }

文章来源: springlearn.blog.csdn.net,作者:西魏陶渊明,版权归原作者所有,如需转载,请联系作者。

原文链接:springlearn.blog.csdn.net/article/details/102425342

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。