百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

浅析Java反射_java反射的用处

wptr33 2025-02-21 15:38 19 浏览

前言

上篇文章我们提到了可以使用反射机制破解单例模式。这篇文章我们就来谈一谈什么是反射,反射有什么用,怎么用,怎么实现反射。

概述

Java的反射(reflection)机制:是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法。这种动态获取程序信息以及动态调用对象的功能称为Java语言的反射机制。反射被视为动态语言的关键

功能

  • 在运行时判断任意一个对象所属的类;
  • 在运行时构造任意一个类的对象;
  • 在运行时判断任意一个类所具有的成员变量和方法;
  • 在运行时调用任意一个对象的方法;
  • 生成动态代理。

java虽然不是动态语言,但是它却有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。
换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。

特点

优点:

  • 能够运行时动态获取类的实例,大大提高系统的灵活性和扩展性。
  • 与Java动态编译相结合,可以实现无比强大的功能

反射机制的功能非常强大,但不能滥用。在能不使用反射完成时,尽量不要使用,原因有以下几点:
缺点:

  • 性能问题。
    Java反射机制中包含了一些动态类型,所以Java虚拟机不能够对这些动态代码进行优化。因此,反射操作的效率要比正常操作效率低很多。我们应该避免在对性能要求很高的程序或经常被执行的代码中使用反射。而且,如何使用反射决定了性能的高低。如果它作为程序中较少运行的部分,性能将不会成为一个问题。
  • 安全限制。
    使用反射通常需要程序的运行没有安全方面的限制。如果一个程序对安全性提出要求,则最好不要使用反射。
  • 程序健壮性。
    反射允许代码执行一些通常不被允许的操作,所以使用反射有可能会导致意想不到的后果。反射代码破坏了Java程序结构的抽象性,所以当程序运行的平台发生变化的时候,由于抽象的逻辑结构不能被识别,代码产生的效果与之前会产生差异。

获取Class对象的三种方式

三种方式

  • Class.forName("全类名") 多用于配置文件时
  • 类名.class 多用于参数传递时
  • 对象.getClass 多用于有对象实例时

==我们创建一个Person类用于实验,分别使用三种方法获取他们的对象,并打印他们的hashCode(),我们会发现他们的hashcode是相同的,证明他们是获取的是相同的,所有统一个字节码文件(*.class)在一次运行中,只会被加载一次。不论通过哪一种方式获取的对象都是同一个。

package hello;

public class ReflectDemo {

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

        //Class.forName("全类名")
        Class aClass = Class.forName("hello.Person");
        System.out.println(aClass);
        System.out.println(aClass.hashCode());
        //类名.clss
        Class personClass = Person.class;
        System.out.println(personClass);
        System.out.println(personClass.hashCode());
        //对象.getClass()
        Person person = new Person();
        System.out.println(person.getClass());
        System.out.println(person.getClass().hashCode());

    }
}


class Person{

    private String name;

    public Person(){

    }

    public Person(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

Class对象功能

Class对象里面大概有120多个方法,我们不可能全部都记住,也没必要全部都记住,我们只需要记住常用的方法和功能就可以了。

获取功能

  • 获取成员变量
    • Field[] getFields()
    • Field getFields(String name)
    • FIeld[] getDeclaredFields()
    • FIeld getDeclaredField(String name)
  • 获取构造方法
    • Constructor[] getConstructors()
    • Constructor getConstructor(类..., parameterTypes)
    • Constructor getDeclaredConstructor(类..., parameterTypes)
    • Constructor[] getDeclaredConstructors()
  • 获取成员方法
    • Method[] getMethods()
    • Method getMethod(String name,类..., parameterType)
    • Method[] getDeclaredMethods()
    • Method getDeclaredMethod(String name,类...,parameterTypes)
  • 获取类名
    • String getName()

方法

作用

Field[] getFields()

获取public的成员变量名

Field getFields(String name)

获取public的成员变量名,需要指定名称

FIeld[] getDeclaredFields()

获取所有的成员变量,包括private

FIeld getDeclaredField(String name)

获取指定的成员变量,私有的也可以获取,如果要改这个变量的值需要setAccessible(ture)暴力反射

Constructor[] getConstructors()

用于返回一个构造函数对象数组,该数组反映此Class对象表示的类的所有公共构造函数。

Constructor getConstructor(类..., parameterTypes)

获取指定参数的构造方法


ConstructorgetDeclaredConstructor(类..., parameterTypes)

方法返回一个 Constructor 对象,该对象反映了此 Class 对象表示的类或接口的指定构造函数

Constructor[] getDeclaredConstructors()

用于返回一个Constructor对象数组,该数组指示此Class对象所表示的类定义的构造函数的类型(Constructor可以是public,private,protected或default)

Method[] getMethods()

公共的所有方法

Method getMethod(String name,类..., parameterType)

指定公共的其中的方法,parameterType时一个数组

Method[] getDeclaredMethods()

所有的包括私有的

Method getDeclaredMethod(String name,类...,parameterTypes)

指定的,parameterTypes表示数组

String getName()

获取类名程


package hello;

import com.sun.org.apache.xerces.internal.impl.XMLEntityScanner;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class ReflectDemo {

    public static void main(String[] args) throws Exception{
        //获取class对象
        Class personClass = Person.class;

        //Field[] getFields()
        Field[] fields = personClass.getFields();
        for (Field field : fields) {
            System.out.println(field);
        }
        System.out.println("******************************");
       //FIeld[] getDeclaredFields()
        Field[] declaredFields = personClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println(declaredField);
        }
        System.out.println("******************************");
        //Constructor[] getConstructors()
        Constructor constructor = personClass.getConstructor();
        System.out.println(constructor);
        System.out.println("******************************");

        //Method[] getMethods()
        Method[] methods = personClass.getMethods();
        for (Method method : methods) {
            System.out.println(method);
        }

        System.out.println("******************************");
        //String getName()
        System.out.println(personClass.getName());



    }
}


class Person{

