java 动态性之反射机制 详解 案例

}

long endTime = System.currentTimeMillis();

System.out.println(“反射动态方法调用 ,执行10亿次,耗时为:”+(endTime-startTime)+“ms”);

}

public static void test03() throws Exception{

User u = new User();

Class clazz = u.getClass();

Method m = clazz.getDeclaredMethod(“getUname”, null);

m.setAccessible(true); //不需要执行访问安全检查

long startTime = System.currentTimeMillis();

for (int i = 0; i < 1000000000L; i++) {

m.invoke(u, null);

}

long endTime = System.currentTimeMillis();

System.out.println(“反射方法调用,跳过安全检查 ,执行10亿次,耗时为:”+(endTime-startTime)+“ms”);

}

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

test01();

test02();

test03();

}

}

1.5 反射操作泛型(Generic)

1.5.1):java采用泛型擦除的机制引入泛型。java中的泛型仅仅是给编译器javac使用的,确保数据的安全性和免去强制类型装换的麻烦。但是,一旦编译完成,所有和泛型有关的类型全部擦除

1.5.2):为了通过反射操作这些类型已迎合实际开发的需要,java就新增了ParameterizedType,GenericArrayType,TypeVariable和WildcardTyoe几种类型代表不能被归一到Class类中的类型但是又和原始类型齐名的类型

ParameterizedType:表示一种参数化的类型,比如Collection

GenericArrayType:表示一种元素类型是参数化类型或者类型变量的数组类型

TypeVariable:是各种类型变量的公共父接口

WildcardTyoe:代表一种通配符类型表达式,比如:?,? extends Number ,? super Inter

package com.lyy.test;

import java.lang.reflect.Method;

import java.lang.reflect.ParameterizedType;

import java.lang.reflect.Type;

import java.util.List;

import java.util.Map;

import com.lyy.test.bean.User;

/**

  • 通过反射获取泛型信息

  • @author lyy

*/

public class Demo4 {

public void test01(Map<String,User> map,List list){

System.out.println(“Demo04.test01()”);

}

public Map<Integer,User> test02(){

System.out.println(“Demo04.test02()”);

return null;

}

public static void main(String[] args) {

try {

//获得指定方法参数泛型信息

Method m = Demo4.class.getMethod(“test01”, Map.class,List.class);

Type[] t = m.getGenericParameterTypes();//获取泛型信息

for (Type paramType : t) {

System.out.println(“#”+paramType);

if(paramType instanceof ParameterizedType){

Type[] genericTypes = ((ParameterizedType) paramType).getActualTypeArguments();

for (Type genericType : genericTypes) {

System.out.println(“泛型类型:”+genericType);

}

}

}

//获取指定方法返回值泛型信息

Method m2 = Demo4.class.getMethod(“test02”, null);

Type returnType = m2.getGenericReturnType();

if(returnType instanceof ParameterizedType){

Type[] genericTypes = ((ParameterizedType) returnType).getActualTypeArguments();

for (Type genericType : genericTypes) {

System.out.println(“返回值,泛型类型”+genericType);

}

}

} catch (Exception e) {

e.printStackTrace();

}

}

}

package com.lyy.test.bean;

public class User {

private int id;

private int age;

private String uname;

public int getId() {

return id;

}

public void setId(int id) {

this.id = id;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

public String getUname() {

return uname;

}

public void setUname(String uname) {

this.uname = uname;

}

public void setUname() {

this.uname = “lyy”;

}

public User(int id, int age, String uname) {

super();

this.id = id;

this.age = age;

this.uname = uname;

}

//javabean必须要有无参构造方法

public User() {

super();

}

}

1.6 反射操作注解(annotation)

1.6.1):可以通过反射API:getAnnotations、getAnnotation获取相关的注解信息

package com.lyy.test;

import java.lang.annotation.Annotation;

import java.lang.reflect.Field;

import com.lyy.test.annotation.SxtField;

import com.lyy.test.annotation.SxtTable;

/**

*通过反射获取注解信息

  • @authorlyy

*/

public class Demo5 {

public static void main(String[] args) {

try {

Class clazz = Class.forName(“com.lyy.test.annotation.SxtStudent”);

//获取所有类的所有有效注解

Annotation[] annotations=clazz.getAnnotations();

for (Annotation a : annotations) {

System.out.println(a);

}

//获取类的指定的注解

SxtTable st = (SxtTable) clazz.getAnnotation(SxtTable.class);

System.out.println(st.value());

//获得类的属性的注解

Field f = clazz.getDeclaredField(“studentName”);

SxtField sxtField = f.getAnnotation(SxtField.class);

System.out.println(sxtField.columnName()+“–”+sxtField.type()+“–”+sxtField.length());

//根据获取的表名、字段的信息、拼出DDL语句、然后,使用JDBC执行这个SQL,在数据库中生成相关的表

} catch (Exception e) {

e.printStackTrace();

}

}

}

package com.lyy.test.annotation;

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target(value={ElementType.TYPE})

@Retention(RetentionPolicy.RUNTIME)

public @interface SxtTable {

String value();

}

package com.lyy.test.annotation;

@SxtTable(“tb_student”)

public class SxtStudent {

@SxtField(columnName=“id”,type=“int”,length=10)

private int id;

@SxtField(columnName=“sname”,type=“varchar”,length=10)

private String studentName;

@SxtField(columnName=“age”,type=“int”,length=3)

private int age;

public int getId() {

return id;

}

public void setId(int id) {

this.id = id;

}

public String getStudentName() {

return studentName;

}

public void setStudentName(String studentName) {

this.studentName = studentName;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

}

package com.lyy.test.annotation;

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target(value={ElementType.FIELD})

@Retention(RetentionPolicy.RUNTIME)

public @interface SxtField {

String columnName();

String type();

int length();

}

反射的核心是class ,反射降低了运行效率,但是提高了开发效率

2、动态编译

2.1):JAVA 6.0 引入了动态编译机制

2.1.2):动态编译的应用场景:

