java 反射技术

本文深入探讨了Java反射机制的原理与应用,包括通过反射获取完整包名、实例化Class对象、调用构造函数与方法、操作属性及数组信息等核心功能。详细解释了Class对象的使用场景与注意事项,并通过多个案例展示了反射机制在动态代理、属性操作等高级应用中的作用。

【案例1】通过一个对象获得完整的包名和类名

package Reflect;
 
/**
 * 通过一个对象获得完整的包名和类名
 * */
class Demo{
    //other codes...
}
 
class hello{
    public static void main(String[] args) {
        Demo demo=new Demo();
        System.out.println(demo.getClass().getName());
    }
}

【运行结果】:Reflect.Demo


添加一句:所有类的对象其实都是Class的实例。


【案例2】实例化Class类对象

package Reflect;
class Demo{
    //other codes...
}
 
class hello{
    public static void main(String[] args) {
        Class<?> demo1=null;
        Class<?> demo2=null;
        Class<?> demo3=null;
        try{
            //一般尽量采用这种形式
            demo1=Class.forName("Reflect.Demo");
        }catch(Exception e){
            e.printStackTrace();
        }
        demo2=new Demo().getClass();
        demo3=Demo.class;
         
        System.out.println("类名称   "+demo1.getName());
        System.out.println("类名称   "+demo2.getName());
        System.out.println("类名称   "+demo3.getName());
         
    }
}


  //一般尽量采用这种形式,可以在编译期进行检查
  demo1=Class.forName( "Reflect.Demo" );

关于 Class<?>

Class is a parametrizable class, hence you can use the syntax Class<T> where T is a type. By writing Class<?>, you're declaring a Class object which can be of any type (? is a wildcard). TheClass type is a type that contains metainformation about a class.

