实战Java的反射机制

  众所周知,Java要调用某个对象的方法首先需要对象实例化后才能调用。

  而实例化对象常见的就是new执行和spring(DI)的依赖注入了。

  Spring的DI其实就是以反射作为最基础的技术手段。

  

一、反射机制实战

  1.反射概念

  指项目在运行状态下,可以知道任意一个类所有属性和方法,并且能够调用它的任意一个方法来获取信息,将这种动态调用对象方法的功能叫java语言的反射机制。

  2.应用场景:

  生成动态代理,面向切片编程。(ps:这两种是最常见的使用场景。而我写这篇文章的时候,遇到的不是这两种需求,而是需要知道某个方法的返回值。情况如下:正式环境的对接接口返回值解析错误,对接厂家的项目升级的时候,接口返回值进行了改动却没有进行通知,但是生产环境又不能debug,日志刚刚好没记录返回值,对接厂家又刚刚好非常忙没空理我,为了预防日后类似的场景,需要知道某个方法的返回值,这时我就想到了反射机制)

  3.实例化

  正常情况下我们用new方式实例化对象需要两步,(1)引入包名:import xx.xx.xx。(2)然后new 对象类名。

  反射通过字符串实例化对象也是两步:(1)Class clazz = Class.forName("包名");相当于上面的第一步。(2)clazz.newInstance()

  4.代码实战

@RequestMapping("agent")
public ModelAndView agent(HttpServletRequest request) throws Exception {
//返回值
Object obj = null;
ModelAndView mv = getAutoView();
try {
ISysUser user = ContextUtil.getCurrentUser();
if (user==null)
return getAutoView();
String un =user.getFullname();
    //简单的权限判断,这判断可以根据实际情况来
ResultMessage resultMessage = new ResultMessage();
if(un.equals("超级管理员")) {
Class clazz = null;
String action = request.getParameter("action");
mv.addObject("action", action);
if (action==null || action.isEmpty())
return mv; Object object = null;
if (action.contains(".")) {
//获取对应包的class类
clazz = Class.forName(action);
//找到IOC容器中的实例化
object = AppUtil.getBean(clazz);
} else {
object = AppUtil.getBean(action);
}
String method = request.getParameter("method");
mv.addObject("method", method);
//参数类型
String param = request.getParameter("paramNum");
mv.addObject("paramNum", param);
//参数值
String reqBody = request.getParameter("reqBody");
mv.addObject("reqBody", reqBody);
if(!reqBody.isEmpty()) {
JSONArray jsonArray = JSONArray.fromObject(reqBody);
String[] params = param.split(";");
if (params.length == jsonArray.size()){
Object[] objects = new Object[params.length];
Class[] objMethods = new Class[params.length];
for (int i = 0; i < params.length; i++) {
if (param.startsWith("java.lang.")) {
Object objParam = jsonArray.get(i);
             //基础类型判断
if (param.equals("java.lang.Integer")){
String str = String.valueOf(objParam);
Integer inv = Integer.parseInt(str);
objects[i] = inv;
} else if (param.equals("java.lang.Short")){
String str = String.valueOf(objParam);
Short inv = Short.parseShort(str);
objects[i] = inv;
} else if (param.equals("java.lang.Long")){
String str = String.valueOf(objParam);
Long inv = Long.parseLong(str);
objects[i] = inv;
} else if (param.equals("java.lang.Double")){
String str = String.valueOf(objParam);
Double inv = Double.parseDouble(str);
objects[i] = inv;
} else if (param.equals("java.lang.Float")){
String str = String.valueOf(objParam);
Float inv = Float.parseFloat(str);
objects[i] = inv;
} else if (param.equals("java.lang.Boolean")){
String str = String.valueOf(objParam);
Boolean inv = Boolean.parseBoolean(str);
objects[i] = inv;
} else {
objects[i] = objParam;
}
} else if (param.equals("java.util.Map")) {
Map map = jsonArray.getJSONObject(i);
objects[i] = map;
}else {
JSONObject jsonObject = jsonArray.getJSONObject(i);
Object cls = Class.forName(params[i]).newInstance();
Object objParam = JsonUtil.jsonTurnObject(jsonObject, cls);
objects[i] = objParam;
}
objMethods[i] = Class.forName(params[i]);
}
//获取想要执行的方法
Method md = object.getClass().getDeclaredMethod(method, objMethods);
obj = md.invoke(object, objects);//执行方法
System.out.println(obj);
} else
return getAutoView();
} else {
//获取想要执行的方法
Method md = object.getClass().getMethod(method);
obj = md.invoke(object);
System.out.println(obj);
} } else {
resultMessage.setMessage("你没有该权限!");
resultMessage.setResult(0);
}
} catch (Exception e) {
e.printStackTrace();
throw e;
}
if (obj!=null){
System.out.println(obj.getClass().toString());
switch (obj.getClass().toString()){
case "class java.lang.String":
break;
case "class java.lang.Long":
break;
case "class java.lang.Integer":
break;
case "class java.lang.Short":
break;
case "class java.lang.Byte":
break;
case "class java.lang.Float":
break;
case "class java.lang.Double":
break;
case "class java.lang.Boolean":
break;
case "class java.lang.Char":
break;
default:
obj = JSONArray.fromObject(obj);
break;
}
}
return mv.addObject("obj", obj);
}

