5.13反射学习

错误:Class path contains multiple SLF4J bindings.

jar包冲突导致
解决办法:参考https://blog.csdn.net/mn_kw/article/details/80579678,在pom.xml里把相关依赖删了

反射学习参考:https://www.imooc.com/learn/199

普通数据类型java静态成员不是对象。类是对象,任何一个类都是java.lang.Class类的实例对象
类的实例对象有三种表示方法:(c1,c2,c3)

package fanshe;

public class Fe {
	public static void main(String[] args) {
		Foo foo1 = new Foo();
		Class c1 = Foo.class; // 任何一个类都有一个隐含的静态成员变量class
		Class c2 = foo1.getClass(); // 已知该类对象foo1,通过getClass方法
		// c1,c2表示Foo类的类类型=Foo类本身是Class类的实例(Foo类本身也有实例对象foo1)
		// 简而言之Foo类的对象是foo1,Foo类的类类型是c1,c2,且c1=c2
		Class c3 = null;
		System.out.println(c1==c2);
		try {
			c3 = Class.forName("fanshe.Fe");
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		//可以通过c1 or c2 or c3创造Foo的实例
		try {
			Foo foo=(Foo)c1.newInstance();//需要无参数的构造方法
			foo.print();
		} catch (InstantiationException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		
		}
	}
}

class Foo {
	void print(){
		System.out.println("hehe");
		
	}
}

new创建对象,是静态加载类,在编译时刻就要加载所有的可能使用到的类:假如有100个功能,只要有一个功能有问题,其他的都用不了。
运行时刻加载类是动态加载类,可处理该问题。Class c = Class.forName(“args[0]”); args[0]表示的接收的第一个参数
参考学习:https://www.imooc.com/video/3733

基本的数据类型,void关键字都存在类类型

Class类的API操作

获取方法信息,成员变量信息和构造函数信息

/**
 * @author pc
 * @Time 2019年5月13日 下午3:57:05
 * @Version 1.0
 * 
 */
package fanshe;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class ClassUtil {
	/**打印类的信息,包括类的成员函数,成员变量(成员函数),构造函数
	 * obj该对象所属类的信息
	*/
	public static void printMethod(Object obj) {
		//获取类的类类型信息
		Class c=obj.getClass();//传递的是那个子类的对象,c就是该子类的类类型
		System.out.println("类的名称是:"+c.getName());
		/*Method类,方法对象
		 * 一个成员方法就是一个Method对象
		 * getMethods()方法获取的是所有的public的函数,包括父类继承而来的
		 * getDeclaredMethods()获取的是该类所有的自己声明方法,不问访问权限
		 */
		Method[] ms=c.getMethods();
		Method[] ms1=c.getDeclaredMethods();
		for(int i=0;i<ms1.length;i++){
			//得到方法的返回值类型的类类型
			Class returnType=ms1[i].getReturnType();
			System.out.print(returnType.getName()+" ");
			//得到方法的名称
			System.out.print(ms1[i].getName()+"(");
			//获取参数列表的类型的类类型
			Class[] p=ms1[i].getParameterTypes();
			for(Class class1:p){
				System.out.print(class1.getName()+",");
			}
		System.out.println(")");
		}
		
	}
	/**
	 * 获取成员变量的信息*/
	public static void printField(Object obj) {
		Class c=obj.getClass();
		/*
		 * 成员变量也是对象
		 * java.lang.reflact.Field
		 * Field类封装了关于成员变量的操作
		 *getFields()方法获取的是所有的public的成员变量的信息
		 *getDeclaredFields()获取的是该类所有的自己声明成员变量的信息,不问访问权限*/
		Field[] fs=c.getDeclaredFields();
		for(Field field:fs){
			//得到成员变量的类型的类类型
			Class fieldType=field.getType();
			String typeName=fieldType.getName();
			System.out.print(typeName+" ");
			//得到成员变量的名称
			String fieldName=field.getName();
			System.out.print(fieldName+" ");
			
		}
	}
	/**
	 * 打印构造函数的信息*/
public static void printCon(Object obj){
	Class c=obj.getClass();
	/*
	 * 构造函数也是对象
	 * java.lang.Constructor封装 了构造函数的信息
	 * getConstructors获得了所有的public的构造函数
	 * getDeclaredConstructors得到所有的构造函数
	*/
	Constructor[] cs=c.getDeclaredConstructors();
	for (Constructor constructor : cs) {
		System.out.print(constructor.getName()+"(");
		//获取构造函数的参数列表,得到参数列表的类类型
		Class[] param=constructor.getParameterTypes();
		for (Class class1 : param) {
			System.out.print(class1.getName()+",");
		}
		System.out.println(")");
	}
}
}

通过该类调用


package fanshe;

public class Fe3 {
	
	public static void main(String[] args) {
		String s="hello";
		ClassUtil.printMethod(s);
		System.out.println("==============");
		Integer n1=1;
		ClassUtil.printField(n1);
		System.out.println("==============");
		ClassUtil.printCon(s);
		
	}
}

方法的反射

方法的名称+列表 唯一决定某一方法

方法反射的操作:method.invoke(对象,参数列表)

  1. 要获取一个方法,就是要获取类的信息,获取类的信息就是要获取类类型
  2. 方法的反射操作, 是指用m对象来进行方法调用
package fanshe;

import java.lang.reflect.Method;

public class Method1 {

	public static void main(String[] args) {
		A a1=new A();
		Class c=a1.getClass();
		     //获取方法名称和参数列表
			/*方法的反射操作
		     * 是指用m对象来进行方法调用
		     * 方法如果没有返回值返回null
			*/
		try {
			Method m=c.getMethod("print",int.class,int.class);
		
		    
			Object o=m.invoke(a1,10,20);
			
			System.out.println("====================");
			Method m1=c.getMethod("print",String.class,String.class);
			o=m1.invoke(a1,"hell0"," world");
			System.out.println("====================");
			Method m2=c.getMethod("print");
			o=m2.invoke(a1);
		} catch (Exception e) {
			
			e.printStackTrace();
		} 		
	}
}
class A{
	public void print(int a,int b){
		System.out.println(a+b);
	
	}
	public void print(){
		System.out.println("ahhhh");
	
	}
	public void print(String a,String b){
		System.out.println(a.toUpperCase()+","+b.toLowerCase());
		
	}
		
}


通过Class,Method了解泛型本质

反射的操作都是编译之后的操作,编译之后集合的泛型就去泛型化的。集合的泛型是防止错误输入的,只在编译阶段有效。绕过编译就无效了
参考:https://www.imooc.com/video/3738

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值