It's always good practice to refer to a generic type by specifying his specific type, by using Class<?> you're respecting this practice (you're aware of Class to be parametrizable) but you're not restricting your parameter to have a specific type.

Reference about Generics and Wildcards:http://docs.oracle.com/javase/tutorial/java/generics/wildcards.html

Reference about Class object and reflection (the feature of Java language used to introspect itself):http://java.sun.com/developer/technicalArticles/ALT/Reflection/


http://stackoverflow.com/questions/9921676/what-does-class-mean-in-java

【案例3】 通过 Class 实例化其他类的对象


class Person{
     
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString(){
        return "["+this.name+"  "+this.age+"]";
    }
    private String name;
    private int age;
}
 
class hello{
    public static void main(String[] args) {
        Class<?> demo=null;
        try{
            demo=Class.forName("Reflect.Person");
        }catch (Exception e) {
            e.printStackTrace();
        }
        Person per=null;
        try {
            per=(Person)demo.newInstance();
        } catch (InstantiationException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        per.setName("Rollen");
        per.setAge(20);
        System.out.println(per);
    }
}

但是注意一下,当我们把Person中的默认的无参构造函数取消的时候,比如自己定义只定义一个有参数的构造函数之后,会出现错误:

比如我定义了一个构造函数:


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


然后继续运行上面的程序,会出现:

java.lang.InstantiationException: Reflect.Person

    at java.lang.Class.newInstance0(Class.java:340)

    at java.lang.Class.newInstance(Class.java:308)

    at Reflect.hello.main(hello.java:39)

Exception in thread "main" java.lang.NullPointerException

    at Reflect.hello.main(hello.java:47)

所以大家以后再编写使用Class实例化其他类的对象的时候,一定要自己定义无参的构造函数。


【案例4】通过Class调用其他类中的构造函数 (也可以通过这种方式通过Class创建其他类的对象)


import java.lang.reflect.Constructor;
 
class Person{
     
    public Person() {
         
    }
    public Person(String name){
        this.name=name;
    }
    public Person(int age){
        this.age=age;
    }
    public Person(String name, int age) {
        this.age=age;
        this.name=name;
    }
    public String getName() {
        return name;
    }
    public int getAge() {
        return age;
    }
    @Override
    public String toString(){
        return "["+this.name+"  "+this.age+"]";
    }
    private String name;
    private int age;
}
 
class hello{
    public static void main(String[] args) {
        Class<?> demo=null;
        try{
            demo=Class.forName("Reflect.Person");
        }catch (Exception e) {
            e.printStackTrace();
        }
        Person per1=null;
        Person per2=null;
        Person per3=null;
        Person per4=null;
        //取得全部的构造函数
        Constructor<?> cons[]=demo.getConstructors();
        try{
            per1=(Person)cons[0].newInstance();
            per2=(Person)cons[1].newInstance("Rollen");
            per3=(Person)cons[2].newInstance(20);
            per4=(Person)cons[3].newInstance("Rollen",20);
        }catch(Exception e){
            e.printStackTrace();
        }
        System.out.println(per1);
        System.out.println(per2);
        System.out.println(per3);
        System.out.println(per4);
    }
}

PS: 这里面的程序最好自己打印一下构造函数的顺序,因为可能构造函数和上面的不一样,加上一下的代码打印一下,看看构造数组里面构造函数的顺序是什么?

for(Constructor<?> constructor : cons)
{
  System.out.println(constructor.toString());
}

【案例5】 返回一个类实现的接口


interface China{
    public static final String name="Rollen";
    public static  int age=20;
    public void sayChina();
    public void sayHello(String name, int age);
}
 
class Person implements China{
    public Person() {
         
    }
    public Person(String sex){
        this.sex=sex;
    }
    public String getSex() {
        return sex;
    }
    public void setSex(String sex) {
        this.sex = sex;
    }
    @Override
    public void sayChina(){
        System.out.println("hello ,china");
    }
    @Override
    public void sayHello(String name, int age){
        System.out.println(name+"  "+age);
    }
    private String sex;
}
 
class hello{
    public static void main(String[] args) {
        Class<?> demo=null;
        try{
            demo=Class.forName("Reflect.Person");
        }catch (Exception e) {
            e.printStackTrace();
        }
        //保存所有的接口
        Class<?> intes[]=demo.getInterfaces();
        for (int i = 0; i < intes.length; i++) {
            System.out.println("实现的接口   "+intes[i].getName());
        }
    }
}

主要的获取类的接口的方法代码:

demo = Class.forName("test8.Person");

Class<?> intes[] = demo.getInterfaces();
for (int i = 0; i < intes.length; i++)
{
	System.out.println("实现的接口   " + intes[i].getName());
}


【案例6】:取得其他类中的父

 Class<?> temp=demo.getSuperclass();
 System.out.println("继承的父类为:   "+temp.getName());


【案例7】:获得其他类中的全部构造函数

 Constructor<?>cons[]=demo.getConstructors();
  for (int i = 0; i < cons.length; i++) {
   System.out.println("构造方法:  "+cons[i]);
}

【案例8】 : 获得构造函数的各个部分,例如构造函数名,构造函数的参数,构造函数的修饰符等等。

class hello{
    public static void main(String[] args) {
        Class<?> demo=null;
        try{
            demo=Class.forName("Reflect.Person");
        }catch (Exception e) {
            e.printStackTrace();
        }
        Constructor<?>cons[]=demo.getConstructors();
        for (int i = 0; i < cons.length; i++) {
            Class<?> p[]=cons[i].getParameterTypes();
            System.out.print("构造方法:  ");
            int mo=cons[i].getModifiers();
            System.out.print(Modifier.toString(mo)+" ");
            System.out.print(cons[i].getName());
            System.out.print("(");
            for(int j=0;j<p.length;++j){
                System.out.print(p[j].getName()+" arg"+i);
                if(j<p.length-1){
                    System.out.print(",");
                }
            }
            System.out.println("){}");
        }
    }
}

获取构造函数的修饰符:

int  mo=cons[i].getModifiers();
System.out.print(Modifier.toString(mo)+" ");

获取构造函数的名字:
System.out.print(cons[i].getName())

获取构造函数的参数:
 Class<?> p[]=cons[i].getParameterTypes();
打印参数:
  for ( int  j= 0 ;j<p.length;++j)
{
   System.out.print(p[j].getName()+ " arg" +i);
   if (j<p.length- 1 )
  {
      System.out.print( "," );
    }
}
  System.out.println( "){}" );



【案例9】 获取方法的异常信息

 Class<?> exce[]=method[i].getExceptionTypes();
            if(exce.length>0){
                System.out.print(") throws ");
                for(int k=0;k<exce.length;++k){
                    System.out.print(exce[k].getName()+" ");
                    if(k<exce.length-1){
                        System.out.print(",");
                    }
                }
            }else{
                System.out.print(")");
            }

打印构造函数的代码:

Class<?> exce[] = method[i].getExceptionTypes();
			if (exce.length > 0)
			{
				System.out.print(") throws ");
				for (int k = 0; k < exce.length; ++k)
				{
					System.out.print(exce[k].getName() + " ");
					if (k < exce.length - 1)
					{
						System.out.print(",");
					}
				}
			} else
			{
				System.out.print(")");
			}
			System.out.println();

【案例10】 取得类的属性

class hello {
    public static void main(String[] args) {
        Class<?> demo = null;
        try {
            demo = Class.forName("Reflect.Person");
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("===============本类属性========================");
        // 取得本类的全部属性
        Field[] field = demo.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 = demo.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() + ";");
        }
    }
}


里面主要有两个方法

1. 取得本类的全部属性,包括public,private等。

 Field[] field = demo.getDeclaredFields();


2. 取得本类的或者是父类的public 属性

 Field[] filed1 = demo.getFields();


通过循环可取得每一个属性的 权限修饰符,属性名,属性的类型

属性的权限修饰符:

  int  mo = filed1[j].getModifiers();
  String priv = Modifier.toString(mo);

属性的类型:

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

属性的名字:

  filed1[j].getName()


【案例11】 通过反射调用类中的方法 

class hello {
    public static void main(String[] args) {
        Class<?> demo = null;
        try {
            demo = Class.forName("Reflect.Person");
        } catch (Exception e) {
            e.printStackTrace();
        }
        try{
            //调用Person类中的sayChina方法
            Method method=demo.getMethod("sayChina");
            method.invoke(demo.newInstance());
            //调用Person的sayHello方法
            method=demo.getMethod("sayHello", String.class,int.class);
            method.invoke(demo.newInstance(),"Rollen",20);
             
        }catch (Exception e) {
            e.printStackTrace();
        }
    }
}

调用特定的方法:

//1. 通过方法名以及参数得到指定的方法

 Method method=demo.getMethod("sayChina");

 method=demo.getMethod("sayHello", String.class,int.class);


//2. 通过invoke调用特定的方法

method.invoke(demo.newInstance());

method.invoke(demo.newInstance(),"Rollen",20);


【案例12】调用其他类的setget方法

这里面是利用反射得到通用的getter和setter方法


class hello {
    public static void main(String[] args) {
        Class<?> demo = null;
        Object obj=null;
        try {
            demo = Class.forName("Reflect.Person");
        } catch (Exception e) {
            e.printStackTrace();
        }
        try{
         obj=demo.newInstance();
        }catch (Exception e) {
            e.printStackTrace();
        }
        setter(obj,"Sex","男",String.class);
        getter(obj,"Sex");
    }
 
    /**
     * @param obj
     *            操作的对象
     * @param att
     *            操作的属性
     * */
    public static void getter(Object obj, String att) {
        try {
            Method method = obj.getClass().getMethod("get" + att);
            System.out.println(method.invoke(obj));
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
 
    /**
     * @param obj
     *            操作的对象
     * @param att
     *            操作的属性
     * @param value
     *            设置的值
     * @param type
     *            参数的属性
     * */
    public static void setter(Object obj, String att, Object value,
            Class<?> type) {
        try {
            Method method = obj.getClass().getMethod("set" + att, type);
            method.invoke(obj, value);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}// end class

这里面主要的setter的关键点:

Method method = object.getClass().getMethod("set" + att, type);
method.invoke(object, value);


object 这里面用到了多态,利用 getclass得到多态类的一个class对象,class对象可以调用getmethod方法,里面放的参数 1. 是方法名,2.是 方法需要的参数

然后运用method.invoke去实际调用某个对象的方法,参数 1. object 是通过class对象newinstance出来的类的对象 2. 调用方法的参数。


这里面主要的gettter的关键点:

Method method = object.getClass().getMethod("get"+att);
System.out.println(method.invoke(object));


分析和上面的相同


【案例13】 通过反射操作属性 

关键点:

1. 得到某个属性

Field field = demo.getDeclaredField("sex");

2.对于private的属性,要设置成可accesss

field.setAccessible(true);

3.调用属性的get或者set方法

field.set(obj, "男");
System.out.println(field.get(obj));


【案例14】通过放射取得并修改数组的信息

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));


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

注意:里面有一个Array是放射里面用来操作数组的 

 Array.getLength(temp)); 得到数组的长度

Array.get(temp, 0)  得到数组的某一个元素

Array.set(temp, 0, 100); 设置某一个数组的某个元素为一个新值




下面的开始反射的应用,关于动态代理的知识


【案例15】如何获得类加载器

class test{
     
}
class hello{
    public static void main(String[] args) {
        test t=new test();
        System.out.println("类加载器  "+t.getClass().getClassLoader().getClass().getName());
    }
}

其实在java中有三种类类加载器。

1)Bootstrap ClassLoader 此加载器采用c++编写,一般开发中很少见。

2)Extension ClassLoader 用来进行扩展类的加载,一般对应的是jre\lib\ext目录中的类

