反射

本文详细介绍了Java反射机制,包括获取Class对象的四种方式、获取类的基本信息、属性操作、方法操作和构造器操作。同时,通过实例展示了如何使用反射创建对象、访问私有属性和方法,以及如何打破单例模式的限制。反射机制虽然增加了程序的灵活性,但也存在性能损耗和安全风险。
摘要由CSDN通过智能技术生成

1、反射的定义
反向探知,在程序运行中动态的获取或操作类中的属性就是反射。
1.1、初探反射:
 

获取Class对象的四种方式
Class clazz1 = User.class;
 Class<?> clazz2 = Class.forName("com.tledu.pojo.User");
 Class<? extends User> clazz3 = new User().getClass();
 Class<?> clazz4 = UserTest.class.getClassLoader().loadClass("com.tledu.p
ojo.User");
1.2、获取类的基本信息
 System.out.println(clazz1.getPackage());//获取的包名
 System.out.println(Modifier.toString(clazz1.getModifiers()));//获取类的修饰符
 System.out.println(clazz1.getSimpleName());//获取的类名
 System.out.println(clazz1.getClassLoader());//获取的类加载器
 System.out.println(clazz1.getInterfaces().length);//获取的是这个类实现的接口 
 System.out.println(clazz1.getAnnotations().length);//获取类的注解
 System.out.println(clazz1.getName()); //获取全类名

优缺点
优点:
增加程序的灵活性,避免固有逻辑写死到程序中
代码相对简洁,可以提高程序的复用性
缺点:
相比于直接调用反射有比较大的性能销毁
内部暴露和安全隐患

通过反射获取对象为什么性能销毁较大?
1、native //调用底层
2、checkMemberAccess //安全校验

2、反射的属性操作

public static void main(String[] args) throws Exception {
//获取User的Class对象
 Class clazz = User.class;
 //通过反射的newInstance()方法创建对象
 User user = (User) clazz.newInstance();

//getFields()获取本类及父类中的公共的属性
 Field[] fields = clazz.getFields();
 for (Field field : fields) {
 System.out.println(Modifier.toString(field.getModifiers())+" "+field.ge
tName());
 }

 System.out.println("‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐");
 //getDeclaredFields()获取本类中的所有属性包括私有的
 Field[] fields2 = clazz.getDeclaredFields();
 for (Field field : fields2) {
 System.out.println(Modifier.toString(field.getModifiers())+" "+field.ge
tName());
 }
 System.out.println("‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐");
 //
 Field field = clazz.getDeclaredField("name");
 //对于私有的属性在操作时要开放权限
 field.setAccessible(true);
 field.set(user, "钱枫");

 System.out.println(field.get(user));

 //关于静态属性的赋值
 Field field2 = clazz.getDeclaredField("address");
 field2.set(null, "湖南");
 System.out.println(field2.get(null));

 }

3、反射的方法操作

public static void main(String[] args) throws InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException {
		//获得类对象
		Class<User> clazz=User.class;
		//通过反射创建类对象
		User user=clazz.newInstance();
		
		//getMethods()获取本类及父类的公共方法
		Method[] methods=clazz.getMethods();
		for (Method method : methods) {
		//获取 修饰符  并且将修饰符tostring
			System.out.println(Modifier.toString(method.getModifiers())+""+method.getName());
		}
		
		//getDeclaredMethods();获取本类中所有的方法
		Method[] methods1=clazz.getDeclaredMethods();
		for (Method method : methods1) {
		//获取 修饰符  并且将修饰符tostring
			System.out.println(Modifier.toString(method.getModifiers())+""+method.getName());
		}
		System.out.println("-----------------");
		//普通方法的执行调用
		Method method=clazz.getDeclaredMethod("eat");
		method.setAccessible(true);
		
		method.invoke(user);
		System.out.println("-----------------------------------");
		//静态方法的调用
		Method method2=clazz.getDeclaredMethod("say", String.class );
		method2.invoke(null, "上海话");
		
	}

3、反射的构造器操作

public static void main(String[] args) throws Exception {
		//获取类对象
		Class<User> clazz=User.class;
		//创建类
		User user=(User)clazz.newInstance();
		//获取类中的构造器
		Constructor<?>[] constructors=clazz.getDeclaredConstructors();
		for (Constructor<?> constructor : constructors) {
			System.out.println(constructor.getName());
		}
		System.out.println("-----------------");
		//构造器的调用
		Constructor<User>constructor=clazz.getDeclaredConstructor(String.class,String.class);
		User user2=constructor.newInstance("田园","男");
		System.out.println(user2.getName()+" "+user2.getSex());
	}

4、单例破局

//创建一个私有的静态变量
	private static Singleton singleton;
	//私有化构造函数
	private Singleton(){
		if (singleton!=null) {
			throw new  RuntimeException();
		}
	}
	//创建一个公共的方法并在里面new一个对象 让其只能创建一次
	public static Singleton getlnstance(){
		if (singleton==null) {
			singleton=new Singleton();
		}
		return singleton;
	}


public static void main(String[] args) throws Exception {
		//单例:只能初始化一次,也就只能创建一个实例
	Singleton s1=Singleton.getlnstance();
	Singleton s2=Singleton.getlnstance();
	
	System.out.println(s1);
	System.out.println(s2);
	System.out.println("-----------");
	//获取类对象
	Class<Singleton> clazz=Singleton.class;
	//获取类的加载器
	Constructor<Singleton> constructor=clazz.getDeclaredConstructor();
	constructor.setAccessible(true);
	//通过反射的方法创建类对象
	Singleton s3 = constructor.newInstance();
	Singleton s4 = constructor.newInstance();
	System.out.println(s3);
	System.out.println(s4);
	}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值