赵雅智_Java JDK 5.0 内省

标签: java jdk 内省 | 发表时间:2012-09-18 20:35 | 作者:zhaoyazhi2129
出处:http://blog.csdn.net

为什么要学内省?

开发框架时,经常需要使用java对象的属性来封装程序的数据,每次都使用反射技术完成此类操作过于麻烦

所以sun公司开发了一套API,专门用于操作java对象的属性。

什么是Java对象的属性和属性的读写方法?

内省访问JavaBean属性的两种方式:

  1. 通过PropertyDescriptor类操作Bean的属性
  2. 通过Introspector类获得Bean对象的 BeanInfo,然后通过 BeanInfo 来获取属性的描述器( PropertyDescriptor ),通过这个属性描述器就可以获取某个属性对应的 getter/setter 方法,然后通过反射机制来调用这些方法。

内省是 Java 语言对 Bean 类属性的一种缺省处理方法。

例如类 A 中有属性 name, 可以通过 getName,setName 来得到其值或者设置新的值。通过 getName/setName 来访问 name 属性,这是默认的规则。 Java 中提供了一套 API 来访问某个属性的 getter/setter 方法。

一般的做法是通过类 Introspector 来获取某个对象的 BeanInfo 信息,然后通过 BeanInfo 来获取属性的描述器( PropertyDescriptor ),通过这个属性描述器就可以获取某个属性对应的 getter/setter 方法,然后通过反射机制来调用这些方法

package net.csdn.high;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import org.junit.Test;

public class StudentDemo {
	/**
	 * @param args
	 * @throws Exception 
	 */
	public static void main(String[] args) throws Exception {
		// TODO Auto-generated method stub
		StudentDemo sd = new StudentDemo();
		sd.test1();
		sd.test2();
		sd.Test3();
		sd.Test4();

	}
	@Test
	//通过内省获取person bean的所有属性
	public void test1() throws IntrospectionException{
		BeanInfo bi = Introspector.getBeanInfo(Student.class,Object.class);
		PropertyDescriptor[] pds=bi.getPropertyDescriptors();
		for(PropertyDescriptor pd:pds){
			String name=pd.getName();
			System.out.println(name);
		}
		System.out.println("--------");
	}
	
	//通过内省给person的name属性赋值:张三   setName("张三")
	@Test
	public void test2() throws IntrospectionException, IllegalArgumentException, IllegalAccessException, InvocationTargetException{
		
		Student s = new Student();
		
		BeanInfo bi = Introspector.getBeanInfo(Student.class);
		
		PropertyDescriptor[] pds = bi.getPropertyDescriptors();
		
		for(PropertyDescriptor pd:pds){
			String name = pd.getName();
			if(name.equals("name")){
				Method m = pd.getWriteMethod();
				m.invoke(s, "张三");
			}
		}
		System.out.println(s.getName());
		System.out.println("--------");
	}
	
	
	//通过PropertyDescriptor类操作Bean的属性  name属性赋值:张三   setName("李四")
	
	@Test
	public void Test3() throws IntrospectionException, IllegalArgumentException, IllegalAccessException, InvocationTargetException{
		
		Student s = new Student();
		
		PropertyDescriptor pd = new PropertyDescriptor("name", s.getClass());
		
		Method m=pd.getWriteMethod();
		
		m.invoke(s,"李四");
		
		System.out.println(s.getName());
		System.out.println("--------");
		
	}
	
	@Test
	public void Test4() throws IntrospectionException, IllegalArgumentException, IllegalAccessException, InvocationTargetException{
		Student s = new Student();
		s.setName("王五");
		
		PropertyDescriptor pd = new PropertyDescriptor("name",s.getClass());
		Method m = pd.getReadMethod();	
		String str = (String)m.invoke(s, null);
		System.out.println(str);
	}

}



Sun公司的内省API过于繁琐,所以Apache组织结合很多实际开发中的应用场景开发了一套简单、易用的API操作Bean的属性——BeanUtils

