热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

什么是ClassLoader类加载器

本篇内容主要讲解“什么是ClassLoader类加载器”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“什么是Cl

本篇内容主要讲解“什么是ClassLoader类加载器”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“什么是ClassLoader类加载器”吧!

ClassLoader类加载器

在Java语言中提供了一个系统的环境变量:CLASSPATH,这个环境属性的作用主要是在JVM进程启动时进行类加载路径的定义,在JVM中可以根据类加载器而后进行指定路径中类的加载,也就是说找到了类的加载器就意味着找到了类的来源。

什么是ClassLoader类加载器
ClassLoader

系统类加载器

如果现在要想获得类的加载器,那么一定要通过ClassLoader来获取,而要想获取ClassLoader类的对象,则必须利用Class类(反射的根源)实现,方法:
public ClassLoader getClassLoader()
当获取了ClassLoader后还可以获取其父类的ClassLoader类对象:
public final ClassLoader getParent()
范例:观察类加载器

class Message{}
public class JavaAPIDemo {
    public static void main(String[] args)throws Exception{
        Class clazz=Message.class;System.out.println(clazz.getClassLoader());//获取当前类加载器//1.8:sun.misc.Launcher$AppClassLoader@6659c656//1.9+:jdk.internal.loader.ClassLoaders$AppClassLoader@4f8e5cdeSystem.out.println(clazz.getClassLoader().getParent());//获取父类加载器//1.8:sun.misc.Launcher$ExtClassLoader@60e53b93//1.9+:jdk.internal.loader.ClassLoaders$PlatformClassLoader@5d3411dSystem.out.println(clazz.getClassLoader().getParent().getParent());//获取祖父类加载器 ,null}
}

从JDK1.8之后的版本(JDK1.9,JDK1.10)提供有一个“PlatformClassLoader”类加载器,而在JDK1.8及以前的版本中提供的加载器为“ ExtClassLoader”,因为在JDK的安装目录中提供了一个ext的目录,开发者可以将*.jar文件拷贝到此目录中,这样就可以直接执行了,但是这样的处理开发并不安全,最初的时候也是不提倡使用的,所以从JDK1.9开始将其彻底废除了,同时为了与系统类加载器和应用类加载器之间保持设计的平衡,提供有平台类加载器。

当你获得了类加载器后就可以利用类加载器来实现类的反射加载处理:
protected Class findClass(String name) throws ClassNotFoundException

自定义类加载器

清楚了类加载器的功能后就可以根据自身的需求来实现自定义的类加载器,但是千万要记住一点:自定义的类加载器其加载的顺序是在所有系统类加载器的最后。系统类中的类加载器都是根据CLASSPATH路径进行加载的,而如果有了自定义的类加载器,就可以由开发者任意指定类的加载位置。

什么是ClassLoader类加载器
自定义类加载器

1、随意编写一个程序类,并且将这个类保存在磁盘上。

public class Message {public void send(){
        System.out.println("www.mldn.cn");
    }
}

2、将此类直接拷贝到系统磁盘上(非项目路径)进行编译处理,并且不打包:javac Message.java,此时并没有进行打包处理,所以这个类无法通过CLASSPATH正常加载。
javac /Users/david/Documents/mydir/Message.java
3、自定义一个类加载器,并且继承自ClassLoader类。在ClassLoader类中提供有一个字节转换为类结构的方法:

定义类:
protected final Class defineClass(String name, byte[] b, int off, int len) throws ClassFormatError

import java.io.*;
public class MLDNClassLoader extends ClassLoader {private static final String MESSAGE_CLASS_PATH = "D:" + File.separator + "Message.class";/**
     * 进行指定类的加载
     *
     * @param className 类的完整名称“包.类”
     * @return 返回一个指定类的Class对象
     * @throws Exception 如果类文件不存在,则无法加载
     */public Class loadData(String className) throws Exception {
        byte[] data = loadClassData();//读取二进制数据文件if (data != null) {  //读取到了return super.defineClass(className, data, 0, data.length); 
        }return null;
    }private byte[] loadClassData() throws Exception {//通过文件进行类的加载InputStream input = null;ByteArrayOutputStream bos = null;  //将数据加载到内存之中byte data [] = null;try { 
            bos = new ByteArrayOutputStream();  //实例化内存流input = new FileInputStream(new File(MESSAGE_CLASS_PATH));  //文件加载流input.transferTo(bos);  //读取数据data = bos.toByteArray();  //将所有读取到的字节数取出} catch (Exception e) {
            e.printStackTrace();
        } finally {if (input != null) {
                input.close();
            }if (bos != null) {
                bos.close();
            }
        }return data;
    }
}

