java反射初学习

java反射初学习

“程序运行时,允许改变程序结构或变量类型,这种语

言称为动态语言”,如Python,Ruby是动态语言;显然

C++,Java,C#不是动态语言,但是JAVA有着一个非常

突出的动态相关机制:Reflection。

JAVA反射机制是在运行状态中,对于任意一个类,都能

够知道这个类的所有属性和方法;对于任意一个对象,

都能够调用它的任意一个方法和属性;这种动态获取的

信息以及动态调用对象的方法的功能称为java语言的反

射机制,很多优秀的开源框架都是通过反射完成的。

Java反射机制,可以实现以下功能:

①在运行时判断任意一个对象所属的类;

②在运行时构造任意一个类的对象;

③在运行时判断任意一个类所具有的成员变量和方法;

④在运行时调用任意一个对象的方法;

⑤生成动态代理;

可变参数

/**

\* 入门级示例:通过对象获取 包名.类名 

\* @author Administrator 

*/ 

public class Simple { 

public static void main(String[] args) { 

Simple s=new Simple(); 

System.out.println(s.2.getName()); 

} 

}

当我们能够确定一系列参数的类型,类型必须是统一

的, 但是我们确定不了参数的个数的时候,我们可以使

用可变参数。可变代表个数可变[0,+),如果一个方法

中带有可变参数,这个可变参数只能在参数列表最后。

获取源头Class(重点)

所有类的对象其实都是Class**的实例。**这个 Class 实例

可以理解为类的模子,就是包含了类的结构信息,类似

于图纸。我们日常生活中,需要创造一个产品,如想山

寨一个iphone手机,怎么办?有三种方式可以实现:

⑴买个iphone手机,拆的七零八落的,开始山寨;

⑵到iphone工厂参观,拿到iphone磨具,开始山寨;

⑶跑到美国盗取iphone的图纸,开始山寨,最后一种最

暴力,最爽。

同理,获取类的Class对象也有三种方式

⑴Class.forName(”包名.类名”) //一般尽量采用该形式

(2)类.class

(3)对象.getClass()

类加载器(了解)

类的生命周期

在一个类编译完成之后,下一步就需要开始使用类,如

果要使用一个类,肯定离不开JVM。在程序执行中JVM通

装载链接初始化这3个步骤完成。

从类的生命周期而言,一个类包括如下阶段:

加载、验证、准备、初始化和卸载这5个阶段的顺序是确

定的,类的加载过程必须按照这种顺序进行

类的装载 是通过 类加载器 完成的,加载器将 .class 文

件的二进制文件装入JVM的方法区,并且在堆区创建描

述这个类的 java.lang.Class 对象。用来封装数据。

但是同一个类只会被类装载器装载一次。

链接 就是把二进制数据组装为可以运行的状态。链接分

为校验,准备,解析这3个阶段

类加载器

顾名思义,类加载器(class loader)用来加载 Java 类到

Java 虚拟机中。一般来说,Java 虚拟机使用 Java 类的

方式如下:Java 源程序(.java 文件)在经过 Java 编译

器编译之后就被转换成 Java 字节代码(.class 文件)。

类加载器负责读取 Java 字节代码,并转换成

java.lang.Class 类的一个实例。每个这样的实例用

来表示一个 Java 类。通过此实例的 newInstance()

方法就可以创建出该类的一个对象。实际的情况可能更

加复杂,比如 Java 字节代码可能是通过工具动态生成

的,也可能是通过网络下载的。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-f173Onk3-1603201534777)(C:\Users\lwf\AppData\Roaming\Typora\typora-user-images\image-20201020214106170.png)]

在java中有三种类类加载器:

⑴ Bootstrap ClassLoader 此加载器采用c++编写,一般开

发中很少见。⑵ Extension ClassLoader 用来进行扩展类的加载,一般

对应的是jre\lib\ext目录中的类

⑶ AppClassLoader 加载classpath指定的类,是最常用的

加载器。同时也是java中默认的加载器。 了解即可。

public static void main(String[] args) throws 

Exception { 

System.out.println("类加载器 

"+ClassLoader.class.getClassLoader().getClass( 

).getName()); 

}

反射的运用

package com.lwf;

import java.io.*;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/** * @author lwf * @title: RefTest * @projectName Learn * @description: 反射 * @date 2020/10/2019:51 */
public class RefTest {
    
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, CloneNotSupportedException, IOException, NoSuchFieldException {
    
        //new对象
        Student student0=new Student();
        student0.setName("lwf");
        student0.setAge(18);
        student0.setClassName("class 1");
        System.out.println(student0);
        Class  c=Class.forName("com.lwf.Student");
        Constructor[] constructors = c.getDeclaredConstructors();
        for(Constructor c1 : constructors){
    
            System.out.println(c1);
        }
        Field[] declaredFields = c.getDeclaredFields();
        System.out.println("属性");
        for(Field field:declaredFields){
    
            System.out.println(field.getName()+","+field.getType());
        }
        //修改访问权限,将私有属性直接修改值
        Field field = c.getDeclaredField("name");
        field.setAccessible(true);
        field.set(student0, "罗卫飞");
        System.out.println(student0);
        //调用方法
        Method method=c.getMethod("getName");
        System.out.println(method.invoke(student0));


        //反射获取对象
        Student student1= (Student)c.getConstructor(String.class,Integer.class,String.class).newInstance("lwf",18,"一班");
        Student student= (Student)c.getConstructor().newInstance();
        System.out.println(student);
        System.out.println(student1);



        //克隆对象 继承Coneable接口
        Student student2=(Student)student1.clone();
        System.out.println(student2);

        //序列化获取对象
        ObjectOutputStream outputStream = new ObjectOutputStream(new FileOutputStream("1.txt"));
        outputStream.writeObject(student);
        ObjectInputStream obj=new ObjectInputStream(new FileInputStream("1.txt"));
        Student object =(Student) obj.readObject();
        System.out.println(object);
    }
}
  1. 构造器 :

    //反射获取对象
               Student student1= (Student)c.getConstructor(String.class,Integer.class,String.class).newInstance("lwf",18,"一班");
               Student student= (Student)c.getConstructor().newInstance();
     
    
  2. 获取属性并赋值

     //修改访问权限,将私有属性直接修改值
            Field field = c.getDeclaredField("name");//从所有属性中找;getField从public修饰属性中找属性
            field.setAccessible(true);
            field.set(student0, "罗卫飞");
            System.out.println(student0);
    
  3. 获取方法并执行 Method对象调用invoke(实例,方法参数列表)

    //调用方法
            Method method=c.getMethod("getName");
            System.out.println(method.invoke(student0));
    

联系:对象的创建

//new对象
        Student student0=new Student();


//反射获取对象
        Student student1= (Student)c.getConstructor(String.class,Integer.class,String.class).newInstance("lwf",18,"一班");
        Student student= (Student)c.getConstructor().newInstance();
        System.out.println(student);
        System.out.println(student1);
        
   //克隆对象 继承Coneable接口
    Student student2=(Student)student1.clone();
    System.out.println(student2);

    //序列化获取对象
    ObjectOutputStream outputStream = new ObjectOutputStream(new FileOutputStream("1.txt"));
    outputStream.writeObject(student);
    ObjectInputStream obj=new ObjectInputStream(new FileInputStream("1.txt"));
    Student object =(Student) obj.readObject();
全部评论

相关推荐

评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务