该工具类就是将json转为相对应的类对象。

public static <T> T jsonTurnObject(JSONObject json, T clazz) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Field[] field = clazz.getClass().getDeclaredFields(); //获取实体类的所有属性,返回Field数组
for(int j=0 ; j<field.length ; j++){ //遍历所有属性
String name = field[j].getName(); //获取属性的名字
// System.out.println("attribute name:"+name);
Object object = json.get(name);//获取json属性
if (object != null) {
name = name.substring(0,1).toUpperCase()+name.substring(1); //将属性的首字符大写,方便构造get,set方法
Method fieldSetMet = clazz.getClass().getMethod("set"
+ name, field[j].getType()); String type = field[j].getGenericType().toString(); //获取属性的类型
if(type.equals("class java.lang.String")){ //如果type是类类型,则前面包含"class ",后面跟类名
fieldSetMet.invoke(clazz, object);
} else if (type.equals("class java.lang.Integer")){
String str = String.valueOf(object);
Integer inv = Integer.parseInt(str);
fieldSetMet.invoke(clazz, inv);
} else if (type.equals("class java.lang.Short")){
String str = String.valueOf(object);
Short inv = Short.parseShort(str);
fieldSetMet.invoke(clazz, inv);
} else if (type.equals("class java.lang.Long")){
String str = String.valueOf(object);
Long inv = Long.parseLong(str);
fieldSetMet.invoke(clazz, inv);
} else if (type.equals("class java.lang.Double")){
String str = String.valueOf(object);
Double inv = Double.parseDouble(str);
fieldSetMet.invoke(clazz, inv);
} else if (type.equals("class java.lang.Float")){
String str = String.valueOf(object);
Float inv = Float.parseFloat(str);
fieldSetMet.invoke(clazz, inv);
} else if (type.equals("class java.lang.Boolean")){
String str = String.valueOf(object);
Boolean inv = Boolean.parseBoolean(str);
fieldSetMet.invoke(clazz, inv);
}
}
} return clazz;
}

页面:

<form id="testWebserviceForm" method="post" action="agent.ht">
<table class="table table-striped table-hover table-bordered custom-table2" cellpadding="0" cellspacing="0" border="0" type="main">
<tr>
<th width="20%">包名: </th>
<td colspan="3"><input type="text" id="action" name="action" class="inputText" value="${action}"
validate="{required:false,maxlength:32}" placeholder="例:java.util.Map" /></td>
</tr>
<tr>
<th width="20%">方法名: </th>
<td colspan="3"><input type="text" id="method" name="method" class="inputText" value="${method}"
validate="{required:false,maxlength:255}" placeholder="例:put" /></td>
</tr>
<tr>
<th width="20%">参数类型: </th>
<td colspan="3"><input type="text" id="paramNum" name="paramNum" class="inputText" value="${paramNum}"
validate="{required:false,maxlength:255}" placeholder="例:java.lang.String(多个用;隔开)" /></td>
</tr>
<tr>
<th width="20%">参数: </th>
<td colspan="3">
<textarea id="reqBody" name="reqBody" class="inputText" rows="15" cols="50"
placeholder="例JSONArray:[{},{}]">${reqBody}</textarea>
</td>
</tr>
<tr>
<th width="20%">操作: </th>
<td colspan="3">
<button class="btn btn-xs btn-circle blue save" type="submit"><i
class="fa fa-sticky-note"></i>提交</button>
</td>
</tr>
</table>
</form>

附上运行结果

实战Java的反射机制

上一篇:C#基础系列 - 反射基础


下一篇:Kotlin反射实践操作详解