Beanutils工具包的常用类:

  • BeanUtils
  • PropertyUtils
  • ConvertUtils.regsiter(Converter convert, Class clazz)
package net.csdn.high;

import java.util.Date;

public class Student {
	private String name;
	private int age;
	private Date birthday;
	
	public Date getBirthday() {
		return birthday;
	}
	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}
	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;
	}
	
}


 

package net.csdn.high;
import java.lang.reflect.InvocationTargetException;
import org.apache.commons.beanutils.BeanUtils;
import org.junit.Test;

public class StudentBeanutilsDemo {
	@Test
	public void test1() throws IllegalAccessException, InvocationTargetException{
		Student s = new Student();
		BeanUtils.setProperty(s,"name","张三");
		System.out.println(s.getName());
		
	}

}


 

当有日期类型时候,如果有空格,程序会出错:

package net.csdn.high;
import java.lang.reflect.InvocationTargetException;
import java.util.Date;

import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.beanutils.ConvertUtils;

import org.apache.commons.beanutils.locale.converters.DateLocaleConverter;
import org.junit.Test;

public class StudentBeanutilsDemo {

	@Test
	public void test3() throws IllegalAccessException, InvocationTargetException{
		
		ConvertUtils.register(new DateLocaleConverter(), Date.class);
		
		Student s = new Student();
		BeanUtils.setProperty(s,"birthday","   ");
		//System.out.println(p.getBirthday());
		
	}
	
	

}


自定义转换器

package net.csdn.high;
import java.lang.reflect.InvocationTargetException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.beanutils.ConversionException;
import org.apache.commons.beanutils.ConvertUtils;
import org.apache.commons.beanutils.Converter;
import org.junit.Test;

public class StudentBeanutilsDemo {
	//beanutils工具对基本数据类型可以自动转换类型
	@Test
	public void test2() throws IllegalAccessException, InvocationTargetException{
		//自定义转换器
		ConvertUtils.register(new Converter(){

			@Override
			public Object convert(Class type, Object value) {
				
				if(value == null){
					return null;
				}
				if(!(value instanceof String)){
					throw new ConversionException("只能转String数据");
				}
				String s=(String)value;
				
				if(s.trim().equals("")){
					return null;
				}
				
				SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
				try {
					Date d=sdf.parse(s);
					return d;
				} catch (ParseException e) {
					// TODO Auto-generated catch block
					throw new ConversionException("转换错误");
				}			
			}		
		}, Date.class);
		
		
		
		String name = "张三";
		String age = "23";
		String birthday = "     ";
		
		Student s = new Student();
		BeanUtils.setProperty(s,"name",name);
		BeanUtils.setProperty(s,"age",age);
		BeanUtils.setProperty(s,"birthday",birthday);
		
		System.out.println(s.getName()+"..."+s.getAge()+"...");
		
		
	}

}


 

作者:zhaoyazhi2129 发表于2012-9-18 20:36:47 原文链接
阅读:0 评论:0 查看评论

相关 [java jdk 内省] 推荐:

JAVA之JDK动态代理

- - Java - 编程语言 - ITeye博客
在Java的java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,通过这个类和接口,可以生成JDK动态代理类或动态代理对象. Proxy提供了用于创建动态代理类和代理对象的静态方法,它也是所有动态代理类的父类,如果在程序中为一个或多个接口动态的生成实现类,就可以使用Proxy来创建动态代理类,.

赵雅智_Java JDK 5.0 内省

- - CSDN博客推荐文章
开发框架时,经常需要使用java对象的属性来封装程序的数据,每次都使用反射技术完成此类操作过于麻烦. 所以sun公司开发了一套API,专门用于操作java对象的属性. 什么是Java对象的属性和属性的读写方法?. 内省访问JavaBean属性的两种方式:. 通过PropertyDescriptor类操作Bean的属性.

[iteye]每个Java开发者都应该知道的5个JDK工具

