浅谈Java程序加载机制

    大家都知道Java程序在编译时,会将源文件(.java)装换成字节码文件(.class),但是class文件并不是本地的可执行文件。运行Java程序时,首先需要运行Java虚拟机(JVM),然后再把class文件加载到JVM中运行,负责加载class文件的这部分就叫做ClassLoader,今天我们就从这聊聊Java程序的加载机制。

    java.lang包下ClassLoader(将其他类加载进内存)是一个描述加载器的类。JVM本身包含了一个用C++写的加载器Bootstrap ,它负责加载核心JavaClass(即所有java.*开头的类  可以查看 jre-->lib-->rt.jar)。另外JVM还会提供两个加载器,它们都是用Java语言编写的,由Bootstrap加载,其中ExtClassLoader负责加载扩展的Javaclass(例如所有javax.*开头的类和存放在jre-->lib-->ext下的jar包)ApplicationClassLoader负责加载应用程序自身的类。

     当运行一个程序的时候,JVM启动,运行bootstrapclassloader,该ClassLoader加载java核心API(ExtClassLoader和AppClassLoader也在此时被加载),然后调用ExtClassLoader加载扩展API,最后AppClassLoader加载classpath目录下定义的Class,这就是一个程序最基本的加载流程。
    一个类编译后只有一份字节码文件,可见JDK自带三个基本的类加载器,不会同时加载一个类。我们可以通过代码演示说明。   
<span style="font-size:18px;">public class Test {
	
	public static void main(String[] args) {
		//获取Test的类加载器
		ClassLoader  loader=Test.class.getClassLoader();
		//获取类的字节码信息
		Class loaderClass=loader.getClass(); 
		//获取加载器名称
		String className=	loaderClass.getName();
		//打印输出
		System.out.println(className);
        }</span>
<span style="font-size:18px;">  }</span>

运行结果图:

    由于Bootstrap并非Java语言编写,这里会报空,所以单独打印

<span style="font-size:18px;">	//遍历出树形结构
		while(loader!=null){
			System.out.println(loader.getClass().getName());
			loader=loader.getParent();
		}
		//避免出现空指针异常,
		System.out.println(loader);</span>


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个拷贝的Java程序示例: ```java public class Person implements Cloneable { private String name; private int age; private Address address; public Person(String name, int age, Address address) { this.name = name; this.age = age; this.address = address; } public String getName() { return name; } public int getAge() { return age; } public Address getAddress() { return address; } @Override public Object clone() throws CloneNotSupportedException { return super.clone(); } public static void main(String[] args) throws CloneNotSupportedException { Address address = new Address("China", "Beijing"); Person originalPerson = new Person("Tom", 20, address); Person clonedPerson = (Person) originalPerson.clone(); System.out.println("Original Person: " + originalPerson.getName() + " " + originalPerson.getAge() + " " + originalPerson.getAddress()); System.out.println("Cloned Person: " + clonedPerson.getName() + " " + clonedPerson.getAge() + " " + clonedPerson.getAddress()); address.setCity("Shanghai"); System.out.println("Original Person: " + originalPerson.getName() + " " + originalPerson.getAge() + " " + originalPerson.getAddress()); System.out.println("Cloned Person: " + clonedPerson.getName() + " " + clonedPerson.getAge() + " " + clonedPerson.getAddress()); } } class Address { private String country; private String city; public Address(String country, String city) { this.country = country; this.city = city; } public String getCountry() { return country; } public String getCity() { return city; } public void setCity(String city) { this.city = city; } @Override public String toString() { return country + " " + city; } } ``` 这个程序中,我们定义了一个`Person`类和一个`Address`类。`Person`类中包含了一个`Address`对象。在`main`方法中,我们创建了一个`Person`对象`originalPerson`,并将其克隆到`clonedPerson`中。然后我们修改了`address`的`city`属性为"Shanghai",并打印出了两个`Person`对象的信息。 由于拷贝只是拷贝了对象的引用,而不是对象本身,因此我们修改了`address`的`city`属性后,两个`Person`对象的`address`属性都被修改了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值