【Java 反射学习】Java 反射基础

知识点

类是用来描述对象的,而反射就可以理解为是用来描述类的。

类中的属性包括:

  • Class 类本身
  • Package 类所在的包
  • Field 类中的属性
  • Method 类中的方法
  • Constructor 类中的构造方法
  • Annotation 类中的注解

如何获取Class

1.Class的静态方法,forName("全类名")

2.类.class关键字

3.对象引用.getClass()方法 Object中的方法

Class中的常用方法

/*
0--默认不写 1--public 2--private 4--protected 8--static 16--final 32--synchronized 64--volatile 128--transient 256--native 512--interface 1024--abstract
*/
int = getModifiers();       //获取类的修饰符(权限+特征)
String = getName();         //获取类的全类名
String = getSimpleName();    //获取简单名(只有类名 )
Class = getSuperClass();     //获取当前父类的对应Class
Class[] = getInterfaces();   //获取当前父类的接口
Package p = getPackage();    //获取当前类所在的包
        p.getName();        //获取包的名字
Class[] = getClasses();      //获取类中的内部类

Object = newInstance();     //获取当前类的对象(相当于调用了类中的无参数的构造方法)如果类中不存在无参数的构造方法,就会抛出NoSuchMethodException异常
Field = getField("属性名");          //获取类中的属性(公有的 自己类+父类)
Field[] = getFields();              //获取类中的全部属性(公有的 自己类+父类)
Field = getDeclaredField("属性名")    //获取当前类的属性(公有 + 私有 自己类)
Field = getDeclaredFields()    //获取当前类的全部属性(公有 + 私有 自己类)
    如果想修改私有的属性则需要设置属性可以被操作
        setAccessible()
public class TestMain {
    public static void main(String[] args) {

        try {
            Class<?> clazz = Class.forName("com.lili.reflect.People");
            Package aPackage = clazz.getPackage();
            int modifiers = clazz.getModifiers();
            System.out.println(modifiers);
            Method[] methods = clazz.getMethods();
            for (Method method : methods) {
                System.out.println(method.getName());
            }
            System.out.println(aPackage);

            Class<?>[] interfaces = clazz.getInterfaces();
            for (Class c : interfaces) {
                System.out.println(c.getName());
            }

            ArrayList<String> list = new ArrayList<>();
            Class c = ArrayList.class;
            Class superclass = c.getSuperclass();
            while (superclass != null) {
                System.out.println(superclass.getName());
                superclass = superclass.getSuperclass();
            }

        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }

    }
}

利用反射修改String类型的值

/*
注意只能是绕过private去修改属性的值,而不能去修改属性的长度,因为是final修饰的。
String的不可变指的是长度+值的不可变
*/
public class ChangeString {

    public static void main(String[] args) {

        try {
            String str = new String("abc");
            System.out.println(str);

            //1、利用反射技术获取String的Class
            Class clazz = str.getClass();
            //2、获取属性
            Field f = clazz.getDeclaredField("value");
            //3、设置可以修改属性的值
            f.setAccessible(true);
            //4、获取属性的值
            char[] newChar = (char[])f.get(str);
            //5、修改属性的值
            newChar[0] = 'xu';
            newChar[1] = 'Li';
            newChar[2] = 'Li';
            System.out.println(str);

        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }

}

利用反射调用类中的方法

可以获取共有的方法,包括自己类的以及父类的

可以找到私有的方法,但是要通过setAccessible(true)方法来执行私有的方法。

/**
 * 测试使用反射获取类中的方法
 */
public class TestMethod {