4、编写测试类实现类加载控制。

import  java.lang.reflect.Method;import cn.mldn.util.MLDNClassLoader;public class JavaAPIDemo {public static void main(String[] args)throws Exception{
        MLDNClassLoader classLoader = new MLDNClassLoader();  //实例化自定义类加载器Class cls=classLoader.loadData("cn.mldn.util.Message");Object obj = cls.getDeclaredConstructor().newInstance();Method method = cls.getDeclaredMethod("send");method.invoke(obj);
    }
}

如果在以后结合网络程序开发的话,就可以通过一个远程的服务器来确定一个类的功能。

应用项目

5、观察当前的Message类的加载器的情况

public class JavaAPIDemo {
    public static void main(String[] args)throws Exception{
        MLDNClassLoader classLoader = new MLDNClassLoader();//实例化自定义类加载器Class clazz=classLoader.loadData("cn.mldn.util.Message");System.out.println(cls.getClassLoader());  //cn.mldn.util.MLDNClassLoader@6979e8cbSystem.out.println(cls.getClassLoader().getParent()); //jdk.internal.loader.ClassLoaders$AppClassLoader@6659c656System.out.println(cls.getClassLoader().getParent().getParent());  //jdk.internal.loader.ClassLoaders$PlatformClassLoader@763d9750}
}

如果现在定义了一个类:java.lang.String,并且利用了自定义的类加载器进行加载处理,这个类将不会被加载,Java之中针对于类加载器提供有双亲加载机制,如果现在要加载的程序类是由系统提供的类则会由系统类加载器进行加载,在开发者定义的类与系统类名称相同,那么为了保证系统的安全性绝对不会加载。

到此,相信大家对“什么是ClassLoader类加载器”有了更深的了解,不妨来实际操作一番吧!这里是编程笔记网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!


推荐阅读
  • 本文详细解析了Java类加载系统的父子委托机制。在Java程序中,.java源代码文件编译后会生成对应的.class字节码文件,这些字节码文件需要通过类加载器(ClassLoader)进行加载。ClassLoader采用双亲委派模型,确保类的加载过程既高效又安全,避免了类的重复加载和潜在的安全风险。该机制在Java虚拟机中扮演着至关重要的角色,确保了类加载的一致性和可靠性。 ... [详细]
  • 使用 ListView 浏览安卓系统中的回收站文件 ... [详细]
  • 在使用 Qt 进行 YUV420 图像渲染时,由于 Qt 本身不支持直接绘制 YUV 数据,因此需要借助 QOpenGLWidget 和 OpenGL 技术来实现。通过继承 QOpenGLWidget 类并重写其绘图方法,可以利用 GPU 的高效渲染能力,实现高质量的 YUV420 图像显示。此外,这种方法还能显著提高图像处理的性能和流畅性。 ... [详细]
  • 在Java Web服务开发中,Apache CXF 和 Axis2 是两个广泛使用的框架。CXF 由于其与 Spring 框架的无缝集成能力,以及更简便的部署方式,成为了许多开发者的首选。本文将详细介绍如何使用 CXF 框架进行 Web 服务的开发,包括环境搭建、服务发布和客户端调用等关键步骤,为开发者提供一个全面的实践指南。 ... [详细]
  • 本文介绍了如何利用ObjectMapper实现JSON与JavaBean之间的高效转换。ObjectMapper是Jackson库的核心组件,能够便捷地将Java对象序列化为JSON格式,并支持从JSON、XML以及文件等多种数据源反序列化为Java对象。此外,还探讨了在实际应用中如何优化转换性能,以提升系统整体效率。 ... [详细]
  • Java中不同类型的常量池(字符串常量池、Class常量池和运行时常量池)的对比与关联分析
    在研究Java虚拟机的过程中,笔者发现存在多种类型的常量池,包括字符串常量池、Class常量池和运行时常量池。通过查阅CSDN、博客园等相关资料,对这些常量池的特性、用途及其相互关系进行了详细探讨。本文将深入分析这三种常量池的差异与联系,帮助读者更好地理解Java虚拟机的内部机制。 ... [详细]
  • 本文详细解析了使用C++实现的键盘输入记录程序的源代码,该程序在Windows应用程序开发中具有很高的实用价值。键盘记录功能不仅在远程控制软件中广泛应用,还为开发者提供了强大的调试和监控工具。通过具体实例,本文深入探讨了C++键盘记录程序的设计与实现,适合需要相关技术的开发者参考。 ... [详细]
  • Java Socket 关键参数详解与优化建议
    Java Socket 的 API 虽然被广泛使用,但其关键参数的用途却鲜为人知。本文详细解析了 Java Socket 中的重要参数,如 backlog 参数,它用于控制服务器等待连接请求的队列长度。此外,还探讨了其他参数如 SO_TIMEOUT、SO_REUSEADDR 等的配置方法及其对性能的影响,并提供了优化建议,帮助开发者提升网络通信的稳定性和效率。 ... [详细]
  • Python 程序转换为 EXE 文件:详细解析 .py 脚本打包成独立可执行文件的方法与技巧
    在开发了几个简单的爬虫 Python 程序后,我决定将其封装成独立的可执行文件以便于分发和使用。为了实现这一目标,首先需要解决的是如何将 Python 脚本转换为 EXE 文件。在这个过程中,我选择了 Qt 作为 GUI 框架,因为之前对此并不熟悉,希望通过这个项目进一步学习和掌握 Qt 的基本用法。本文将详细介绍从 .py 脚本到 EXE 文件的整个过程,包括所需工具、具体步骤以及常见问题的解决方案。 ... [详细]
  • 在Android平台中,播放音频的采样率通常固定为44.1kHz,而录音的采样率则固定为8kHz。为了确保音频设备的正常工作,底层驱动必须预先设定这些固定的采样率。当上层应用提供的采样率与这些预设值不匹配时,需要通过重采样(resample)技术来调整采样率,以保证音频数据的正确处理和传输。本文将详细探讨FFMpeg在音频处理中的基础理论及重采样技术的应用。 ... [详细]
  • 卓盟科技:动态资源加载技术的兼容性优化与升级 | Android 开发者案例分享
    随着游戏内容日益复杂,资源加载过程已不仅仅是简单的进度显示,而是连接玩家与开发者的桥梁。玩家对快速加载的需求越来越高,这意味着开发者需要不断优化和提升动态资源加载技术的兼容性和性能。卓盟科技通过一系列的技术创新,不仅提高了加载速度,还确保了不同设备和系统的兼容性,为用户提供更加流畅的游戏体验。 ... [详细]
  • Web开发框架概览:Java与JavaScript技术及框架综述
    Web开发涉及服务器端和客户端的协同工作。在服务器端,Java是一种优秀的编程语言,适用于构建各种功能模块,如通过Servlet实现特定服务。客户端则主要依赖HTML进行内容展示,同时借助JavaScript增强交互性和动态效果。此外,现代Web开发还广泛使用各种框架和库,如Spring Boot、React和Vue.js,以提高开发效率和应用性能。 ... [详细]
  • 使用Maven JAR插件将单个或多个文件及其依赖项合并为一个可引用的JAR包
    本文介绍了如何利用Maven中的maven-assembly-plugin插件将单个或多个Java文件及其依赖项打包成一个可引用的JAR文件。首先,需要创建一个新的Maven项目,并将待打包的Java文件复制到该项目中。通过配置maven-assembly-plugin,可以实现将所有文件及其依赖项合并为一个独立的JAR包,方便在其他项目中引用和使用。此外,该方法还支持自定义装配描述符,以满足不同场景下的需求。 ... [详细]
  • 在Java项目中,当两个文件进行互相调用时出现了函数错误。具体问题出现在 `MainFrame.java` 文件中,该文件位于 `cn.javass.bookmgr` 包下,并且导入了 `java.awt.BorderLayout` 和 `java.awt.Event` 等相关类。为了确保项目的正常运行,请求提供专业的解决方案,以解决函数调用中的错误。建议从类路径、依赖关系和方法签名等方面入手,进行全面排查和调试。 ... [详细]
  • C++ 开发实战:实用技巧与经验分享
    C++ 开发实战:实用技巧与经验分享 ... [详细]
author-avatar
手机用户2502859861
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有