2.1.2.1):可以做一个浏览器端编写Java代码,上传服务器编译和运行的在线评测系统

2.1.2.2):服务器动态加载某些类进行进行编译

2.1.3):动态编译的两种做法:

2.1.3.1):通过Rruntime调用javac,启动新的进程去操作

Runtime tun = Runtime.getRuntime();

Process process = run.exec(“java -cp g:/myjava/ HelloWorld.java”);

2.2):通过JavaCompiler动态编译

JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();

int result = compiler.run(null, null, null, “E:/myjava/HelloWorld.java”);

System.out.println(result==0?“编译成功!”:“编译失败!”);

第一个参数:为Java编译器提供参数

第二个参数:得到java编译器的输出信息

第三个参数:接收编译器的错误信息

第四个参数:可变参数(是一个String数组)能传入一个或多个Java源文件

返回值:0表示编译成功,非0表示编译失败

package com.lyy.test;

import java.lang.reflect.Method;

import java.net.URL;

import java.net.URLClassLoader;

import javax.tools.JavaCompiler;

import javax.tools.ToolProvider;

public class Demo1 {

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

//通过IO操作,将字符串存储成一个临时文件(Hi.java),然后调用动态编译方法!

String str = “public class Hi{public static void main(String[] args){”

  • “System.out.println(“haha lyy”);}}”;

JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();

int result = compiler.run(null, null, null, “E:/myjava/HelloWorld.java”);

System.out.println(result==0?“编译成功!”:“编译失败!”);

//通过Runtime调用执行类

// Runtime run = Runtime.getRuntime();

// Process pro = run.exec(“java -cp E:/myjava HelloWorld”);

//

// InputStream in = pro.getInputStream();

// BufferedReader reader = new BufferedReader(new InputStreamReader(in));

// String info = “”;

// while((info=reader.readLine()) != null){

// System.out.println(info);

// }

URL[] urls = new URL[]{new URL(“file:/”+“E:/myjava/”)};

URLClassLoader loader = new URLClassLoader(urls);

Class c = loader.loadClass(“HelloWorld”);

//调用加载类的main方法

Method method = c.getMethod(“main”,String[].class);

method.invoke(null,(Object)new String[]{“aa”,“bb”});

//由于可变参数是5.0之后才有 method.invoke(null,“aa”,“bb”) 会发生参数个数不匹配的问题

//因此,必须要加上(Object)转型,避免这个问题

//public static void main(String[] a,String[] b)

}

}

3、动态执行javassript代码

3.1):JAVA脚本引擎是从JDK6.0之后添加的新功能

3.2):脚本引擎介绍:

使得java应用程序可以通过一套固定的接口与各种脚本引擎交互,从而达到在Java平台上调用各种脚本语言的目的。

java脚本API是连通java平台和脚本语言的桥梁。

可以把一些复杂异变的业务逻辑交给脚本语言处理,这又大大提高了开发效率

3.3):获取脚本引擎对象

//获得脚本引擎对应

ScriptEngineManager sem = new ScriptEngineManager();

ScriptEngine engine = sem.getEngineByName(“javascript”);

3.4):java脚本API为开发者提供了如些功能:

获取本程序输入,通过脚本引擎运行脚本并返回运行结果,这是最核心的接口。

注意是:接口 。java可以使用各种不同的实现,从而通用的调用js、groovy、python等脚本

Rhino 是一种使用Java语言编写的javaScript的开源实现,原先由Mozilla开发,现在被集成进入JDK 6.0

通过脚本引擎的运行上下文在脚本和Java平台间交换数据

通过Java应用程序调用脚本函数

package com.lyy.test;

