JAVA之new一个对象和利用反射创建一个对象的区别

本文探讨了Java中使用new关键字与反射创建对象的性能差异,实验结果显示new方式创建对象更快。然而,反射提供了诸如动态访问私有属性、无需预先知道类名即可创建对象等优势。Java反射主要功能包括运行时判断类、构造对象、访问成员变量和方法,以及改变类的属性值。文中通过示例展示了如何通过反射实现属性的动态赋值,并在测试中成功将Person类的值复制到Child类中。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一. 基础概念

Java中,一般我们创建一个对象可能会选择new一下个实例。但是随着我们技术的不断提升,我们也学习到了,可以通过反射技术实现对象的创建。

//new 方式创建对象
ReflectDemo reflectDemo = new ReflectDemo();
//反射创建对象  反射创建对象的三种方式
(1)Class<ReflectDemo> reflectDemoClass = ReflectDemo.class;
(2)Class<?> aClass = Class.forName ("com.whale.springtransaction.transactiondemo.reflectdemo.ReflectDemo");
(3)Class<? extends Class> aClass = reflectDemoClass.getClass ();
//测试代码如下
public class ReflectDemo {
 public static void main (String[] args) throws IllegalAccessException, InstantiationException {
  proxyObject();
  newObject();
 }

 //new 创建对象
 //5
 public static void newObject(){
  long startTime = System.currentTimeMillis ();
  int i;
  for (i = 0; i < 100000000; i++) {
   ReflectDemo reflectDemo = new ReflectDemo ();
  }
  if (i == 100000000) {
   long endTime = System.currentTimeMillis ();
   System.out.println ("new耗时为:" + (endTime - startTime));
  }
 }

 //反射 创建对象
 //30
 public static void proxyObject() throws IllegalAccessException, InstantiationException {
  long startTime = System.currentTimeMillis ();
  Class<ReflectDemo> reflectDemoClass = ReflectDemo.class;
  int i;
  for (i = 0; i < 100000000; i++) {
   ReflectDemo reflectDemo = reflectDemoClass.newInstance ();
  }
  if (i == 100000000) {
   long endTime = System.currentTimeMillis ();
   System.out.println ("反射耗时为:" + (endTime - startTime));
  }
 }
}

在这里插入图片描述
从结果我们可以看出创建对象new方法更快一些,那反射的优点在哪里?

二、反射的优点

  • new的对象无法访问其中的私有属性,反射出来的可以通过设置setAccessible()方法来省略访问权限符。

  • new必须要知道类名,而反射创建对象不需要知道类型也可以创建
    反射的核心是 JVM 在运行时才动态加载类或调用方法/访问属性,它不需要事先(写代码的时候或编译期)知道运行对象是谁。

Java 反射主要提供以下功能:

  • 在运行时判断任意一个对象所属的类;
  • 在运行时构造任意一个类的对象;
  • 在运行时判断任意一个类所具有的成员变量和方法(通过反射甚至可以调用private方法);
  • 在运行时调用任意一个对象的方法

可以在运行时改变类的属性值。

举例:

1、首先我们定义一个Person类;
在这里插入图片描述
2、再定义一个Child类
在这里插入图片描述
3、写出映射工具类

public static<T> void setValue(T model,T target){
    try {
        Class<?> clazz = model.getClass();
        //getDeclaredFields()返回Class中所有的字段,包括私有字段
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            System.out.println(field);
            String getMethodName = "get" + (field.getName().charAt(0) + "").toUpperCase() + field.getName().substring(1);
           //获取对象所申明的方法
            Method getMethod = clazz.getMethod(getMethodName);
            Object object = getMethod.invoke(model);
            if(object!=null){
                String setMethodName = "set" + (field.getName().charAt(0) + "").toUpperCase() + field.getName().substring(1);
                Method setMethod = target.getClass().getMethod(setMethodName,field.getType());
                //把model里面的值依次往target里面设置
                setMethod.invoke(target,object);
            }
        }
    }catch (ReflectiveOperationException e){
        e.printStackTrace();

    }
}

用法解释:其实就是构造get和set构造器的方法名,然后通过invoke方法实现真正的设置值。

4、测试:

public static void main(String[] args) {
    Person person = new Person(1,"xiaoming",18);
    Child child = new Child();
    ServiceTransToIntegration.setValue(person,child);
    System.out.println(child);
}

最终结果发现:值成功的复制到Child类属性上面去了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

bst@微胖子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值