`

深入理解Java加载类的机制

    博客分类:
  • java
阅读更多

这里从三个方面讲述java classloader

    1,翻译一下classloader的相关技术文章(来自Core Java第二卷的Chapter 9. Security),由于我的英文水平有限,难免有不合适,请大家指出.

    2,将从web应用出发模拟一下web应用的classloader(这节是重点)

    3,通过测试2实现的classloader来加深理解

以下是翻译的关于ClassLoader介绍

ClassLoader
JAVA编译器把源代码转换成一个假想机器(就是我们所说的虚拟机)的语言.虚拟机指令被保存在class后缀的文件里.
每一个类文件包含类和接口的定义以及实现代码.这些类文件必须被一个程序解释,这个程序能够把虚拟机的指令翻译成
宿主机的机器语言.

注意:虚拟机只加载执行一个程序所需要的类文件.举个例子,比如执行MyProgram.class,虚拟机运行的步骤如下:
1,虚拟机有一个加载类文件的机制,比如,从硬盘读取文件或者就网络获得;虚拟机用这个机制加载MyProgram的类文件
2,如果MyProgram有一个实例变量或者是超类,那么实例变量和超类的类文件也被加载.
(加载一个类所依赖的所有类的过程叫做resolving the class-->自己理解吧)
3,然后虚拟机执行MyProgram的main方法(因为是静态方法,所以不需要new MyProgram的实例)
4,如果main 方法或者main方法调用的方法需要其他的类的话,这些类也被加载.

类加载机制不是仅仅用一个类加载器,任何一个java程序至少有以下三个类加载器(为了不影响大家的理解,这里我就不翻译这三个类加载器的名称了)
The bootstrap class loader:加载系统类(有代表性的,jdk的rt.jar里的类),他是虚拟机的必要组成部分,并且一般是用C实现的.
也有类加载器对象(就是指具体的一个类加载器)不关联bootstrap class loader,比如String.class.getClassLoader()返回null.

The extension class loader:加载jre/lib/ext目录下的class,你可以把你的jar文件放到这个目录,extension class loader
将会加载到jar里面的类,即使你不设置classpath.(一些人建议使用这个机制以让你不受classpath的烦扰,不过注意以下的警告)

The system class loader (有时也叫应用程序加载器):加载应用程序类.
他主要加载classpath目录和jar/zip文件里的class,通过设置CLASSPATH环境变量或者是运行java的时候用 [-classpath]选项指定classpath

在SUN的java实现里,the extension and system class loaders都是用java实现的,他们都是URLClassLoader类的实例.

    警告:如果你把jar文件放到jre/lib/ext目录下,并且你的jar文件中的类需要加载一个不是system or extension的类的话,
    你将遇到麻烦.扩展类加载器不使用类路径.如果你想把类放到jre/lib/ext下进行管理的话,请牢记这一点.
    ==>怎么理解这一点:也就是说如果你把自己的x.jar放到jre/lib/ext下的话,如果你自己的x.jar里的class要用到不在
    x.jar里也不在jre/lib/ext的类的话,会导致类加载不了.不难想象吧,因为你x.jar里的类是由extension class loader
    加载的,他不会加载classpath路径下的类.

    警告:把jar文件放到jre/lib/ext目录下,还有第二个缺陷:有时侯,程序员忘记了他很久以前放在这个目录下的类文件.
    当class loader似乎忽略了类路径(其实没有,因为类加载总是先让父的类加载器加载类,只有父的类加载器加载不了的话
    才由自己来加载,"extension class loader是system class loader的父,因此..."),
    而加载了放在扩展目录下的遗忘已久的类的时候,他们就会迷惑不解.

class loader有父子关系,bootstrap class loader以外的每一个class loader都有一个父的类加载器.
类加载器会给父的加载器一个机会加载任何给定的类,如果父加载器加载失败的话自身才去加载.
举个例子,当系统class loader被要求加载一个系统类的时候(比如,java.util.ArrayList),
那么,首先需要extension class loader加载,而extension class loader又先让bootstrap class loader,
最终由bootstrap class loader查找并且加载了rt.jar,其他任何类加载器不需要再搜索.

    注意:当实现一个类加载器的时候,你应该总是授权父加载器去加载类.否则,将会有一些潜在的安全隐患:自定义的类加载器
    可能避开重要的安全检查,意外地加载了系统类.

Applets, servlets, and RMI stubs是用户自定义的类加载器加载的.你甚至可以根据自己的需要写自己的类加载器.
这种方式允许你在传字节码给虚拟机之前实现特殊的安全检查.比如,你可以写一个类加载器拒绝加载没有用"paid for"
表示的类.下一节将展示这么去实现.

大多数时间,你不需要担心类加载器.很多类因为被其他类引用而被加载,这个过程对你来说是透明的.

如果你在程序里调用Class.forName来加载一个类,那么一个新的类被调用Class.forName的类的加载器加载.通常,
这不会有什么问题.然而,在下面的情况下将会失败:
   1.你的lib自己实现了一个有Class.forName方法的类的时候
   2.你的lib里的类的方法被一个不同的类加载器加载的应用程序类所调用的时候(这一先需要好好理解)
   3,被加载的类对于应用程序的类加载器来说是不可见的时候(也就是说classpath下没有相关的类)
  
   这种情况下.库类需要搜索应用程序类加载器(代码如下):
   Thread t = Thread.currentThread();
   ClassLoader loader = t.getContextClassLoader();
   Class cl = loader.loadClass(className);

Using Class Loaders as Namespaces
任何一个java程序员都知道包名是用来消除名字冲突的.在标准类库里有两个叫Date的类(java.util.Date and java.sql.Date).
简单的名字(这里指的是你在程序里直接写Date)只是程序员方便,并且需要包含import语句.在一个运行的程序中,
所有的class都包含他们的包名.

这也许让你吃惊,然而,在同一个虚拟机里面你可以有两个类名和包名都相同的类.一个类是通过他的全名和类加载器来标识的.
This technique is useful for loading code from multiple sources.比如,浏览器为每个web页面使用单独的
applet class loader.这允许虚拟机分开来自不同web页面的类,不管他们是怎么命名的.

    NOTE:这项技术还有其他的用途,比如Sevlet和EJB的"热部署",请参照:http://java.sun.com/developer /TechTips/2000/tt1027.html


Writing Your Own Class Loader
The loadClass method of the ClassLoader superclass takes care of the delegation to
the parent and calls findClass only if the class hasn't already been loaded and
if the parent class loader was unable to load the class.
定义自己的类加载器只需要继承ClassLoader类并且重写findClass(String className)方法.
ClassLoader父类的loadClass方法负责授权给父的类加载器 并且只有在还没有加载并且父的类加载器不能加载的时候
才调用findClass方法.

NOTE:在早期版本的JDK中,程序员必须重写loadClass方法.现在不再建议这种做法.

实现findClass方法必须做下面两件事情,
1,从文件或者其他地方加载类的字节码
2,为了把字节码提交给虚拟机,需要调用ClassLoader类的defineClass方法,

以下是实现一个加载加密类文件的类加载器.

/**  
* This class loader loads encrypted class files.  
*/  
class CryptoClassLoader extends ClassLoader {   
    /**  
     * Constructs a crypto class loader.  
     *   
     * @param k  
     *            the decryption key  
     */  
    public CryptoClassLoader(int k) {   
        key = k;   
    }   
  
    protected Class findClass(String name) throws ClassNotFoundException {   
        byte[] classBytes = null;   
        try {   
            classBytes = loadClassBytes(name);   
        } catch (IOException e) {   
            throw new ClassNotFoundException(name);   
        }   
  
        Class cl = defineClass(name, classBytes, 0, classBytes.length);   
        if (cl == null)   
            throw new ClassNotFoundException(name);   
        return cl;   
    }   
  
    /**  
     * Loads and decrypt the class file bytes.  
     *   
     * @param name  
     *            the class name  
     * @return an array with the class file bytes  
     */  
    private byte[] loadClassBytes(String name) throws IOException {   
        String cname = name.replace('.', '/') + ".caesar";   
        FileInputStream in = null;   
        in = new FileInputStream(cname);   
        try {   
            ByteArrayOutputStream buffer = new ByteArrayOutputStream();   
            int ch;   
            while ((ch = in.read()) != -1) {   
                byte b = (byte) (ch - key);   
                buffer.write(b);   
            }   
            in.close();   
            return buffer.toByteArray();   
        } finally {   
            in.close();   
        }   
    }   
  
    private int key;   
}  

看看怎么实现一个自己的ClassLoader(我们以web应用的classloader为例来讲解)

大家都很熟悉tomcat,比如我们要写一个servlet运行的话.

1,我们通常会在tomcat的webapps目录下建一个自己的web目录(比如myweb),然后让自己的myweb至少具有以下目录结构.

└─WEB-INF
    ├─classes
    └─lib

2,直接把class文件拷贝到classes目录下,或者把自己做的servet打成jar放到lib下

3,启动tomcat就能访问servlet了.

再说一点:不少人了解struts,也做过基于struts的应用.其实struts对于我们来说只是几个jar而已,我们要在自己的应用中用 struts,

只不过就象上面一样把struts的各个lib拷贝到web应用的WEB-INF/lib下就OK了.

再综合前面的基础部分,其实web应用是有自己的classloader的,他专门负责加载WEB-INF/lib和WEB-INF /classes的类。下面我们就来模拟一下web应用的classloader,相信看了下面的代码你就会对classloader的理解又上一个台 阶.

import java.io.File;   
import java.io.FilenameFilter;   
import java.lang.reflect.Method;   
import java.net.MalformedURLException;   
import java.net.URL;   
import java.net.URLClassLoader;   
  
/*  
* web应用的classloader  
* 带你步入classloader的天堂  
*/  
public class WebAppClassLoader{   
  
    URLClassLoader myClassLoader = null;       
    /*  
     * 用法:new WebAppClassLoader(webRoot)  
     *   比如:new WebAppClassLoader("c:/web")  
     */  
    public WebAppClassLoader(String root) {   
        URL[] urls = null;   
        try {   
            urls = getJarURLs(root);   
        } catch (MalformedURLException e) {   
            System.out.println(e.getMessage());   
        }   
        myClassLoader = new URLClassLoader(urls);   
    }   
  
    /*  
     * 获得web应用下的所有class的path的URL.  
     * 1,WEB-INF\classes目录  
     * 2,WEB-INF\lib下的所有jar文件  
     */  
    private URL[] getJarURLs(String rootDirStr) throws MalformedURLException {   
        if (!rootDirStr.endsWith(File.separator)) {   
            rootDirStr += File.separator;   
        }   
           
        // classesDir就是web应用中的"WEB-INF\classes"目录   
        File classesDir = new File(rootDirStr + File.separator + "WEB-INF"  
                + File.separator + "classes");   
           
        // classesDir就是web应用中的"WEB-INF\lib"目录   
        File libDir = new File(rootDirStr + File.separator + "WEB-INF"  
                + File.separator + "lib");   
  
        // 找出目录下所有的jar文件   
        File[] jarFiles = null;   
        if (libDir.isDirectory()) {   
            jarFiles = libDir.listFiles(new FilenameFilter() {   
                public boolean accept(File dir, String name) {   
                    // 注意"dir"参数指的是jar文件的父目录,"name"才是jar文件的   
                    if (dir.isDirectory() // jar文件的父目录必须是一个文件夹   
                            && (name.endsWith(".jar") || name.endsWith(".zip"))) // 注意zip文件也是可以的哦   
                        return true;   
                    return false;   
                }   
            });   
        }   
  
        int jarCount = null == jarFiles?0:jarFiles.length;   
        URL[] urls = new URL[1 + jarCount];   
        urls[0] = classesDir.toURI().toURL();//WEB-INF\classes   
        for (int i = 0; i < jarCount; i++) {//WEB-INF\lib下的所有jar文件   
            urls[i + 1] = jarFiles[i].toURI().toURL();   
        }   
           
        return urls;   
    }   
       
    /*  
     *加载class,直接调用 myClassLoader的loadClass(className)方法  
     */  
    public Class<?> loadClass(String className) throws ClassNotFoundException{   
        return myClassLoader.loadClass(className);   
    }   
       
    /**  
     * 测试WebAppClassLoader  
     * @param args  
     */  
    public static void main(String[] mainArgs) {   
        try {   
            //请根据实际情况指定目录和类名   
            WebAppClassLoader classLoader = new WebAppClassLoader("c:/web/");   
            Class<?> c = classLoader.loadClass("classloader.ButtonTest");   
            //用反射调用main方法   
            String[] args = new String[] {};   
            Method m = c.getMethod("main", args.getClass());   
            m.invoke(null, (Object) args);   
        } catch (Exception e) {   
            // handle exception   
            e.printStackTrace();   
        }   
    }   
}  

测试自己的ClassLoader

1,在c盘建一个web目录(当然了,你可以把上面类的main改一下 使他适合自己的需求),然后子目录如下

└─WEB-INF
    ├─classes
    └─lib

2,新写一个带main方法的类:classloader.ButtonTest(当然了,你可以把上面类的main改一下 使他适合自己的需求),然后打包成jar文件 放到web/WEB-INF/lib下(当然了,你也可以不打包,直接把你的类放到web/WEB-INF/classes下),如果你很懒,就copy 我下面的代码吧,顺便还可以了解了解AWT和SWING;##这个ButtonTest的main方法也可以不象下面这么复杂,就一个 System.out.print也可以的哦

package classloader;   
  
import java.awt.*;   
import java.awt.event.*;   
import javax.swing.*;   
  
public class ButtonTest   
{   
   public static void main(String[] args)   
   {   
      ButtonFrame frame = new ButtonFrame();   
      frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);   
      frame.setVisible(true);   
   }   
}   
  
/**  
   A frame with a button panel  
*/  
class ButtonFrame extends JFrame   
{   
   public ButtonFrame()   
   {   
      setTitle("ButtonTest");   
      setSize(DEFAULT_WIDTH, DEFAULT_HEIGHT);   
  
      // add panel to frame   
  
      ButtonPanel panel = new ButtonPanel();   
      add(panel);   
   }   
  
   public static final int DEFAULT_WIDTH = 300;   
   public static final int DEFAULT_HEIGHT = 200;   
}   
  
/**  
   A panel with three buttons.  
*/  
class ButtonPanel extends JPanel   
{   
   public ButtonPanel()   
   {   
      // create buttons   
  
      JButton yellowButton = new JButton("Yellow");   
      JButton blueButton = new JButton("Blue");   
      JButton redButton = new JButton("Red");   
  
      // add buttons to panel   
  
      add(yellowButton);   
      add(blueButton);   
      add(redButton);   
  
      // create button actions   
  
      ColorAction yellowAction = new ColorAction(Color.YELLOW);   
      ColorAction blueAction = new ColorAction(Color.BLUE);   
      ColorAction redAction = new ColorAction(Color.RED);   
  
      // associate actions with buttons   
  
      yellowButton.addActionListener(yellowAction);   
      blueButton.addActionListener(blueAction);   
      redButton.addActionListener(redAction);   
   }   
  
   /**  
      An action listener that sets the panel's background color.  
   */  
   private class ColorAction implements ActionListener   
   {   
      public ColorAction(Color c)   
      {   
         backgroundColor = c;   
      }   
  
      public void actionPerformed(ActionEvent event)   
      {   
         setBackground(backgroundColor);   
      }   
  
      private Color backgroundColor;   
   }   
}  

3,运行上面的WebAppClassLoader,看看你自己写的,放在web/WEB-INF/lib下的有没有被调用,被调用的话就恭喜你 了。

请注意:不要把你新写的带main方法的类放在WebAppClassLoader所在的工程目录下,如果你这样做的话起不到测试效果,因为这样的 话类不需要自己写的WebAppClassLoader也能被load着的。

http://hi.baidu.com/heroqt/blog/item/94bcb5fdfdfc571b08244d6b.html

分享到:
评论

相关推荐

    Fluent电弧,激光,熔滴一体模拟 UDF包括高斯旋转体热源、双椭球热源(未使用)、VOF梯度计算、反冲压力、磁场力、表面张力,以及熔滴过渡所需的熔滴速度场、熔滴温度场和熔滴VOF

    Fluent电弧,激光,熔滴一体模拟。 UDF包括高斯旋转体热源、双椭球热源(未使用)、VOF梯度计算、反冲压力、磁场力、表面张力,以及熔滴过渡所需的熔滴速度场、熔滴温度场和熔滴VOF。

    基于协同过滤算法商品推荐系统.zip

    基于协同过滤算法商品推荐系统.zip

    锂电池半自动带电液舱标准手套箱(sw16可编辑+工程图)全套技术资料100%好用.zip

    锂电池半自动带电液舱标准手套箱(sw16可编辑+工程图)全套技术资料100%好用.zip

    jquery实现的网页版扫雷小游戏源码.zip

    这是一款基于jQuery实现的经典扫雷小游戏源码,玩家根据游戏规则进行游戏,末尾再在确定的地雷位置单击右键安插上小红旗即可赢得游戏!是一款非常经典的jQuery游戏代码。本源码改进了获胜之后的读数暂停功能。另外建议用户使用支持HTML5与css3效果较好的火狐或谷歌等浏览器预览本源码,可以看到地图的远景拉伸效果。

    Android studio 健康管理系统期末大作业App源码

    Android studio 健康管理系统期末大作业App源码

    校园表白墙网站源码、表白墙网站制作、网页表白墙源码

    校园表白墙网站源码、表白墙网站制作、网页表白墙源码 效果演示https://www.hybiaobai.cn/ 校园表白墙网站源码、表白墙网站制作、网页表白墙源码

    文字生成视频-可灵1.6

    In the video, a person stands alone in a snowy night, holding a delicate wine cup, with a desolate expression. The snowflakes are falling gently, and the person seems lost in deep thoughts and memories. They take a few steps, as if trying to follow the wind, with a sense of yearning and melancholy. The background shows an ancient Chinese-style house with eaves covered in snow, adding to the lonely and nostalgic atmosphere. The person's movements are slow and graceful, reflecting the complex emot

    ①软件 程序 网站开发路面附着系数估计,采用UKF和EKF两种算法 软件为Matlab Simulink,非Carsim联合仿真 dugoff轮胎模块:纯simulink搭非代码 整车模块:7自由

    ①软件 程序 网站开发路面附着系数估计,采用UKF和EKF两种算法。 软件为Matlab Simulink,非Carsim联合仿真。 dugoff轮胎模块:纯simulink搭非代码 整车模块:7自由度整车模型 估计模块:无迹卡尔曼滤波,扩展卡尔曼滤波,均是simulink现成模块应用无需S-function 带有相关文献和估计说明

    基于Spring Boot的在线考试系统--论文.zip

    基于Spring Boot的在线考试系统--论文.zip

    基于多边形逼近与仿射不变量的部分遮挡物体识别算法

    内容概要:本文介绍了一种新方法,用于识别仅由轮廓表示的部分遮挡物体。该方法通过对拐点检测来创建对象的近似多边形形状描述符,并采用一种简单易实施的匹配算法。描述符能够对噪声和部分遮挡保持较好的鲁棒性,在计算机视觉应用中尤其有效。研究涉及多种测试,涵盖人工数据、现实世界图像及不同条件下的变化(如加性高斯噪声、部分遮挡等),展示了良好的效果以及相较于同类方法的优势。 适用人群:从事计算机视觉相关工作的科研人员及技术人员。 使用场景及目标:适用于需要自动化的部分遮挡目标检测和匹配的各种应用场景,尤其是在机器学习项目中涉及光学字符识别等领域。通过使用该算法可以提高复杂环境中物体匹配的成功率,增强系统鲁棒性和适应范围。 其他说明:作者还讨论了关于边界表示法的一些优缺点并提出未来改进方向,例如自动生成迭代次数及引入新的层级化匹配策略。此外,文中提到的所有实验均在标准条件下进行,但当应用于实际环境中时可能需要额外调整参数以达到最佳性能。

    【Python】基于Python的美篇高清图片爬虫.zip

    【Python】基于Python的美篇高清图片爬虫

    node-v14.17.5-x64 msi安装包

    node-v14.17.5-x64 msi安装包

    ie8 升级到ie11 离线安装包

    ie8 升级到ie11 离线安装包 先安装补丁,再安装ie,某个补丁安装不上就跳过,先安装其他补丁,再回来安装。最后能装IE11就可以了

    设计与实现基于JavaWeb的校园兼职信息平台-毕业设计课程设计.zip

    Title: 《设计与实现基于JavaWeb的校园兼职信息平台——毕业设计/课程设计》 项目概述 本项目是一款针对校园环境的兼职信息平台,旨在为学生提供寻找兼职工作的机会,同时为企业提供一个发布兼职信息的平台。该平台采用JavaWeb技术,结合SSM(Spring, SpringMVC, MyBatis)框架开发,专注于解决学生兼职信息不对称的问题。 功能模块 兼职信息发布:企业用户可以发布兼职信息,包括职位描述、要求、薪资等。 兼职信息浏览:学生用户可以浏览兼职信息,并根据条件筛选合适的兼职。 评论与反馈:用户可以对兼职信息和雇主进行评论和反馈。 用户管理:包括学生和企业用户的注册、登录、信息修改等。 消息通知:系统会向用户推送相关的兼职信息和评论通知。 项目特色 评论功能(Comment Part-time):学生可以对企业发布的兼职进行评价,帮助其他学生更好地选择兼职。 信息审核:确保兼职信息的真实性和有效性。 用户互动:提供私信功能,方便学生与企业之间的沟通。 项目目标 帮助学生更快地找到合适的兼职工作。 为企业提供高效的人才招聘渠道。 增强校园内的就业服务和信息交流。 开发流

    基于springboot的应急救援物资管理系统.zip

    基于springboot的应急救援物资管理系统.zip

    用Python开发 Telegram 接口:涵盖用户登录、好友列表及聊天功能-含可运行代码及解释说明

    内容概要:本文档详细讲解了利用 Python 和 python-telegram-bot 库创建一个简易但实用性强的 Telegram 接口的方法。主要内容涵盖了从配置所需环境(如安装相关库)、编写登录验证逻辑,到实现获取好友列表和实施即时通信(聊天)等功能的具体代码演示及解释。文中还提供了关于用户认证的基本方法、简单用户数据模拟、基本的日志记录方式,以及启动机器人并维持监听状态的操作指导,最后提醒开发者替换成自己的 bot token 并指出了一些安全方面的考量,比如严格验证用户输入以保障应用程序的安全性。 适合人群:对于有兴趣探索社交平台集成或是初次接触即时通讯软件自动化构建,尤其是想基于 Python 来快速搭建一个 Telegram Bot 的初学者或是拥有基础编程经验的人士来说非常适合。 使用场景及目标:适用于想要快速建立个人或者小团队之间的信息交流渠道,测试和熟悉 Telegram Bot API 的工作机制,以及进一步理解和提升在社交平台上自动化工具开发技能的情况。这有助于加深理解 API 调用流程、异步消息传输机制等相关知识点,同时也可以作为更大规模项目的基础模块之一来考虑扩展。 其他说明:本指南侧重于理论联系实际的应用层面教学,不仅提供了完整的代码案例让读者可以亲手操作,还强调了良好编码习惯的重要性(像添加适当的注释),并且提及到了未来可能遇到的技术挑战——例如用户数据的真实保存与维护(推荐采用数据库解决方案)。这对于提高读者的实际动手能力和激发更多自主思考都起到了积极作用。

    手搓人工神经网络的教程

    手搓人工神经网络的教程。在CSDN文章中也有,但CSDN文章排版略有偏差,因此附上pdf文档

    回旋提升式柔性链输送机sw16可编辑全套技术资料100%好用.zip

    回旋提升式柔性链输送机sw16可编辑全套技术资料100%好用.zip

    视觉点胶+伺服打螺丝+压装+电测试生产线x_t全套技术资料100%好用.zip

    视觉点胶+伺服打螺丝+压装+电测试生产线x_t全套技术资料100%好用.zip

    基于java的准妈妈孕期交流平台设计新版源码+数据库+说明

    调试过可以运行。 开发语言:Java 框架:springboot JDK版本:JDK1.8 服务器:tomcat7 数据库:mysql 5.7(一定要5.7版本) 数据库工具:Navicat11 开发软件:eclipse/myeclipse/idea Maven包:Maven3.3.9

Global site tag (gtag.js) - Google Analytics