练习(模拟Java内省的功能)

本文介绍如何使用Java反射机制实现从Map数据到特定JavaBean实例的转换,并提供了完整的代码示例及测试过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

★ 准备工作

定义一个Model类,里面所有的属性都是private的,然后为每个属性提供getter和setter方法;
再准备一个Map,map的key值都是类里面的属性字段的字符串表示,值任意。 

★ 真正的工作 

设计一个方法Object getModel(Map map,Class cls),传入一个包含所有值的Map,然后再传入Model类的class,

那么返回Model类的实例,这个实例里面已经包含好了所有相关的数据。

也就是把Map中的数据通过反射,设置回到Model类实例中。 

1)核心代码---模仿spring

package reflect2.spring;

import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Map;

public class SpringDaoUtil {
	/**
	 * 给定Map<K, V> map与Class<C> cls,
	 * 返回一个值对象。
	 * 目的:模仿spring的工作原理
	 */
	public static <V, C> C getModel(Map<String, V> map, Class<C> cls)
			throws InstantiationException, IllegalAccessException,
			SecurityException, NoSuchMethodException, IllegalArgumentException,
			InvocationTargetException {
		C instan = cls.newInstance();
		Field[] flds = cls.getDeclaredFields();
		for (Field f : flds) {
			String key = f.getName();
			V v = map.get(key);
			if (v == null) {
				System.out.println("字段 " + key + "  为空!");
			} else {
				// 定义形参
				String methodName = "set" + key.substring(0, 1).toUpperCase()
						+ key.substring(1);
				@SuppressWarnings("rawtypes")
				Class[] parameterTypes = new Class[] { f.getType() };
				Method m = cls.getDeclaredMethod(methodName, parameterTypes);
				// 传递实参
				Object[] objs = new Object[] { v };
				m.invoke(instan, objs);
			}
		}
		return instan;
	}
}


2)测试代码


测试图片

package reflect2.spring;

import java.util.HashMap;
import java.util.Map;

import reflect2.spring.vo.Person;
import reflect2.spring.vo.UserModel;

public class Client {
	public static void main(String[] args) {
		try {
			Map<String, Object> map = new HashMap<String, Object>();
			map.put("uuid", "001");
			map.put("name", "Lucy");
			map.put("type", Integer.valueOf(3));
			map.put("pwd", "222");
			UserModel user = SpringDaoUtil.getModel(map, UserModel.class);
			System.out.println(user);
			System.out.println("-------测试事件分割线----------");

			Map<String, Object> map2 = new HashMap<String, Object>();
			map2.put("name", "Lily");
			Person p = SpringDaoUtil.getModel(map2, Person.class);
			System.out.println(p);

		} catch (Exception e) {
			e.printStackTrace();
		}
	}
}



3)测试数据(值对象)-----JavaBean

package reflect2.spring.vo;

public class Person {
	private String name;
	public int age;
	public Person(String name, int age) {
		this.name = name;
		this.age = age;
	}
	public Person(){
	}
	
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
	@Override
	public String toString() {
		return "Person [name=" + name + ", age=" + age + "]";
	}
}

/**
 * 
 */
package reflect2.spring.vo;

public class UserModel{
	private String uuid;
	private String name;
	private int type;
	private String pwd;
	
	public UserModel(String uuid, String name, int type, String pwd) {
		this.uuid = uuid;
		this.name = name;
		this.type = type;
		this.pwd = pwd;
	}

	public UserModel(String uuid, int type){
		this.uuid = uuid;
		this.type = type;
	}
	public UserModel(){
	}
	private UserModel(String uuid){
		this.uuid = uuid;
	}
	public String getUuid() {
		return uuid;
	}

	public void setUuid(String uuid) {
		this.uuid = uuid;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getType() {
		return type;
	}

	public void setType(int type) {
		this.type = type;
	}

	public String getPwd() {
		return pwd;
	}

	public void setPwd(String pwd) {
		this.pwd = pwd;
	}

	public String toString() {
		return "{"+uuid+","+name+","+type+","+pwd+"}";
	}
	
}











------


### Java内省机制简介 Java内省是一种用于分析类的属性和方法的技术,主要应用于JavaBeans标准的对象。通过内省,程序能够动态获取对象的信息,例如其属性名称、类型以及对应的getter和setter方法。这种技术广泛应用于框架开发中,尤其是在需要处理大量未知类型的对象时。 #### IntrospectionException 类的作用 当执行内省操作时,可能会遇到`IntrospectionException`异常。该异常通常发生在尝试访问非法或不存在的属性或方法时[^1]。因此,在实际应用中需要注意捕获此类异常以防止程序崩溃。 #### 基本用法概述 Chuck McManis在其文章中提到过如何利用反射来推断Java类内部结构的方法[^2]。下面详细介绍一些常见的场景和技术细节: - **PropertyDescriptor**: 这是一个非常重要的工具类,允许开发者描述特定字段及其关联存取器(accessor methods)。可以通过指定字段名创建实例,并从中提取所需信息。 ```java import java.beans.PropertyDescriptor; public class Example { public static void main(String[] args) throws Exception{ PropertyDescriptor pd = new PropertyDescriptor("propertyName", Class.forName("com.example.MyClass")); System.out.println(pd.getReadMethod()); // 输出 getter 方法 System.out.println(pd.getWriteMethod()); // 输出 setter 方法 } } ``` - **BeanInfo Interface**: 如果想提供更精确控制,则可以定义自己的 `BeanInfo` 实现类覆盖默认行为。这使得应用程序可以根据特殊需求调整元数据展示方式。 - **EventSetDescriptors 和 MethodDescriptors**: 它们分别用来列举事件源支持哪些监听器接口以及公开的所有公共成员函数列表。 上述功能都依赖于核心库中的 `java.beans` 包完成[^3]。 ### OAuth2 Token 自省扩展案例 除了基础用途外,还有高级应用场景比如构建安全认证体系的一部分——即实现OAuth2协议下的Token验证服务。具体做法涉及编写专门负责解析请求参数并调用远程API确认凭证状态的服务组件[^4]。 ### 关闭GraphQL查询能力的安全措施 值得注意的是,在某些情况下为了保护敏感数据免受未经授权访问威胁,可能还需要考虑禁用整个系统的自检特性。这方面有一个开源项目提供了简单有效的解决方案可供参考[^5]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值