错误: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(对象,参数列表)
- 要获取一个方法,就是要获取类的信息,获取类的信息就是要获取类类型
- 方法的反射操作, 是指用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