java反射机制详解与应用,什么是JAVA反射机制

本文目录一览:

java中反射原理,和应用

1、反射的原理,即是jvm通过字节码class文件,生成相应的对象。

就像正常生成一个对象一样,都是来源于字节码class文件,

之所以叫反射,只是因为他不像正常的对象声明,如A a=new A()的方式。

2、反射在框架中的应用太广,只举一个典型的例子,即Spring中Bean的注入。

bean总是先声明class路径,然后依次生成就可以了。

再试下吧。

java中的反射机制是什么?有什么作用呢?求解,谢谢。

Java反射机制详解

1. 反射机制是什么

反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

2. 反射机制能做什么

反射机制主要提供了以下功能:

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

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

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

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

生成动态代理。

3. 反射机制的相关API

通过一个对象获得完整的包名和类名

package net.xsoftlab.baike;

public class TestReflect {

    public static void main(String[] args) throws Exception {

        TestReflect testReflect = new TestReflect();

        System.out.println(testReflect.getClass().getName());

        // 结果 net.xsoftlab.baike.TestReflect

    }

}

实例化Class类对象

package net.xsoftlab.baike;

public class TestReflect {

    public static void main(String[] args) throws Exception {

        Class? class1 = null;

        Class? class2 = null;

        Class? class3 = null;

        // 一般采用这种形式

        class1 = Class.forName(“net.xsoftlab.baike.TestReflect”);

        class2 = new TestReflect().getClass();

        class3 = TestReflect.class;

        System.out.println(“类名称   ” + class1.getName());

        System.out.println(“类名称   ” + class2.getName());

        System.out.println(“类名称   ” + class3.getName());

    }

}

获取一个对象的父类与实现的接口

package net.xsoftlab.baike;

import java.io.Serializable;

public class TestReflect implements Serializable {

    private static final long serialVersionUID = -2862585049955236662L;

    public static void main(String[] args) throws Exception {

        Class? clazz = Class.forName(“net.xsoftlab.baike.TestReflect”);

        // 取得父类

        Class? parentClass = clazz.getSuperclass();

        System.out.println(“clazz的父类为:” + parentClass.getName());

        // clazz的父类为: java.lang.Object

        // 获取所有的接口

        Class? intes[] = clazz.getInterfaces();

        System.out.println(“clazz实现的接口有:”);

        for (int i = 0; i  intes.length; i++) {

            System.out.println((i + 1) + “:” + intes[i].getName());

        }

        // clazz实现的接口有:

        // 1:java.io.Serializable

    }

}

获取某个类中的全部构造函数 – 详见下例

通过反射机制实例化一个类的对象

package net.xsoftlab.baike;

import java.lang.reflect.Constructor;

public class TestReflect {

    public static void main(String[] args) throws Exception {

        Class? class1 = null;

        class1 = Class.forName(“net.xsoftlab.baike.User”);

        // 第一种方法,实例化默认构造方法,调用set赋值

        User user = (User) class1.newInstance();

        user.setAge(20);

        user.setName(“Rollen”);

        System.out.println(user);

        // 结果 User [age=20, name=Rollen]

        // 第二种方法 取得全部的构造函数 使用构造函数赋值

        Constructor? cons[] = class1.getConstructors();

        // 查看每个构造方法需要的参数

        for (int i = 0; i  cons.length; i++) {

            Class? clazzs[] = cons[i].getParameterTypes();

            System.out.print(“cons[” + i + “] (“);

            for (int j = 0; j  clazzs.length; j++) {

                if (j == clazzs.length – 1)

                    System.out.print(clazzs[j].getName());

                else

                    System.out.print(clazzs[j].getName() + “,”);

            }

            System.out.println(“)”);

        }

        // 结果

        // cons[0] (java.lang.String)

        // cons[1] (int,java.lang.String)

        // cons[2] ()

        user = (User) cons[0].newInstance(“Rollen”);

        System.out.println(user);

        // 结果 User [age=0, name=Rollen]

        user = (User) cons[1].newInstance(20, “Rollen”);

        System.out.println(user);

        // 结果 User [age=20, name=Rollen]

    }

}

class User {

    private int age;

    private String name;

    public User() {

        super();

    }

    public User(String name) {

        super();

        this.name = name;

    }

    public User(int age, String name) {

        super();

        this.age = age;

        this.name = name;

    }

    public int getAge() {

        return age;

    }

    public void setAge(int age) {

        this.age = age;

    }

    public String getName() {

        return name;

    }

    public void setName(String name) {

        this.name = name;

    }

    @Override

    public String toString() {

        return “User [age=” + age + “, name=” + name + “]”;

    }

}

获取某个类的全部属性

package net.xsoftlab.baike;

import java.io.Serializable;

import java.lang.reflect.Field;

import java.lang.reflect.Modifier;

public class TestReflect implements Serializable {

    private static final long serialVersionUID = -2862585049955236662L;

