利用java反射调用类的的私有方法--转
原文:http://blog.csdn.net/woshinia/article/details/11766567
1,今天和一位朋友谈到父类私有方法的调用问题,本来以为利用反射很轻松就可以实现,因为在反射看来根本不区分是否是private的,没有想到调用本身的私有方法是可以的,但是调用父类的私有方法则不行,后来纠其原因很有可能是因为getDeclaredMethod方法和getMethod方法并不会查找父类的私有方法,于是只好自己写递归了,经过尝试果然如此。把代码放出来方便更多人。这段代码可以解决很多实际问题,不过利用反射来做的话性能不会太好。
- package com.syj.util.reflect;
- import java.lang.reflect.Method;
- /**
- * <p>
- * Title: 私有方法调用工具类
- * </p>
- *
- * <p>
- * Description:利用java反射调用类的的私有方法
- * </p>
- *
- * <p>
- * Copyright: Copyright (c) 2007
- * </p>
- *
- * @author 孙钰佳
- * @main sunyujia@yahoo.cn
- * @date Jun 1, 2008 10:18:58 PM
- */
- public class PrivateUtil {
- /**
- * 利用递归找一个类的指定方法,如果找不到,去父亲里面找直到最上层Object对象为止。
- *
- * @param clazz
- * 目标类
- * @param methodName
- * 方法名
- * @param classes
- * 方法参数类型数组
- * @return 方法对象
- * @throws Exception
- */
- public static Method getMethod(Class clazz, String methodName,
- final Class[] classes) throws Exception {
- Method method = null;
- try {
- method = clazz.getDeclaredMethod(methodName, classes);
- } catch (NoSuchMethodException e) {
- try {
- method = clazz.getMethod(methodName, classes);
- } catch (NoSuchMethodException ex) {
- if (clazz.getSuperclass() == null) {
- return method;
- } else {
- method = getMethod(clazz.getSuperclass(), methodName,
- classes);
- }
- }
- }
- return method;
- }
- /**
- *
- * @param obj
- * 调整方法的对象
- * @param methodName
- * 方法名
- * @param classes
- * 参数类型数组
- * @param objects
- * 参数数组
- * @return 方法的返回值
- */
- public static Object invoke(final Object obj, final String methodName,
- final Class[] classes, final Object[] objects) {
- try {
- Method method = getMethod(obj.getClass(), methodName, classes);
- method.setAccessible(true);// 调用private方法的关键一句话
- return method.invoke(obj, objects);
- } catch (Exception e) {
- throw new RuntimeException(e);
- }
- }
- public static Object invoke(final Object obj, final String methodName,
- final Class[] classes) {
- return invoke(obj, methodName, classes, new Object[] {});
- }
- public static Object invoke(final Object obj, final String methodName) {
- return invoke(obj, methodName, new Class[] {}, new Object[] {});
- }
- /**
- * 测试反射调用
- *
- * @param args
- */
- public static void main(String[] args) {
- PrivateUtil.invoke(new B(), "printlnA", new Class[] { String.class },
- new Object[] { "test" });
- PrivateUtil.invoke(new B(), "printlnB");
- }
- }
- class A {
- private void printlnA(String s) {
- System.out.println(s);
- }
- }
- class B extends A {
- private void printlnB() {
- System.out.println("b");
- }
- }
程序的输出结果为
test
b
说明private方法调用成功了不管是自己的私有方法还是父类的私有方法。\
2,
- package me.test;
- import java.lang.reflect.*; //导入反射需要的包
- public class ReflectTest {
- public static void main(String[] args) throws Exception
- {
- /* 下面通过反射完成对一个对象中成员的替换
- * 并且执行执行私有方法
- * 完成对 Poiont类的对象中所有的 String的对象的d换成x
- * 并且类中无修改方法
- */
- Point pt=new Point(3,5); //创建一个Point对象
- Field fx=pt.getClass().getField("x") ; //获取x的映射类对象
- Field fy=pt.getClass().getDeclaredField("y");//因为y是私有的所以要调用这个方法
- Method m2=Point.class.getDeclaredMethod("showPrivate") ;//获得私有方法映射类
- //利用反射调用共有输出
- m2.setAccessible(true) ;// 修改showPrivate 权限 改变为可以调用
- m2.invoke(pt) ;//执行私有方法
- //利用成员反射输出x 和 私有的 y
- System.out.println(fx.getInt(pt));//反射输出x
- fy.setAccessible(true) ;//改变私有为可访问
- System.out.println(fy.getInt(pt));//输出私有y
- //替换成员后并且反射私有方法输出
- changeString(pt) ;//反射替换成员值
- System.out.println(pt);
- }
- public static void changeString(Object obj) throws Exception//反射替换对所有String进行替换
- {
- Field[] f=obj.getClass().getFields() ; //获得成员映射数组
- for(Field tem : f) //迭代for循环
- {
- if(tem.getType()==String.class) //内存中只有一份String字节码
- {
- String oldString=(String)tem.get(obj) ; //返回内容
- String newString=oldString.replace('d', 'x');//将所有b替换为x
- tem.setAccessible(true);
- tem.set(obj, newString) ;//替换成员值
- }
- }
- }
- }
- public class Point
- {
- public int x ;
- private int y ;
- public Point(int x, int y) {
- super();
- this.x = x;
- this.y = y;
- }
- public String a="dsfdsfd" ; //只有 共有可以替换
- public String b="fdsfdsfewewwwww" ;
- public String c="adddssss" ;
- private void showPrivate() //私有方法输出
- {
- System.out.println("x="+this.x+"\n"+"y="+this.y);
- System.out.println(this.a);
- System.out.println(this.b);
- System.out.println(this.c);
- }
- public String toString()
- {
- return this.a+"\n"+this.b+"\n"+this.c;
- }
- }
3,
Java利用反射来获取一个方法的 范型化参数 Vector<Integer>的类型
- class A
- {
- public void show(Vector<Integer> v) {}
- }
- 在我们不知道Vector中数据的类型的时候 这时候我们只知道这个方法的名字 和参数的个数 ,我们来获取 范型化的实际类型 。
- 我们不可能通过 Vector对应的Class类来反射出 泛型集合中的类型 ,但是 我们却可以通过 这个方法所对应的Method类来实现 。
- 具体如下 :
- import java.lang.reflect.Method;
- import java.lang.reflect.ParameterizedType;
- import java.lang.reflect.Type;
- import java.util.Vector;
- public class Test3
- {
- public static void main(String []args) throws SecurityException, NoSuchMethodException
- {
- Method m=A.class.getMethod("show", Vector.class) ; //反射获得show方法的Method对象
- Type[]t=m.getGenericParameterTypes() ; //获得范型参数的 一个Type数组 Type是Class类的基类 GenericArrayType, ParameterizedType, TypeVariable<D>, WildcardType 这些都是 实现Type的子接口
- ParameterizedType p=(ParameterizedType)t[0]; //强制转换成Type的子接口 ParameterizedType类型 因为这个接口又可以获得 范型化集合中元素的类型 System.out.println(p.getRawType()); //获得集合的类型
- System.out.println(p.getActualTypeArguments()[0]); //获得集合中元素的类型
- }
- }
利用java反射调用类的的私有方法--转的更多相关文章
- 利用Java反射根据类的名称获取属性信息和父类的属性信息
代码: import java.lang.reflect.Field; import java.util.ArrayList; import java.util.Arrays; import java ...
- 利用java反射机制 读取配置文件 实现动态类载入以及动态类型转换
作者:54dabang 在spring的学习过程之中,我们能够看出通过配置文件来动态管理bean对象的优点(松耦合 能够让零散部分组成一个总体,而这些总体并不在意之间彼此的细节,从而达到了真正的物理上 ...
- Java 反射 调用私有域和方法(setAccessible)
Java 反射 调用私有域和方法(setAccessible) @author ixenos AccessibleObject类 Method.Field和Constructor类共同继承了Acces ...
- 利用Java反射机制对实体类的常用操作工具类ObjectUtil
代码: ObjectUtil类: import java.lang.reflect.Field; import java.math.BigDecimal; import java.text.Simpl ...
- PHP通过反射方法调用执行类中的私有方法
PHP 5 具有完整的反射 API,添加了对类.接口.函数.方法和扩展进行反向工程的能力. 下面我们演示一下如何通过反射,来调用执行一个类中的私有方法: <?php //MyClass这个类中包 ...
- C# 利用反射调用类下的方法
namespace TestReflection { public partial class Form1 : Form { public Form1() { InitializeComponent( ...
- java反射工具类
package com.yingchao.kgou.core; import java.lang.reflect.Field; import java.lang.reflect.InvocationT ...
- Java 反射 分析类和对象
Java 反射 分析类和对象 @author ixenos 摘要:优化程序启动策略.在运行时使用反射分析类的结构和对象 优化程序启动策略 在启动时,包含main方法的类被加载.它会加载所有它需要的类. ...
- 利用Java反射实现JavaBean对象相同属性复制并初始化目标对象为空的属性的BeanUtils
有时遇到将数据传输对象转换成JSON串会将属性值为空的属性去掉,利用Java反射实现JavaBean对象数据传输对象的相同属性复制并初始化数据传输对象属性为空的属性,然后转换成JSON串 packag ...
随机推荐
- python接口自动化测试 - unittest框架基本使用
unittest简单介绍 单元测试框架 还可以适用WEB自动化测试用例的开发与执行 提供丰富的断言方法 官方文档:https://docs.python.org/zh-cn/3/library/uni ...
- selenium 使用close和quit关闭driver的不同点
Driver.Quit()与Driver.Close()的不同:Driver.Quit(): Quit this dirver, closing every associated windows;Dr ...
- java8date
Java 8:新的时间和日期API 在Java 8之前,所有关于时间和日期的API都存在各种使用方面的缺陷,因此建议使用新的时间和日期API,分别从旧的时间和日期的API的缺点以及解决方法.Java ...
- zookeeper使用及安装
1.安装查看:http://ifeve.com/zookeeper-talk-quick-start/ 2.使用: Zookeeper是一个高性能的分布式应用协调服务的框架.Zookeeper=Zoo ...
- canvas的描述
// 1.找到DOM节点 const canvas = document.getElementById('canvas'); // 2.画笔 --- canvas的上下文对象 const ctx = ...
- Mac安装php扩展redis遇到的问题,执行phpize问题
1.安装redis在mac OS中可以使用brew命令进行安装redis:mac OS使用brew命令安装软件安装命令:brew install redis因为我已经安装过了,这里就不在赘述.安装完之 ...
- (学习1)最小生成树-Prim算法与Kruskal算法
最小生成树: 求一个有 n 个结点的连通图的生成树是原图的极小连通子图,且包含原图中的所有 n 个结点,并且有保持图连通的最少的边. 1:Prim算法(适合稠密图) 伪代码: Prim(G){ //G ...
- selenium+python实现自动化登录
工作需要实现一个微博自动登录的操作,在网上差了一些资料,决定使用selenium+python实现 selenium 是一个web的自动化测试工具,主流一般配合java或者python使用,我这里使用 ...
- VMWare tools
一.首先是安装VMWare tools1.以ROOT身份进入LINUX2.在虚拟机软件VMWARE状态栏中,点击 SETTING菜单下的ENABLE VMWARE TOOLS子菜单,此时在linux的 ...
- CSS实现心形、六角星、六边形、平行四边形等几何
本文将利用border属性实现简单几何的绘制: 效果图: 正八角星 说明:采用两个正方形以中心进行旋转叠加: /* 八角星 */ #burst-8 { background: #6376ff1f; w ...