import java.io.FileReader;

import java.net.URL;

import java.util.List;

import javax.script.Invocable;

import javax.script.ScriptEngine;

import javax.script.ScriptEngineManager;

/**

  • 测试脚本引擎执行javascript代码

  • @author lyy

*/

public class Demo1 {

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

//相当于Java和Js脚本语言的一个中介

//获得脚本引擎对应

ScriptEngineManager sem = new ScriptEngineManager();

ScriptEngine engine = sem.getEngineByName(“javascript”);

//定义变量 存储到引擎的上下文中

engine.put(“msg”, “lyy is good man!”);

String str = “var user = {name:‘lyy’,age:18,schools:[‘清华大学’,‘湖北大学’]};”;

str +=“println(user.age)”;

//执行脚本

engine.eval(str);

engine.eval(“msg=‘lyy is goods !!!’”);

System.out.println(engine.get(“msg”));

System.out.println(“==============================================”);

//定义函数

engine.eval(“function add(a,b){var sum = a+b; return sum;}”);

//取得调用接口

Invocable jsInvoke = (Invocable)engine;

//执行脚本中定义的方法!

Object obj=jsInvoke.invokeFunction(“add”, new Object[]{13,20});

System.out.println(obj);

//导入其他的Java包,使用其他包中的java类,如果需要了解,可以详细学习了解Rhino的语法

String jsCode = “importPackage(java.util); var list = Arrays.asList([“清华大学”,“湖北工程学院”,“湖北职业技术学院”]);”;

engine.eval(jsCode);

List list2 = (List)engine.get(“list”);

for (String temp :list2) {

System.out.println(temp);

}

//执行一个js文件

URL url = Demo1.class.getClassLoader().getResource(“in.js”);

FileReader fr = new FileReader(url.getPath());

engine.eval(fr);

fr.close();

}

}

in.js

//定义test方法

function test(){

var a = 3;

var b = 4;

println(“invoke js file:”+(a+b));

}

//执行test方法

test();

**4、动态字节码操作

**

4.1):java动态性的两种常见实现方式:

4.1.1):字节码操作

4.2.1):反射

4.2):运行时操作字节码可以让我们实现如下功能:

4.2.1):动态生成新的类

4.2.2):动态改变某个类的结构(添加/删除/修改 新的属性/方法)

4.3):优势:

4.3.1):比反射开销小,性能高

4.3.2):JAVAssist性能高于反射,低于ASM

4.4):JAVAssist库的API详解

4.4.1):javassist的最外层的API和JAVA的反射包中的API颇为类型。

4.4.2):它主要由CtClass、CtMethod、以及CtField几个类组成。用以执行和JDK反射API中 java.lang.Class,java.lang.reflect.Method ,java.lang.reflect.Method.Field相同的操作

package com.lyy.test;

import javassist.CannotCompileException;

import javassist.ClassPool;

import javassist.CtClass;

import javassist.CtConstructor;

import javassist.CtField;

import javassist.CtMethod;

import javassist.NotFoundException;

/**

  • 测试使用Javasost生成一个新的类

  • @author lyy

*/

public class Demo1 {

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

ClassPool pool = ClassPool.getDefault();

CtClass cc = pool.makeClass(“com.lyy.test.Emp1”);

//创建属性

CtField f1 = CtField.make(“private int empno;”, cc);

CtField f2 = CtField.make(“private String ename;”, cc);

cc.addField(f1);

cc.addField(f2);

小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Java)
img

线程、数据库、算法、JVM、分布式、微服务、框架、Spring相关知识

一线互联网P7面试集锦+各种大厂面试集锦

学习笔记以及面试真题解析

ist.CtField;

import javassist.CtMethod;

import javassist.NotFoundException;

/**

  • 测试使用Javasost生成一个新的类

  • @author lyy

*/

public class Demo1 {

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

ClassPool pool = ClassPool.getDefault();

CtClass cc = pool.makeClass(“com.lyy.test.Emp1”);

//创建属性

CtField f1 = CtField.make(“private int empno;”, cc);

CtField f2 = CtField.make(“private String ename;”, cc);

cc.addField(f1);

cc.addField(f2);

小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-csjOws2J-1710402146038)]
[外链图片转存中…(img-0BhYrI7n-1710402146039)]
[外链图片转存中…(img-0RID9Ztu-1710402146039)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Java)
[外链图片转存中…(img-j6E0uRY1-1710402146040)]

线程、数据库、算法、JVM、分布式、微服务、框架、Spring相关知识

[外链图片转存中…(img-oUUH5Gyp-1710402146040)]

一线互联网P7面试集锦+各种大厂面试集锦

[外链图片转存中…(img-rKOQdzvV-1710402146041)]

学习笔记以及面试真题解析

[外链图片转存中…(img-fNp9ZM4L-1710402146041)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

  • 6
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值