    public int id;
    private String name;

    public Person(){

    }

    public Person(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

案例

创建一个功能,不改里面的代码,只需要修改配置文件就能调用不同的方法

第一步:创建配置文件,”data.properties"
这个路径一定要写成hello.Preson,写成斜杠会识别不了。

# 这个写全路径类名
className=hello.Preson
# 这个写类里面想要调用的方法
methodName=eat

第二步:具体类"Preson.java"

package hello;

public class Preson {

    public String name;

    public void setName(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }
    public void eat(){
        System.out.println("eat....");
    }

}

第三步:编写实现代码,名成ReflectDemo.java

package hello;

import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.Properties;

public class ReflectDemo {

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

        //加载配置文件
        //创建properties对象
        Properties properties = new Properties();
        //加载配置文件,转为一个集合
        ClassLoader classLoader = ReflectDemo.class.getClassLoader();
        InputStream resourceAsStream = classLoader.getResourceAsStream("hello/data.properties");
        properties.load(resourceAsStream);

        //获取配置文件定义的数据
        String className = properties.getProperty("className");
        String methodName = properties.getProperty("methodName");

        //加载该类进入内存
        Class aClass = Class.forName(className);
        Object o = aClass.newInstance();
        Method method = aClass.getMethod(methodName);
        method.invoke(o);
    }
}

第四步:运行,我们只需要修改properties文件里面的类名和方法就能实现不同类不同方法的创建了。

  • 前言
  • 概述
  • 功能
  • 特点
  • 获取Class对象的三种方式
  • Class对象功能
  • 获取功能
  • 案例



__EOF__

本文作者: hjk-airl

本文链接:
https://www.cnblogs.com/hjk-airl/p/16055642.html

相关推荐

redis的八种使用场景

前言:redis是我们工作开发中,经常要打交道的,下面对redis的使用场景做总结介绍也是对redis举报的功能做梳理。缓存Redis最常见的用途是作为缓存,用于加速应用程序的响应速度。...

基于Redis的3种分布式ID生成策略

在分布式系统设计中,全局唯一ID是一个基础而关键的组件。随着业务规模扩大和系统架构向微服务演进,传统的单机自增ID已无法满足需求。高并发、高可用的分布式ID生成方案成为构建可靠分布式系统的必要条件。R...

基于OpenWrt系统路由器的模式切换与网页设计

摘要:目前商用WiFi路由器已应用到多个领域,商家通过给用户提供一个稳定免费WiFi热点达到吸引客户、提升服务的目标。传统路由器自带的Luci界面提供了工厂模式的Web界面,用户可通过该界面配置路...

这篇文章教你看明白 nginx-ingress 控制器

主机nginx一般nginx做主机反向代理(网关)有以下配置...

如何用redis实现注册中心

一句话总结使用Redis实现注册中心:服务注册...

爱可可老师24小时热门分享(2020.5.10)

No1.看自己以前写的代码是种什么体验?No2.DooM-chip!国外网友SylvainLefebvre自制的无CPU、无操作码、无指令计数器...No3.我认为CS学位可以更好,如...

Apportable:拯救程序员,IOS一秒变安卓

摘要:还在为了跨平台使用cocos2d-x吗,拯救objc程序员的奇葩来了,ApportableSDK:FreeAndroidsupportforcocos2d-iPhone。App...

JAVA实现超买超卖方案汇总,那个最适合你,一篇文章彻底讲透

以下是几种Java实现超买超卖问题的核心解决方案及代码示例,针对高并发场景下的库存扣减问题:方案一:Redis原子操作+Lua脚本(推荐)//使用Redis+Lua保证原子性publicbo...

3月26日更新 快速施法自动施法可独立设置

2016年3月26日DOTA2有一个79.6MB的更新主要是针对自动施法和快速施法的调整本来内容不多不少朋友都有自动施法和快速施法的困扰英文更新日志一些视觉BUG修复就不翻译了主要翻译自动施...

Redis 是如何提供服务的

在刚刚接触Redis的时候,最想要知道的是一个’setnameJhon’命令到达Redis服务器的时候,它是如何返回’OK’的?里面命令处理的流程如何,具体细节怎么样?你一定有问过自己...

lua _G、_VERSION使用

到这里我们已经把lua基础库中的函数介绍完了,除了函数外基础库中还有两个常量,一个是_G,另一个是_VERSION。_G是基础库本身,指向自己,这个变量很有意思,可以无限引用自己,最后得到的还是自己,...

China's top diplomat to chair third China-Pacific Island countries foreign ministers' meeting

BEIJING,May21(Xinhua)--ChineseForeignMinisterWangYi,alsoamemberofthePoliticalBureau...

移动工作交流工具Lua推出Insights数据分析产品

Lua是一个适用于各种职业人士的移动交流平台,它在今天推出了一项叫做Insights的全新功能。Insights是一个数据平台,客户可以在上面实时看到员工之间的交流情况,并分析这些情况对公司发展的影响...

Redis 7新武器:用Redis Stack实现向量搜索的极限压测

当传统关系型数据库还在为向量相似度搜索的性能挣扎时,Redis7的RedisStack...

Nginx/OpenResty详解,Nginx Lua编程,重定向与内部子请求

重定向与内部子请求Nginx的rewrite指令不仅可以在Nginx内部的server、location之间进行跳转,还可以进行外部链接的重定向。通过ngx_lua模块的Lua函数除了能实现Nginx...