//List<Object> newarrayList = arrayList.stream().distinct().collect(Collectors.toList());
ArrayList<String> newarrayList = new ArrayList<String>(new HashSet<String>(arrayList));
反射,当时经常听他们说,自己也看过一些资料,也可能在设计模式中使用过,但是感觉对它没有一个较深入的了解,这次重新学习了一下,感觉还行吧!
一,先看一下反射的概念:
主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。
反射是java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接。但是反射使用不当会成本很高!
看概念很晕的,继续往下看。
二,反射机制的作用:
1,反编译:.class-->.java
2,通过反射机制访问java对象的属性,方法,构造方法等;
这样好像更容易理解一些,下边我们具体看怎么实现这些功能。
三,在这里先看一下sun为我们提供了那些反射机制中的类:
java.lang.Class;
java.lang.reflect.Constructor; java.lang.reflect.Field;
java.lang.reflect.Method;
java.lang.reflect.Modifier;
很多反射中的方法,属性等操作我们可以从这四个类中查询。还是哪句话要学着不断的查询API,那才是我们最好的老师。
四,具体功能实现:
1,反射机制获取类有三种方法,我们来获取Employee类型
[java] view plain copy print?
- //第一种方式:
- Classc1 = Class.forName("Employee");
- //第二种方式:
- //java中每个类型都有class 属性.
- Classc2 = Employee.class;
- //第三种方式:
- //java语言中任何一个java对象都有getClass 方法
- Employeee = new Employee();
- Classc3 = e.getClass(); //c3是运行时类 (e的运行时类是Employee)
2,创建对象:获取类以后我们来创建它的对象,利用newInstance:
[java] view plain copy print?
- Class c =Class.forName("Employee");
- //创建此Class 对象所表示的类的一个新实例
- Objecto = c.newInstance(); //调用了Employee的无参数构造方法.
3,获取属性:分为所有的属性和指定的属性:
a,先看获取所有的属性的写法:
[java] view plain copy print?
- //获取整个类
- Class c = Class.forName("java.lang.Integer");
- //获取所有的属性?
- Field[] fs = c.getDeclaredFields();
- //定义可变长的字符串,用来存储属性
- StringBuffer sb = new StringBuffer();
- //通过追加的方法,将每个属性拼接到此字符串中
- //最外边的public定义
- sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() +"{\n");
- //里边的每一个属性
- for(Field field:fs){
- sb.append("\t");//空格
- sb.append(Modifier.toString(field.getModifiers())+" ");//获得属性的修饰符,例如public,static等等
- sb.append(field.getType().getSimpleName() + " ");//属性的类型的名字
- sb.append(field.getName()+";\n");//属性的名字+回车
- }
- sb.append("}");
- System.out.println(sb);
b,获取特定的属性,对比着传统的方法来学习:
[java] view plain copy print?
- public static void main(String[] args) throws Exception{
- <span style="white-space:pre"> </span>//以前的方式:
- /*
- User u = new User();
- u.age = 12; //set
- System.out.println(u.age); //get
- */
- //获取类
- Class c = Class.forName("User");
- //获取id属性
- Field idF = c.getDeclaredField("id");
- //实例化这个类赋给o
- Object o = c.newInstance();
- //打破封装
- idF.setAccessible(true); //使用反射机制可以打破封装性,导致了java对象的属性不安全。
- //给o对象的id属性赋值"110"
- idF.set(o, "110"); //set
- //get
- System.out.println(idF.get(o));
- }
4,获取方法,和构造方法,不再详细描述,只来看一下关键字:
方法关键字
含义
getDeclaredMethods()
获取所有的方法
getReturnType()
获得方法的放回类型
getParameterTypes()
获得方法的传入参数类型
getDeclaredMethod("方法名",参数类型.class,……)
获得特定的方法
构造方法关键字
含义
getDeclaredConstructors()
获取所有的构造方法
getDeclaredConstructor(参数类型.class,……)
获取特定的构造方法
父类和父接口
含义
getSuperclass()
获取某类的父类
getInterfaces()
获取某类实现的接口
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,并满足下列任一条件:
- 调用
s.checkMemberAccess(this, Member.DECLARED)
拒绝访问已声明方法 - 调用者的类加载器不同于也不是当前类的类加载器的一个祖先,并且对
s.checkPackageAccess()
的调用拒绝访问该类的包
public Field getDeclaredField(String name)
throws NoSuchFieldException,
SecurityException
返回一个 Field
对象,该对象反映此 Class
对象所表示的类或接口的指定已声明字段。name
参数是一个String
,它指定所需字段的简称。注意,此方法不反映数组类的length
字段。
参数:
name
- 字段名
返回:
此类中指定字段的 Field
对象
抛出:
NoSuchFieldException
- 如果找不到带有指定名称的字段。
NullPointerException
- 如果name
为null
SecurityException
- 如果存在安全管理器s,并满足下列任一条件:
- 调用
s.checkMemberAccess(this, Member.DECLARED)
拒绝访问已声明字段 - 调用者的类加载器不同于也不是当前类的类加载器的一个祖先,并且对
s.checkPackageAccess()
的调用拒绝访问该类的包
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)
throws NoSuchMethodException,
SecurityException
返回一个 Constructor
对象,该对象反映此 Class
对象所表示的类或接口的指定构造方法。parameterTypes
参数是Class
对象的一个数组,它按声明顺序标识构造方法的形参类型。 如果此Class
对象表示非静态上下文中声明的内部类,则形参类型作为第一个参数包括显示封闭的实例。
参数:
parameterTypes
- 参数数组
返回:
带有指定参数列表的构造方法的 Constructor
对象
抛出:
NoSuchMethodException
- 如果找不到匹配的方法。
SecurityException
- 如果存在安全管理器s,并满足下列任一条件:
- 调用
s.checkMemberAccess(this, Member.DECLARED)
拒绝访问已声明的构造方法 - 调用者的类加载器不同于也不是当前类的类加载器的一个祖先,并且对
s.checkPackageAccess()
的调用拒绝访问该类的包
首先得根据传入的类的全名来创建Class对象。
Class c=Class.forName("className");注明:className必须为全名,也就是得包含包名,比如,cn.netjava.pojo.UserInfo;
也可以使用:对象名.getClass();或者类名.class
Object obj=c.newInstance();//创建对象的实例
OK,有了对象就什么都好办了,想要什么信息就有什么信息了。
获得构造函数的方法
Constructor getConstructor(Class[] params)//根据指定参数获得public构造器
Constructor[] getConstructors()//获得public的所有构造器
Constructor getDeclaredConstructor(Class[] params)//根据指定参数获得public和非public的构造器
Constructor[] getDeclaredConstructors()//获得public的所有构造器
获得类方法的方法
Method getMethod(String name, Class[] params),根据方法名,参数类型获得方法
Method[] getMethods()//获得所有的public方法
Method getDeclaredMethod(String name, Class[] params)//根据方法名和参数类型,获得public和非public的方法
Method[] getDeclaredMethods()//获得所以的public和非public方法
获得类中属性的方法
Field getField(String name)//根据变量名得到相应的public变量
Field[] getFields()//获得类中所以public的方法
Field getDeclaredField(String name)//根据方法名获得public和非public变量
Field[] getDeclaredFields()//获得类中所有的public和非public方法
这样我们就可以获得类的各种内容,进行了反编译。对于JAVA这种先编译再运行的语言来说,反射机制可以使代码更加灵活,更加容易实现面向对象。
五,反射加配置文件,使我们的程序更加灵活:
在设计模式学习当中,学习抽象工厂的时候就用到了反射来更加方便的读取数据库链接字符串等,当时不是太理解,就照着抄了。看一下.NET中的反射+配置文件的使用:
当时用的配置文件是app.config文件,内容是XML格式的,里边填写链接数据库的内容:
[html] view plain copy print?
- <configuration>
- lt;appSettings>
- <add key="" value=""/>
- lt;/appSettings>
- </configuration>
反射的写法:
[csharp] view plain copy print?
- assembly.load("当前程序集的名称").CreateInstance("当前命名空间名称".要实例化的类名);
这样的好处是很容易的方便我们变换数据库,例如我们将系统的数据库从SQL Server升级到Oracle,那么我们写两份D层,在配置文件的内容改一下,或者加条件选择一下即可,带来了很大的方便。
当然了,JAVA中其实也是一样,只不过这里的配置文件为.properties,称作属性文件。通过反射读取里边的内容。这样代码是固定的,但是配置文件的内容我们可以改,这样使我们的代码灵活了很多!
AOP的拦截功能是由java中的动态代理来实现的。说白了,就是在目标类的基础上增加切面逻辑,生成增强的目标类(该切面逻辑或者在目标类函数执行之前,或者目标类函数执行之后,或者在目标类函数抛出异常时候执行。不同的切入时机对应不同的Interceptor的种类,如BeforeAdviseInterceptor,AfterAdviseInterceptor以及ThrowsAdviseInterceptor等)。
那么动态代理是如何实现将切面逻辑(advise)织入到目标类方法中去的呢?下面我们就来详细介绍并实现AOP中用到的两种动态代理。
AOP的源码中用到了两种动态代理来实现拦截切入功能:jdk动态代理和cglib动态代理。两种方法同时存在,各有优劣。jdk动态代理是由java内部的反射机制来实现的,cglib动态代理底层则是借助asm来实现的。总的来说,反射机制在生成类的过程中比较高效,而asm在生成类之后的相关执行过程中比较高效(可以通过将asm生成的类进行缓存,这样解决asm生成类过程低效问题)。还有一点必须注意:jdk动态代理的应用前提,必须是目标类基于统一的接口。如果没有上述前提,jdk动态代理不能应用。由此可以看出,jdk动态代理有一定的局限性,cglib这种第三方类库实现的动态代理应用更加广泛,且在效率上更有优势。。
1、定义接口和实现
[java] view plain copy print?
- package com.meituan.hyt.test3.service;
- public interface UserService {
- public String getName(int id);
- public Integer getAge(int id);
- }
[java] view plain copy print?
- package com.meituan.hyt.test3.service.impl;
- import com.meituan.hyt.test3.service.UserService;
- public class UserServiceImpl implements UserService {
- @Override
- public String getName(int id) {
- System.out.println("------getName------");
- return "Tom";
- }
- @Override
- public Integer getAge(int id) {
- System.out.println("------getAge------");
- return 10;
- }
- }
2、jdk动态代理实现
[java] view plain copy print?
- package com.meituan.hyt.test3.jdk;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- public class MyInvocationHandler implements InvocationHandler {
- private Object target;
- MyInvocationHandler() {
- super();
- }
- MyInvocationHandler(Object target) {
- super();
- this.target = target;
- }
- @Override
- public Object invoke(Object o, Method method, Object[] args) throws Throwable {
- if("getName".equals(method.getName())){
- System.out.println("++++++before " + method.getName() + "++++++");
- Object result = method.invoke(target, args);
- System.out.println("++++++after " + method.getName() + "++++++");
- return result;
- }else{
- Object result = method.invoke(target, args);
- return result;
- }
- }
- }
[java] view plain copy print?
- package com.meituan.hyt.test3.jdk;
- import com.meituan.hyt.test3.service.UserService;
- import com.meituan.hyt.test3.service.impl.UserServiceImpl;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Proxy;
- public class Main1 {
- public static void main(String[] args) {
- UserService userService = new UserServiceImpl();
- InvocationHandler invocationHandler = new MyInvocationHandler(userService);
- UserService userServiceProxy = (UserService)Proxy.newProxyInstance(userService.getClass().getClassLoader(),
- userService.getClass().getInterfaces(), invocationHandler);
- System.out.println(userServiceProxy.getName(1));
- System.out.println(userServiceProxy.getAge(1));
- }
- }
运行结果
++++++before getName++++++
------getName------
++++++after getName++++++
Tom
------getAge------
10
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
throws IllegalArgumentException
返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。此方法相当于:
Proxy.getProxyClass(loader, interfaces).
getConstructor(new Class[] { InvocationHandler.class }).
newInstance(new Object[] { handler });
Proxy.newProxyInstance
抛出 IllegalArgumentException
,原因与Proxy.getProxyClass
相同。
参数:
loader
- 定义代理类的类加载器
interfaces
- 代理类要实现的接口列表
h
- 指派方法调用的调用处理程序
返回:
一个带有代理类的指定调用处理程序的代理实例,它由指定的类加载器定义,并实现指定的接口
抛出:
IllegalArgumentException
- 如果违反传递到getProxyClass
的参数上的任何限制
NullPointerException
- 如果interfaces
数组参数或其任何元素为null
,或如果调用处理程序h
为null
3、cglib动态代理实现
Cglib是一个优秀的动态代理框架,它的底层使用ASM在内存中动态的生成被代理类的子类,使用CGLIB即使代理类没有实现任何接口也可以实现动态代理功能。CGLIB具有简单易用,它的运行速度要远远快于JDK的Proxy动态代理:
CGLIB的核心类:
net.sf.cglib.proxy.Enhancer – 主要的增强类
net.sf.cglib.proxy.MethodInterceptor – 主要的方法拦截类,它是Callback接口的子接口,需要用户实现
net.sf.cglib.proxy.MethodProxy – JDK的java.lang.reflect.Method类的代理类,可以方便的实现对源对象方法的调用,如使用:
Object o = methodProxy.invokeSuper(proxy, args);//虽然第一个参数是被代理对象,也不会出现死循环的问题。
net.sf.cglib.proxy.MethodInterceptor接口是最通用的回调(callback)类型,它经常被基于代理的AOP用来实现拦截(intercept)方法的调用。这个接口只定义了一个方法
public Object intercept(Object object, java.lang.reflect.Method method,
Object[] args, MethodProxy proxy) throws Throwable;
第一个参数是代理对像,第二和第三个参数分别是拦截的方法和方法的参数。原来的方法可能通过使用java.lang.reflect.Method对象的一般反射调用,或者使用 net.sf.cglib.proxy.MethodProxy对象调用。net.sf.cglib.proxy.MethodProxy通常被首选使用,因为它更快。
[java] view plain copy print?
- package com.meituan.hyt.test3.cglib;
- import net.sf.cglib.proxy.MethodInterceptor;
- import net.sf.cglib.proxy.MethodProxy;
- import java.lang.reflect.Method;
- public class CglibProxy implements MethodInterceptor {
- @Override
- public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
- System.out.println("++++++before " + methodProxy.getSuperName() + "++++++");
- System.out.println(method.getName());
- Object o1 = methodProxy.invokeSuper(o, args);
- System.out.println("++++++before " + methodProxy.getSuperName() + "++++++");
- return o1;
- }
- }
[java] view plain copy print?
- package com.meituan.hyt.test3.cglib;
- import com.meituan.hyt.test3.service.UserService;
- import com.meituan.hyt.test3.service.impl.UserServiceImpl;
- import net.sf.cglib.proxy.Enhancer;
- public class Main2 {
- public static void main(String[] args) {
- CglibProxy cglibProxy = new CglibProxy();
- Enhancer enhancer = new Enhancer();
- enhancer.setSuperclass(UserServiceImpl.class);
- enhancer.setCallback(cglibProxy);
- UserService o = (UserService)enhancer.create();
- o.getName(1);
- o.getAge(1);
- }
- }
运行结果:
++++++before CGLIB$getName$0++++++
getName
------getName------
++++++before CGLIB$getName$0++++++
++++++before CGLIB$getAge$1++++++
getAge
------getAge------
++++++before CGLIB$getAge$1++++++
JAVA反射机制
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
Java反射机制主要提供了以下功能: 在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。
1. 得到某个对象的属性
public Object getProperty(Object owner, String fieldName) throws Exception {
Class ownerClass = owner.getClass();
Field field = ownerClass.getField(fieldName);
Object property = field.get(owner);
return property;
}
public Object getProperty(Object owner, String fieldName) throws Exception {
Class ownerClass = owner.getClass();
Field field = ownerClass.getField(fieldName);
Object property = field.get(owner);
return property;
}
[java] view plain copy
- public Object getProperty(Object owner, String fieldName) throws Exception {
- Class ownerClass = owner.getClass();
- Field field = ownerClass.getField(fieldName);
- Object property = field.get(owner);
- return property;
- }
Class ownerClass = owner.getClass():得到该对象的Class。
Field field = ownerClass.getField(fieldName):通过Class得到类声明的属性。
Object property = field.get(owner):通过对象得到该属性的实例,如果这个属性是非公有的,这里会报IllegalAccessException。
2. 得到某个类的静态属性
public Object getStaticProperty(String className, String fieldName)
throws Exception {
Class ownerClass = Class.forName(className);
Field field = ownerClass.getField(fieldName);
Object property = field.get(ownerClass);
return property;
}
public Object getStaticProperty(String className, String fieldName)
throws Exception {
Class ownerClass = Class.forName(className);
Field field = ownerClass.getField(fieldName);
Object property = field.get(ownerClass);
return property;
}
[java] view plain copy
- public Object getStaticProperty(String className, String fieldName)
- throws Exception {
- Class ownerClass = Class.forName(className);
- Field field = ownerClass.getField(fieldName);
- Object property = field.get(ownerClass);
- return property;
- }
Class ownerClass = Class.forName(className) :首先得到这个类的Class。
Field field = ownerClass.getField(fieldName):和上面一样,通过Class得到类声明的属性。
Object property = field.get(ownerClass) :这里和上面有些不同,因为该属性是静态的,所以直接从类的Class里取。
3. 执行某对象的方法
public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {
Class ownerClass = owner.getClass();
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Method method = ownerClass.getMethod(methodName,argsClass);
return method.invoke(owner, args);
}
public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {
Class ownerClass = owner.getClass();
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Method method = ownerClass.getMethod(methodName,argsClass);
return method.invoke(owner, args);
}
[java] view plain copy
- public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {
- Class ownerClass = owner.getClass();
- Class[] argsClass = new Class[args.length];
- for (int i = 0, j = args.length; i < j; i++) {
- argsClass[i] = args[i].getClass();
- }
- Method method = ownerClass.getMethod(methodName,argsClass);
- return method.invoke(owner, args);
- }
Class owner_class = owner.getClass() :首先还是必须得到这个对象的Class。
5~9行:配置参数的Class数组,作为寻找Method的条件。
Method method = ownerClass.getMethod(methodName, argsClass):通过methodName和参数的argsClass(方法中的参数类型集合)数组得到要执行的Method。
method.invoke(owner, args):执行该Method.invoke方法的参数是执行这个方法的对象owner,和参数数组args,可以这么理解:owner对象中带有参数args的method方法。返回值是Object,也既是该方法的返回值。
4. 执行某个类的静态方法
public Object invokeStaticMethod(String className, String methodName,
Object[] args) throws Exception {
Class ownerClass = Class.forName(className);
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Method method = ownerClass.getMethod(methodName,argsClass);
return method.invoke(null, args);
}
public Object invokeStaticMethod(String className, String methodName,
Object[] args) throws Exception {
Class ownerClass = Class.forName(className);
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Method method = ownerClass.getMethod(methodName,argsClass);
return method.invoke(null, args);
}
[java] view plain copy
- public Object invokeStaticMethod(String className, String methodName,
- Object[] args) throws Exception {
- Class ownerClass = Class.forName(className);
- Class[] argsClass = new Class[args.length];
- for (int i = 0, j = args.length; i < j; i++) {
- argsClass[i] = args[i].getClass();
- }
- Method method = ownerClass.getMethod(methodName,argsClass);
- return method.invoke(null, args);
- }
基本的原理和实例3相同,不同点是最后一行,invoke的一个参数是null,因为这是静态方法,不需要借助实例运行。
5. 新建实例
public Object newInstance(String className, Object[] args) throws Exception {
Class newoneClass = Class.forName(className);
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Constructor cons = newoneClass.getConstructor(argsClass);
return cons.newInstance(args);
}
public Object newInstance(String className, Object[] args) throws Exception {
Class newoneClass = Class.forName(className);
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Constructor cons = newoneClass.getConstructor(argsClass);
return cons.newInstance(args);
}
[java] view plain copy
- public Object newInstance(String className, Object[] args) throws Exception {
- Class newoneClass = Class.forName(className);
- Class[] argsClass = new Class[args.length];
- for (int i = 0, j = args.length; i < j; i++) {
- argsClass[i] = args[i].getClass();
- }
- Constructor cons = newoneClass.getConstructor(argsClass);
- return cons.newInstance(args);
- }
这里说的方法是执行带参数的构造函数来新建实例的方法。如果不需要参数,可以直接使用newoneClass.newInstance()来实现。
Class newoneClass = Class.forName(className):第一步,得到要构造的实例的Class。
第5~第9行:得到参数的Class数组。
Constructor cons = newoneClass.getConstructor(argsClass):得到构造子。
cons.newInstance(args):新建实例。
6. 判断是否为某个类的实例
public boolean isInstance(Object obj, Class cls) {
return cls.isInstance(obj);
}
public boolean isInstance(Object obj, Class cls) {
return cls.isInstance(obj);
}
[java] view plain copy
- public boolean isInstance(Object obj, Class cls) {
- return cls.isInstance(obj);
- }
7. 得到数组中的某个元素
public Object getByArray(Object array, int index) {
return Array.get(array,index);
}
java.lang.Class.getDeclaredConstructor() 方法返回一个Constructor对象,它反映此Class对象所表示的类或接口指定的构造函数。parameterTypesparameter是确定构造函数的形参类型,在Class对象声明顺序的数组。
声明
以下是java.lang.Class.getDeclaredConstructor()方法的声明
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) throws NoSuchMethodException, SecurityException
Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) throws NoSuchMethodException, SecurityException
参数
-
parameterTypes -- 这是参数数组。
返回值
此方法返回具有指定参数列表构造函数的构造函数对象。
异常
-
NoSuchMethodException -- 如果没有找到匹配的方法。
-
SecurityException --如果安全管理存在。
例子
下面的例子显示java.lang.Class.getDeclaredConstructor()方法的使用。
package com.yiibai;
import java.lang.reflect.*;
public class ClassDemo {
public static void main(String[] args) {
try {
ClassDemo cls = new ClassDemo();
Class c = cls.getClass();
// constructor with arguments as Double and Long
Class[] cArg = new Class[2];
cArg[0] = Double.class;
cArg[1] = Long.class;
Constructor ct = c.getDeclaredConstructor(cArg);
System.out.println("Constructor = " + ct.toString());
}
catch(NoSuchMethodException e) {
System.out.println(e.toString());
}
catch(SecurityException e) {
System.out.println(e.toString());
}
}
private ClassDemo() {
System.out.println("no argument constructor");
}
public ClassDemo(Double d, Long l) {
this.d = d;
this.l = l;
}
Double d = new Double(3.9d);
Long l = new Long(7687);
}
com.yiibai;
import java.lang.reflect.*;
public class ClassDemo {
public static void main(String[] args) {
try {
ClassDemo cls = new ClassDemo();
Class c = cls.getClass();
// constructor with arguments as Double and Long
Class[] cArg = new Class[2];
cArg[0] = Double.class;
cArg[1] = Long.class;
Constructor ct = c.getDeclaredConstructor(cArg);
System.out.println("Constructor = " + ct.toString());
}
catch(NoSuchMethodException e) {
System.out.println(e.toString());
}
catch(SecurityException e) {
System.out.println(e.toString());
}
}
private ClassDemo() {
System.out.println("no argument constructor");
}
public ClassDemo(Double d, Long l) {
this.d = d;
this.l = l;
}
Double d = new Double(3.9d);
Long l = new Long(7687);
}
让我们来编译和运行上面的程序,这将产生以下结果:
no argument constructor
Constructor = public ClassDemo(java.lang.Double,java.lang.Long)
argument constructor
Constructor = public ClassDemo(java.lang.Double,java.lang.Long)