zoukankan      html  css  js  c++  java
  • Java反射调用某个类的方法(带参数和不带参数)


     
     

    不多说,直接贴代码,不懂查API,现在就列取要用的方法和类。

     

    Class类:

    public Method getDeclaredMethod(String name, Class<?>... parameterTypes) throws NoSuchMethodException,   SecurityException
    返回一个 Method 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明方法。name 参数是一个String,它指定所需方法的简称,parameterTypes 参数是 Class 对象的一个数组,它按声明顺序标识该方法的形参类型。如果在某个类中声明了带有相同参数类型的多个方法,并且其中有一个方法的返回类型比其他方法的返回类型都特殊,则返回该方法;否则将从中任选一个方法。如果名称是 "<init>” 或 “<clinit>",则引发一个 NoSuchMethodException

     

    参数:
    name - 方法名
    parameterTypes - 参数数组
    返回:
    该类与指定名和参数相匹配的方法的 Method 对象
    抛出:
    NoSuchMethodException - 如果找不到匹配的方法。
    NullPointerException - 如果 name  null
    SecurityException - 如果存在安全管理器 s,并满足下列任一条件:
    从以下版本开始:
    JDK1.1
    Method类:
    public Object invoke(Object obj, Object... args)   throws IllegalAccessException,IllegalArgumentException, InvocationTargetException
    对带有指定参数的指定对象调用由此 Method 对象表示的底层方法。个别参数被自动解包,以便与基本形参相匹配,基本参数和引用参数都随需服从方法调用转换。

    如果底层方法是静态的,那么可以忽略指定的 obj 参数。该参数可以为 null。

    如果底层方法所需的形参数为 0,则所提供的 args 数组长度可以为 0 或 null。

    如果底层方法是实例方法,则使用动态方法查找来调用它,这一点记录在 Java Language Specification, Second Edition 的第 15.12.4.4 节中;在发生基于目标对象的运行时类型的重写时更应该这样做。

    如果底层方法是静态的,并且尚未初始化声明此方法的类,则会将其初始化。

    如果方法正常完成,则将该方法返回的值返回给调用者;如果该值为基本类型,则首先适当地将其包装在对象中。但是,如果该值的类型为一组基本类型,则数组元素 被包装在对象中;换句话说,将返回基本类型的数组。如果底层方法返回类型为 void,则该调用返回 null。

     

    参数:
    obj - 从中调用底层方法的对象
    args - 用于方法调用的参数
    返回:
    使用参数 args  obj 上指派该对象所表示方法的结果
    抛出:
    IllegalAccessException - 如果此 Method 对象强制执行 Java 语言访问控制,并且底层方法是不可访问的。
    IllegalArgumentException - 如果该方法是实例方法,且指定对象参数不是声明底层方法的类或接口(或其中的子类或实现程序)的实例;如果实参和形参的数量不相同;如果基本参数的解包转换失败;如果在解包后,无法通过方法调用转换将参数值转换为相应的形参类型。
    InvocationTargetException - 如果底层方法抛出异常。
    NullPointerException - 如果指定对象为 null,且该方法是一个实例方法。
    ExceptionInInitializerError - 如果由此方法引起的初始化失败。

     

    这个动态调用类的方法非常好用,可以根据字符串动态调用某个类的方法,灵活性大,但记住反射是比较耗效率,要保留的用。可以这么用,通过一个文本来解析,动态实现某个类的计算方法,通过一个文本就可以方便的调用方法的机制。比较灵活。

     

     

    下面是代码,已经封装好了:

     

    /*
     * To change this template, choose Tools | Templates
     * and open the template in the editor.
     */
    package com.jijing.method;

    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    import java.util.logging.Level;
    import java.util.logging.Logger;

    /**
     *
     * @author Administrator
     * 用来通过类名反射当前类的某个方法,包括带参数的方法,当然只可以是public的方法
     */
    public class ClassMethod {

        /**
         * @param args the command line arguments
         */
        public static void main(String[] args) {
            // TODO code application logic here
            MyMethod m=new MyMethod();
            getMethod("show",m,null);//这个方法可以根据一个字符串来动态创建某个类的相应方法,这个思路非常好。
            getMethod("show",m,new Object[]{20});
            //包括自定义类 
            getMethod("show",m,new Object[]{100,"今天天气不错",new NewClass()});
        }
        
        /**
         * 
         * @param className 类的路劲
         * @return 返回Class的类信息 
         */
        public static Class getclass(String className){
            Class c=null;
            try {
                c=Class.forName(className);
            } catch (ClassNotFoundException ex) {
                Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
            }
            return c;
        }
        
        /**
         * 
         * @param MethodName 
         * @param o           调用此方法的对象
         * @param paras       调用的这个方法的参数参数列表
         */
        public static void  getMethod(String MethodName,Object o,Object []paras){
             Class c[]=null;
             if(paras!=null){//存在
                 int len=paras.length;
                 c=new Class[len];
                 for(int i=0;i<len;++i){
                     c[i]=paras[i].getClass();
                 }
             }
            try {
                Method method=o.getClass().getDeclaredMethod(MethodName,c);
                try {
                    method.invoke(o,paras);//调用o对象的方法
                } catch (IllegalAccessException ex) {
                    Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
                } catch (IllegalArgumentException ex) {
                    Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
                } catch (InvocationTargetException ex) {
                    Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
                }
            } catch (NoSuchMethodException ex) {
                Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
            } catch (SecurityException ex) {
                Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
            }
        }
    }

    /**
     * 
     * @author Administrator
     *   反射这个类的某个方法
     */
    class MyMethod{
        public MyMethod(){
        }
        
        /**
         * 
         */
        public void show(){
            System.out.println("我是MyMethod");
        }
        
        /**
         * 
         * @param temp 带参数的的
         */
        public void show(Integer temp){
            System.out.println("**************************="+temp.intValue());
        }
        
        public void show(Integer a,String name,NewClass nc){
            System.out.println("Integer="+a+" name="+name+" NewClass="+nc);
        }
    }

    class NewClass{
        public void print(){
            System.out.println("我是NewClass()");
        }
        @Override
        public String toString(){
            return "我是NewClass";
        }
    }

     

    上面封装的方法是:

     

     

    /**
         * 
         * @param MethodName 
         * @param o           调用此方法的对象
         * @param paras       调用的这个方法的参数参数列表
         */
        public static void  getMethod(String MethodName,Object o,Object []paras){
             Class c[]=null;
             if(paras!=null){//存在
                 int len=paras.length;
                 c=new Class[len];
                 for(int i=0;i<len;++i){//这里不是java内置类基本数据是不支持的,请一步一步完成,可以参考我的上篇文章<Java通过反射创建对象>
                     c[i]=paras[i].getClass();
                 }
             }
            try {
                Method method=o.getClass().getDeclaredMethod(MethodName,c);
                try {
                    method.invoke(o,paras);//调用o对象的方法
                } catch (IllegalAccessException ex) {
                    Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
                } catch (IllegalArgumentException ex) {
                    Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
                } catch (InvocationTargetException ex) {
                    Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
                }
            } catch (NoSuchMethodException ex) {
                Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
            } catch (SecurityException ex) {
                Logger.getLogger(ClassMethod.class.getName()).log(Level.SEVERE, null, ex);
            }
        }
    }

     

     

    注意:上面的方法是不支持内置类,因为内置类没有进行基本数据的转换,所以无法完成。为了可以完全正常运行,请转换上面的代码。

  • 相关阅读:
    angular.isDefined()
    angular.isDate()
    angular.isArray()
    .NET中栈和堆的比较
    SQL Server 2012配置Always On可用性组
    一分钟了解负载均衡的一切
    C# 线程并发锁
    获取Http请求参数
    什么是WCF
    Bitmap算法应用
  • 原文地址:https://www.cnblogs.com/ningxu/p/3381342.html
Copyright © 2011-2022 走看看