👉这是一个或许对你有用的开源项目
国产 Star 破 10w+ 的开源项目,前端包括管理后台 + 微信小程序,后端支持单体和微服务架构。
功能涵盖 RBAC 权限、SaaS 多租户、数据权限、商城、支付、工作流、大屏报表、微信公众号等等功能:
Boot 地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro Cloud 地址:https://gitee.com/zhijiantianya/yudao-cloud 视频教程:https://doc.iocoder.cn
来源:blog.csdn.net/sunny_json
/article/details/113047372
一. 基础概念
Java中,一般我们创建一个对象可能会选择new一下个实例。但是随着我们技术的不断提升,我们也学习到了,可以通过反射技术实现对象的创建。
可是,你有没有想一下,什么时候我们改用new创建对象,什么时候我们改用反射创建对象呢?
两者创建对象的效率又是如何呢?
//new 方式创建对象
ReflectDemo reflectDemo = new ReflectDemo();
//反射创建对象 反射创建对象的三种方式
(1)Class reflectDemoClass = ReflectDemo.class;
(2)Class> aClass = Class.forName ("com.whale.springtransaction.transactiondemo.reflectdemo.ReflectDemo");
(3)Class extends Class> aClass = reflectDemoClass.getClass ();
基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
项目地址:https://github.com/YunaiV/ruoyi-vue-pro 视频教程:https://doc.iocoder.cn/video/
二. new 对象和反射创建对象的效率对比
//测试代码如下
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 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 100000000
个对象和反射创建 100000000 个对象,效率相差了很多倍。
所以下面我们来探讨一下为什么这么大差别?
首先第一点,一般我们的Java代码是需要编译后在虚拟机里面运行的。
我们一般都是通过一个前端编辑器,比如javac,把java文件转为class文件。
接下来,程序运行期间,可能会通过一个JIT,即时编译器将字节码文件转换为计算机认识的机器码文件。
另外一种可能是通过一个AOT编译器,直接把java文件编译为本地机器码文件。其中JIT在程序运行期会对程序进行优化,但是反射是通过动态解析的方式,因此可能无法执行某些java虚拟机的优化。
总结起来有下面几个原因:
Method#invoke
方法会对参数做封装和解封操作
-
需要检查方法可见 -
需要校验参数 -
反射方法难以内联 -
JIT 无法优化
基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
项目地址:https://github.com/YunaiV/yudao-cloud 视频教程:https://doc.iocoder.cn/video/
三. 反射和new 的使用场景
反射的部分使用场景
-
Spring通过反射来帮我们实例化对象,并放入到Ioc容器中 -
使用JDBC链接数据库时加载数据库驱动 Class.forName()
-
逆向代码 例如反编译 -
利用反射,在泛型为int的arryaList集合中存放一个String类型的对象
new 对象和反射的区别
-
new的对象无法访问其中的私有属性,反射出来的可以通过设置 setAccessible()
方法来省略访问权限符。 -
new必须要知道类名,而反射创建对象不需要知道类型也可以创建