    public static void main(String[] args) throws Exception {

        Class? clazz = Class.forName(“net.xsoftlab.baike.TestReflect”);

        System.out.println(“===============本类属性===============”);

        // 取得本类的全部属性

        Field[] field = clazz.getDeclaredFields();

        for (int i = 0; i  field.length; i++) {

            // 权限修饰符

            int mo = field[i].getModifiers();

            String priv = Modifier.toString(mo);

            // 属性类型

            Class? type = field[i].getType();

            System.out.println(priv + ” ” + type.getName() + ” ” + field[i].getName() + “;”);

        }

         

        System.out.println(“==========实现的接口或者父类的属性==========”);

        // 取得实现的接口或者父类的属性

        Field[] filed1 = clazz.getFields();

        for (int j = 0; j  filed1.length; j++) {

            // 权限修饰符

            int mo = filed1[j].getModifiers();

            String priv = Modifier.toString(mo);

            // 属性类型

            Class? type = filed1[j].getType();

            System.out.println(priv + ” ” + type.getName() + ” ” + filed1[j].getName() + “;”);

        }

    }

}

通过反射机制调用某个类的方法

package net.xsoftlab.baike;

import java.lang.reflect.Method;

public class TestReflect {

    public static void main(String[] args) throws Exception {

        Class? clazz = Class.forName(“net.xsoftlab.baike.TestReflect”);

        // 调用TestReflect类中的reflect1方法

        Method method = clazz.getMethod(“reflect1”);

        method.invoke(clazz.newInstance());

        // Java 反射机制 – 调用某个类的方法1.

        // 调用TestReflect的reflect2方法

        method = clazz.getMethod(“reflect2”, int.class, String.class);

        method.invoke(clazz.newInstance(), 20, “张三”);

        // Java 反射机制 – 调用某个类的方法2.

        // age – 20. name – 张三

    }

    public void reflect1() {

        System.out.println(“Java 反射机制 – 调用某个类的方法1.”);

    }

    public void reflect2(int age, String name) {

        System.out.println(“Java 反射机制 – 调用某个类的方法2.”);

        System.out.println(“age – ” + age + “. name – ” + name);

    }

}

通过反射机制操作某个类的属性

package net.xsoftlab.baike;

import java.lang.reflect.Field;

public class TestReflect {

    private String proprety = null;

    public static void main(String[] args) throws Exception {

        Class? clazz = Class.forName(“net.xsoftlab.baike.TestReflect”);

        Object obj = clazz.newInstance();

        // 可以直接对 private 的属性赋值

        Field field = clazz.getDeclaredField(“proprety”);

        field.setAccessible(true);

        field.set(obj, “Java反射机制”);

        System.out.println(field.get(obj));

    }

}

4. 反射机制的应用实例

在泛型为Integer的ArrayList中存放一个String类型的对象。

package net.xsoftlab.baike;

import java.lang.reflect.Method;

import java.util.ArrayList;

public class TestReflect {

    public static void main(String[] args) throws Exception {

        ArrayListInteger list = new ArrayListInteger();

        Method method = list.getClass().getMethod(“add”, Object.class);

        method.invoke(list, “Java反射机制实例。”);

        System.out.println(list.get(0));

    }

}

通过反射取得并修改数组的信息

package net.xsoftlab.baike;

import java.lang.reflect.Array;

public class TestReflect {

    public static void main(String[] args) throws Exception {

        int[] temp = { 1, 2, 3, 4, 5 };

        Class? demo = temp.getClass().getComponentType();

        System.out.println(“数组类型: ” + demo.getName());

        System.out.println(“数组长度  ” + Array.getLength(temp));

        System.out.println(“数组的第一个元素: ” + Array.get(temp, 0));

        Array.set(temp, 0, 100);

        System.out.println(“修改之后数组第一个元素为: ” + Array.get(temp, 0));

    }

}

将反射机制应用于工厂模式

package net.xsoftlab.baike;

interface fruit {

    public abstract void eat();

}

class Apple implements fruit {

    public void eat() {

        System.out.println(“Apple”);

    }

}

class Orange implements fruit {

    public void eat() {

        System.out.println(“Orange”);

    }

}

class Factory {

    public static fruit getInstance(String ClassName) {

        fruit f = null;

        try {

            f = (fruit) Class.forName(ClassName).newInstance();

        } catch (Exception e) {

            e.printStackTrace();

        }

        return f;

    }

}

/**

 * 对于普通的工厂模式当我们在添加一个子类的时候,就需要对应的修改工厂类。 当我们添加很多的子类的时候,会很麻烦。

 * Java 工厂模式可以参考

 * 

 * 

 * 现在我们利用反射机制实现工厂模式,可以在不修改工厂类的情况下添加任意多个子类。

 * 

 * 但是有一点仍然很麻烦,就是需要知道完整的包名和类名,这里可以使用properties配置文件来完成。

 * 

 * java 读取 properties 配置文件 的方法可以参考

 * 

 * 

 * @author xsoftlab.net

 */