    public static void main(String[] args) {
        try {
            //1、获取People类对应的Class
            Class clazz = People.class;
            //2、获取对象
            People p = (People) clazz.newInstance();
            //3、通过clazz获取其中的方法,通过方法名以及方法的参数类型来定位方法。
            Method m = clazz.getMethod("eat", String.class);
            //4、调用方法,第一个参数是要执行方法的对象,第二个则是传进去的参数列表
            String n = (String) m.invoke(p, "lili要开始吃饭啦");
            System.out.println(n);

        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }

}

利用反射执行构造方法

/**
 * 利用反射执行构造方法
 */
public class TestConstructor {
    public static void main(String[] args) {
        try {
            //1、获取People对应的Class
            Class<People> clazz = People.class;
            //2、获取People中的构造方法,其中省去了构造方法的名称,因为是与类同名
            //无参的就是调用的无参数的构造方法
            //有参数的就是传的构造方法中形参的类型.class
            Constructor<People> constructor = clazz.getConstructor(String.class);
            //3、执行构造方法,同理参数就是要传参数的实参
            People people = constructor.newInstance("哈哈哈哈哈");
            System.out.println(people);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

设计一个小工具

这个小工具可以代替我们自己创建对象的功能,通过传递一个字符串,来帮我们创建一个对象,同时还能将对象内的所有属性赋值。

其实这就是简单的模拟了Spring中IOC思想的原理,IOC(Inversion Of Control)控制反转:将对象的控制权反转,交给Spring容器去处理;DI(Dependency Injection)依赖注入:Spring容器创建对象的同时帮我们自动的注入属性的值。

public class MySpring {
    //设计一个方法,将我们创建对象的过程交给该方法去执行。
    //参数String类型的全类名
    //返回值 创建出来的对象 Object类型--->再添加上DI依赖注入
    public Object getBean(String classPath) {
        Object obj = null;
        //模拟输入的实参
        Scanner scanner = new Scanner(System.in);
        System.out.println("请给"+ classPath +"的属性赋值");
        try {
            //1、获取该路径下对应的Class
            Class clazz = Class.forName(classPath);
            //2、创建一个对象
            obj = clazz.newInstance();
            //使用set方法对对象的属性进行赋值,找到每一个不同对象对应的set方法。
            //也就是字符串set+属性的名字
            //3、获取类中的属性
            Field[] fields = clazz.getDeclaredFields();
            for (Field field : fields) {
                //获取属性的名称
                String fieldName = field.getName();
                //改变属性名中第一个字母的大小写
                String first = fieldName.substring(0, 1).toUpperCase();
                //获取属性名中除开第一个字母的字段
                String last = fieldName.substring(1);
                //拼接set方法
                StringBuilder methodName = new StringBuilder("set");
                methodName.append(first);
                methodName.append(last);
                //4、获取属性的类型
                Class fieldType = field.getType();
                //5、获取方法
                Method method = clazz.getMethod(methodName.toString(), fieldType);
                //接收实参
                System.out.println("请给"+ fieldName +"属性赋值");
                String value = scanner.nextLine();
                /*为了解决参数类型不一致的问题,可以将参数的类型都设置未相应的包装类,
                并且将它们都转换成String的类型,除了Char类型之外需要另外的判断。
                可以利用其它包装类带String类型的构造方法进行处理。
                */
                Constructor con = fieldType.getConstructor(String.class);

                //6、执行方法
                method.invoke(obj, con.newInstance(value));

            }

        } catch (Exception e) {
            e.printStackTrace();
        }
        return obj;
    }
}

原文地址:https://www.cnblogs.com/liliHexiaogou/p/11470593.html

时间: 2024-10-04 00:04:36

【Java 反射学习】Java 反射基础的相关文章

java多线程学习--java.util.concurrent

CountDownLatch,api 文档:http://docs.oracle.com/javase/7/docs/api/java/util/concurrent/CountDownLatch.html A synchronization aid that allows one or more threads to wait until a set of operations being performed in other threads completes. 假设我们要打印1-100,最

Java培训学习需要什么基础?

随着互联网的飞速发展,其对相关人才的需求更是不断增加.高薪资高待遇吸引不少其它行业的人员也投身于此.作为最受企业欢迎Java语言,更是备受学员青睐.来千锋报名学习Java的不少学员都会问到同一个问题,学Java需要什么基础? 在回答这个问题前,我们先来简单的了解一下Java语言. Java是一种可以撰写跨平台应用程序的面向对象的程序设计语言.Java 技术具有卓越的通用性.高效性.平台移植性和安全性,广泛应用于PC.数据中心.游戏控制台.科学超级计算机.移动电话和互联网,同时拥有全球最大的开发者

吴裕雄--天生自然 JAVA开发学习:Applet 基础

import java.applet.*; import java.awt.*; public class HelloWorldApplet extends Applet { public void paint (Graphics g) { g.drawString ("Hello World", 25, 50); } } <html> <title>The Hello, World Applet</title> <hr> <app

如何学习Java?学习Java顺序?

Java相对于Asp.Net或Asp.Net MVC来讲,入门是比较困难和烦琐的!它不像.Net哪样有安装开发工具就可以跑程序了,不需要配置复杂的运行环境. 推荐的学习Java的学习顺序如下: 一.Java基础知识1.基本数据类型2.字符串的操作:3.数据类型之间的转换  4.日期和时间 5.数组和集合6.文件和目录(I/O)操作  7.Java多态的实现(继承.重载.覆盖)8.编码转换,怎样实现将GB2312编码的字符串转换为ISO-8859-1编码的字符串.9.Java中SQL和数据库 二.

[转载] java多线程学习-java.util.concurrent详解(四) BlockingQueue

转载自http://janeky.iteye.com/blog/770671 --------------------------------------------------------------------------------- 7.BlockingQueue     “支持两个附加操作的 Queue,这两个操作是:获取元素时等待队列变为非空,以及存储元素时等待空间变得可用.“ 这里我们主要讨论BlockingQueue的最典型实现:LinkedBlockingQueue 和Arra

[转载] java多线程学习-java.util.concurrent详解(一) Latch/Barrier

转载自http://janeky.iteye.com/blog/769965 Java1.5提供了一个非常高效实用的多线程包:java.util.concurrent, 提供了大量高级工具,可以帮助开发者编写高效.易维护.结构清晰的Java多线程程序.从这篇blog起,我将跟大家一起共同学习这些新的Java多线程构件 1. CountDownLatch     我们先来学习一下JDK1.5 API中关于这个类的详细介绍: “一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个

[转载] java多线程学习-java.util.concurrent详解(二)Semaphore/FutureTask/Exchanger

转载自http://janeky.iteye.com/blog/770393 ----------------------------------------------------------------------------- 3. Semaphore     我们先来学习一下JDK1.5 API中关于这个类的详细介绍: “一个计数信号量.从概念上讲,信号量维护了一个许可集.如有必要,在许可可用前会阻塞每一个 acquire(),然后再获取该许可.每个 release() 添加一个许可,从

[转载] java多线程学习-java.util.concurrent详解(三)ScheduledThreadPoolExecutor

转载自http://janeky.iteye.com/blog/770441 ---------------------------------------------------------------------------------- 6. ScheduledThreadPoolExecutor     我们先来学习一下JDK1.5 API中关于这个类的详细介绍: "可另行安排在给定的延迟后运行命令,或者定期执行命令.需要多个辅助线程时,或者要求 ThreadPoolExecutor 具

Java Demo 学习 理解 反射机制 (基础学习)

目录 反射机制是什么 反射机制能做什么 反射机制的相关API ·通过一个对象获得完整的包名和类名 ·实例化Class类对象 ·获取一个对象的父类与实现的接口 ·获取某个类中的全部构造函数 - 详见下例 ·通过反射机制实例化一个类的对象 ·获取某个类的全部属性 ·获取某个类的全部方法 ·通过反射机制调用某个类的方法 ·通过反射机制操作某个类的属性 ·反射机制的动态代理 反射机制的应用实例 ·在泛型为Integer的ArrayList中存放一个String类型的对象. ·通过反射取得并修改数组的信息

java反射学习之一反射机制概述

一.反射机制背景概述 1.反射(reflection)是java被视为动态语言的一个关键性质 2.反射机制指的是程序在运行时能获取任何类的内部所有信息 二.反射机制实现功能概述 1.只要给定类的全名,就能通过反射获取类的所有信息 2.程序在运行是能够获取任何一个对象所属的类对象 3.在运行时可以获取类的所有属性,并对其操作 4.在运行时可以获取类.父类的所有方法并调用 5.当前的主流框架struts2.srping.hiberate等框架的核心功能都是通过java反射机制来实现的 三.反射相关对