- - PHP & Java
【编者按】JDK是Java语言的软件开发工具包,没有它就无法编译Java程序. 目前,有许许多多的JDK工具呈现在大家面前,但最常用的莫过于java.exe、javac.exe、jar等. 本文作者Joe拥有多年的Java开发经验,其在博客上分享了一篇文章: 5 JDK Tools Every Java Developer Should Know,笔者对其进行了编译,以下为译文.

深入理解Java:内省(Introspector)

- - 博客园_首页
  内省(Introspector) 是Java 语言对 JavaBean 类属性、事件的一种缺省处理方法.   JavaBean是一种特殊的类,主要用于传递数据信息,这种类中的方法主要用于访问私有的字段,且方法名符合某种命名规则. 如果在两个模块之间传递信息,可以将信息封装进JavaBean中,这种对象称为“值对象”(Value Object),或“VO”.

Sun JDK 1.6内存管理

- 小丑鱼 - 淘宝JAVA中间件团队博客
分为使用篇、调优篇和实现篇三个部分,使用篇为填鸭式,调优篇为pattern式,实现篇为启发式,三个PPT的目标为:. 1.掌握Sun JDK的内存区域的划分;. 2.掌握Sun JDK垃圾收集器的使用方法和触发时机;. 4.掌握一些基本的GC调优的方法;. 5.了解自动内存管理的常见实现方法,以及Sun JDK所做的优化.

JDK自带监控工具

- - ITeye博客
         系统在生产运行过程中最重要的工作莫过于监控与问题的处理,监控是预防问题产生很重要的手段. 在监控过程中可以发现那些模块或进程出现了问题,出现问题后会及时通知问题负责人.         实现监控的手段非常多,有系统级别监控系统,也有监控小工具等等. Java 就已经自带了一些监控工具,可以不借助外部软件的情况下简单、快速查看相应监控信息.

JDK动态代理机制

- - CSDN博客编程语言推荐文章
    代理模式有两种,一种是静态代理,这种方式需要为每一个被代理类写一个代理类,显示比较麻烦. 还一种是动态代理,动态代理实现方式一般有两种,JDK动态代理与CGLIB动态代理,这里说一下对JDK动态代理的理解.     JDK动态代理最核心的就类就是java.lang.reflect.Proxy,可调用Proxy.newInstance(..)生成动态代理.

Spring AOP 代理机制 JDK&CGLIB

- - 开源软件 - ITeye博客
Spring AOP使用JDK动态代理或者CGLIB来为目标对象创建代理. (建议优先使用JDK的动态代理). 如果被代理的目标对象实现了至少一个接口,则会使用JDK动态代理. 所有该目标类型实现的接口都将被代理. 若该目标对象没有实现任何接口,则创建一个CGLIB代理. 如果你希望强制使用CGLIB代理,(例如:希望代理目标对象的所有方法,而不只是实现自接口的方法) 那也可以.

Hadoop 2.7.0 发布,不再支持 JDK 6

- - 开源中国社区最新新闻
Apache Hadoop 2.7.0 发布,包括大量显著改进,值得关注的改进如下:. 此版本不再支持 JDK 6 运行时,仅支持 JDK 7+. 还有一些重要的问题需要通过测试,用于生产环境的用户请等待 2.7.1/2.7.2. 支持 Windows Azure 存储 —— Blob. 自动分享,全局缓存 YARN 本地化资源(测试阶段).

动态代理双剑客--JDK Proxy与CGLIB

- - CSDN博客推荐文章
研究过设计模式的同胞们都知道代理模式可以有两种实现方案:. 1.接口实现(或继承抽象类). 都可以通过Proxy控制对Target的访问. “接口实现”的方式更加灵活,代理类可以代理所有实现了ISubject接口的类;. “继承父类”的方式代理类只能代理它的父类,因为java中只支持单继承. 如果Target有直接接口,那么这两种方式都可以;.