public class TestReflect {

    public static void main(String[] args) throws Exception {

        fruit f = Factory.getInstance(“net.xsoftlab.baike.Apple”);

        if (f != null) {

            f.eat();

        }

    }

}

我有一个微信公众号,经常会分享一些Java技术相关的干货,还有一些学习资源。

如果你喜欢我的分享,可以用微信搜索“Java团长”或者“javatuanzhang”关注。

java课程分享Java的反射机制

Java反射机制是一个非常强大的功能,在很多大型项目比如Spring,Mybatis都可以看见反射的身影。通过反射机制我们可以在运行期间获取对象的类型信息,利用这一特性我们可以实现工厂模式和代理模式等设计模式,同时也可以解决Java泛型擦除等令人苦恼的问题。下面java课程就从实际应用的角度出发,来应用一下Java的反射机制。

反射基础

p.s:本文需要读者对反射机制的API有一定程度的了解,如果之前没有接触过的话,建议先看一下官方文档的QuickStart。

在应用反射机制之前,首先我们先来看一下如何获取一个对象对应的反射类Class,在Java中我们有三种方法可以获取一个对象的反射类。

通过getClass方法

在Java中,每一个Object都有一个getClass方法,通过getClass方法我们可以获取到这个对象对应的反射类:

Strings=”ziwenxie”;

Class?c=s.getClass();

通过forName方法

我们也可以调用Class类的静态方法forName:

Class?c=Class.forName(“java.lang.String”);

使用.class

或者我们也可以直接使用.class:

Class?c=String.class;

获取类型信息

在文章开头我们就提到反射的一大好处就是可以允许我们在运行期间获取对象的类型信息,下面我们通过一个例子来具体看一下。

首先我们在typeinfo.interfacea包下面新建一个接口A:

packagetypeinfo.interfacea;

publicinterfaceA{voidf();}

接着我们在typeinfo.packageaccess包下面新建一个接口C,接口C继承自接口A,并且我们还另外创建了几个用于测试的方法,注意下面几个方法的权限都是不同的。

原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/247002.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小蓝小蓝
上一篇 2024-12-12 13:18
下一篇 2024-12-12 13:18

相关推荐

  • Java JsonPath 效率优化指南

    本篇文章将深入探讨Java JsonPath的效率问题,并提供一些优化方案。 一、JsonPath 简介 JsonPath是一个可用于从JSON数据中获取信息的库。它提供了一种DS…

    编程 2025-04-29
  • java client.getacsresponse 编译报错解决方法

    java client.getacsresponse 编译报错是Java编程过程中常见的错误,常见的原因是代码的语法错误、类库依赖问题和编译环境的配置问题。下面将从多个方面进行分析…

    编程 2025-04-29
  • Java Bean加载过程

    Java Bean加载过程涉及到类加载器、反射机制和Java虚拟机的执行过程。在本文中,将从这三个方面详细阐述Java Bean加载的过程。 一、类加载器 类加载器是Java虚拟机…

    编程 2025-04-29
  • Java腾讯云音视频对接

    本文旨在从多个方面详细阐述Java腾讯云音视频对接,提供完整的代码示例。 一、腾讯云音视频介绍 腾讯云音视频服务(Cloud Tencent Real-Time Communica…

    编程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介绍

    本文将详细介绍Java Milvus SearchParam withoutFields的相关知识和用法。 一、什么是Java Milvus SearchParam without…

    编程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java语言中的一个版本,于2014年3月18日发布。本文将从多个方面对Java 8中某一周的周一进行详细的阐述。 一、数组处理 Java 8新特性之一是Stream…

    编程 2025-04-29
  • Java判断字符串是否存在多个

    本文将从以下几个方面详细阐述如何使用Java判断一个字符串中是否存在多个指定字符: 一、字符串遍历 字符串是Java编程中非常重要的一种数据类型。要判断字符串中是否存在多个指定字符…

    编程 2025-04-29
  • VSCode为什么无法运行Java

    解答:VSCode无法运行Java是因为默认情况下,VSCode并没有集成Java运行环境,需要手动添加Java运行环境或安装相关插件才能实现Java代码的编写、调试和运行。 一、…

    编程 2025-04-29
  • Java任务下发回滚系统的设计与实现

    本文将介绍一个Java任务下发回滚系统的设计与实现。该系统可以用于执行复杂的任务,包括可回滚的任务,及时恢复任务失败前的状态。系统使用Java语言进行开发,可以支持多种类型的任务。…

    编程 2025-04-29
  • Java 8 Group By 会影响排序吗?

    是的,Java 8中的Group By会对排序产生影响。本文将从多个方面探讨Group By对排序的影响。 一、Group By的概述 Group By是SQL中的一种常见操作,它…

    编程 2025-04-29

发表回复

登录后才能评论