3)AppClassLoader 加载classpath指定的类,是最常用的加载器。同时也是java中默认的加载器。


【案例16】动态代理

import java.lang.reflect.*;
 
//定义项目接口
interface Subject {
    public String say(String name, int age);
}
 
// 定义真实项目
class RealSubject implements Subject {
    @Override
    public String say(String name, int age) {
        return name + "  " + age;
    }
}
 
class MyInvocationHandler implements InvocationHandler {
    private Object obj = null;
 
    public Object bind(Object obj) {
        this.obj = obj;
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
                .getClass().getInterfaces(), this);
    }
 
    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        Object temp = method.invoke(this.obj, args);
        return temp;
    }
}
 
class hello {
    public static void main(String[] args) {
        MyInvocationHandler demo = new MyInvocationHandler();
        Subject sub = (Subject) demo.bind(new RealSubject());
        String info = sub.say("Rollen", 20);
        System.out.println(info);
    }
}


反射机制的应用有java的动态代理,关于java 的动态代理可以看另一篇文章。
























【SCI级别】多策略改进鲸鱼优化算法(HHWOA)和鲸鱼优化算法(WOA)在CEC2017测试集函数F1-F30寻优对比内容概要:本文档主要介绍了一项关于多策略改进鲸鱼优化算法(HHWOA)与标准鲸鱼优化算法(WOA)在CEC2017测试集函数F1-F30上进行寻优性能对比的研究,属于智能优化算法领域的高水平科研工作。文中通过Matlab代码实现算法仿真,重点展示了HHWOA在收敛速度、寻优精度和稳定性方面的优势,体现了多策略改进的有效性。该研究适用于复杂优化问题求解,尤其在工程优化、参数辨识、机器学习超参数调优等领域具有应用潜力。; 适合人群:具备一定算法基础和Matlab编程能力的研究生、科研人员及从事智能优化算法开发与应用的工程技术人员,尤其适合致力于SCI论文写作与算法创新的研究者。; 使用场景及目标:①用于理解鲸鱼优化算法的基本原理及多策略改进思路(如种群初始化、非线性收敛因子、精英反向学习等);②为智能优化算法的性能测试与对比实验提供CEC2017标准测试平台的实现参考;③支撑学术研究中的算法创新与论文复现工作。; 阅读建议:建议结合提供的Matlab代码进行实践操作,重点关注HHWOA的改进策略模块与WOA的差异,通过重复实验验证算法性能,并可将其思想迁移至其他优化算法的改进中,提升科研创新能力。
本资源库提供了赛灵思Zynq-7000系列全系可编程系统芯片的电子设计封装文件。文件格式为Altium Designer专用的集成库(IntLib),内含三维模型视图,并可解构为独立的原理图符号库(SchLib)与印制电路板封装库(PcbLib)。该库旨在为硬件工程师提供精确、完整的物理封装与逻辑符号参考,以加速基于此系列芯片的电路板设计流程。 库中包含的元器件总数为45个,涵盖了该系列多种关键型号。每个型号均详细定义了其核心规格,包括工作频率、处理系统与可编程逻辑单元的输入/输出引脚数量、球栅阵列封装类型、速度等级以及适用的温度范围(商业级、工业级或扩展级)。所有封装均符合无铅环保标准。 具体封装型号示例如下: - XC7Z010-1CLG400C:运行频率667兆赫兹,配备130个处理系统I/O与100个可编程逻辑I/O,采用400球BGA封装,属速度等级1的商业级产品。 - XC7Z010-1CLG400I:规格同上,但适用于工业级温度环境。 - XC7Z010-2CLG400E:频率提升至733兆赫兹,速度等级2,适用于扩展级温度范围。 - XC7Z010-2CLG400I:733兆赫兹频率,速度等级2的工业级型号。 - XC7Z010-3CLG400E:最高频率达800兆赫兹,速度等级3,适用于扩展级环境。 - XC7Z020-1CLG400C:667兆赫兹频率,提供130个PS I/O与125个FPGA I/O的400球BGA封装,商业级。 - XC7Z020-1CLG400I:上述规格的工业级版本。 - XC7Z020-1CLG484C:667兆赫兹频率,I/O配置为130个PS与200个FPGA,采用484球BGA封装,商业级。 - XC7Z020-1CLG484I:同上规格的工业级型号。 - XC7Z020-2CLG400E:频率733兆赫兹,速度等级2,I/O配置为130/125,400球BGA,扩展级。 - XC7Z020-2CLG400I:733兆赫兹的工业级版本。 - XC7Z020-2CLG484E:频率733兆赫兹,配备130个PS I/O与200个FPGA I/O的484球BGA封装,扩展级。 - XC7Z020-2CLG484I:同上规格的工业级型号。 - XC7Z020-3CLG400E:频率800兆赫兹,速度等级3,I/O配置为130/125,400球BGA封装,扩展级。 此封装库经过系统化整理,确保了型号参数描述的严谨性与一致性,为电路设计中的元器件选型与布局布线提供了可靠的底层数据支持。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
本文系统性地阐述了人脸识别技术,重点聚焦于卷积神经网络(CNN)的构建原理及其在MATLAB平台上的具体实施流程。人脸识别作为计算机视觉的关键分支,其核心在于通过提取与比对面部图像中的特征信息,实现对个体身份的精准辨识或验证。CNN凭借其在图像分析与模式识别任务中的卓越性能,已成为该领域的主流技术架构。 一、卷积神经网络(CNN)的基本构成 CNN是一种专为处理网格状数据(如图像)而设计的深度学习模型。其结构通常包含多个功能层:卷积层负责利用卷积核扫描输入图像,从而捕获局部特征;激活函数(如ReLU)为模型引入非线性变换,增强其表征能力;池化层则通过下采样操作压缩数据维度,同时保留重要特征信息;全连接层最终将学习到的高级特征映射至分类输出。 二、面向人脸识别的CNN模型设计 典型的人脸识别CNN模型(例如VGGFace、FaceNet等)通常遵循一套标准流程:首先对输入图像进行归一化、裁剪等预处理,随后通过一系列卷积与池化层逐步提取层次化特征,最后经由全连接层与分类器完成身份判定。模型训练多采用监督学习策略,依托大规模标注人脸数据集,使网络学会生成具有判别力的特征向量,以此作为不同个体的区分依据。 三、基于MATLAB环境的模型实现 MATLAB深度学习工具箱为CNN模型的开发与部署提供了集成化支持。用户可通过预置函数便捷地定义网络架构,包括各层的参数配置。数据预处理环节涉及图像读取、尺寸调整及数值归一化等操作,可利用相关图像处理函数完成。模型训练过程通过指定优化算法、损失函数及验证数据集进行,训练完成后即可调用分类函数对新人脸样本进行预测。 四、技术应用场景 人脸识别技术已广泛应用于多个现实领域:在安防系统中,它为门禁控制与设备解锁提供了高效的身份认证方案;于社交媒体平台,可自动识别并标注用户上传图片中的人物;在公共安全与执法监控场景下,则能协助从视频流中快速筛查目标人员。 综上所述,在MATLAB中实现人脸识别CNN模型涵盖从数据预处理、网络构建、模型训练到实际应用的完整链条。深入理解并掌握这一技术路径,将有力推动其在各行业场景中的落地与实践。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值