Spring BeanWrapper分析

最近在读DispatcherServlet 源代码,看到父级类org.springframework.web.servlet.HttpServletBean中关于BeanWrapper的一段代码, 继续追看下去,发现

BeanWrapper 是spring 底层核心的JavaBean包装接口, 默认实现类BeanWrapperImpl.所有bean的属性设置都是通过它来实现。

  1. @Override
  2. public final void init() throws ServletException {
  3. if (logger.isDebugEnabled()) {
  4. logger.debug("Initializing servlet '" + getServletName() + "'");
  5. }
  6. // Set bean properties from init parameters.
  7. try {
  8. PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
  9. BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
  10. ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
  11. bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
  12. initBeanWrapper(bw);
  13. bw.setPropertyValues(pvs, true);
  14. }
  15. catch (BeansException ex) {
  16. logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);
  17. throw ex;
  18. }
  19. // Let subclasses do whatever initialization they like.
  20. initServletBean();
  21. if (logger.isDebugEnabled()) {
  22. logger.debug("Servlet '" + getServletName() + "' configured successfully");
  23. }

org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory类 自动注入工厂抽象类

  1. @Override
  2. public Object configureBean(Object existingBean, String beanName) throws BeansException {
  3. markBeanAsCreated(beanName);
  4. BeanDefinition mbd = getMergedBeanDefinition(beanName);
  5. RootBeanDefinition bd = null;
  6. if (mbd instanceof RootBeanDefinition) {
  7. RootBeanDefinition rbd = (RootBeanDefinition) mbd;
  8. bd = (rbd.isPrototype() ? rbd : rbd.cloneBeanDefinition());
  9. }
  10. if (!mbd.isPrototype()) {
  11. if (bd == null) {
  12. bd = new RootBeanDefinition(mbd);
  13. }
  14. bd.setScope(BeanDefinition.SCOPE_PROTOTYPE);
  15. bd.allowCaching = false;
  16. }
  17. <span style="color:#FF0000;"> BeanWrapper bw = new BeanWrapperImpl(existingBean);</span>
  18. initBeanWrapper(bw);
  19. populateBean(beanName, bd, bw);
  20. return initializeBean(beanName, existingBean, bd);
  21. }

BeanWrapperImpl 继承了属性编辑注册功能

如何设置值 :

  1. @Override
  2. public void setPropertyValue(String propertyName, Object value) throws BeansException {
  3. BeanWrapperImpl nestedBw;
  4. try {
  5. //获取嵌套的属性, like map[my.key], 没有嵌套属性就返回自己
  6. nestedBw = getBeanWrapperForPropertyPath(propertyName);
  7. }
  8. catch (NotReadablePropertyException ex) {
  9. throw new NotWritablePropertyException(getRootClass(), this.nestedPath + propertyName,
  10. "Nested property in path '" + propertyName + "' does not exist", ex);
  11. }
  12. PropertyTokenHolder tokens = getPropertyNameTokens(getFinalPath(nestedBw, propertyName));
  13. nestedBw.setPropertyValue(tokens, new PropertyValue(propertyName, value));
  14. }

看下面具体方法的实现

  1. /**
  2. * Recursively navigate to return a BeanWrapper for the nested property path.
  3. * @param propertyPath property property path, which may be nested
  4. * @return a BeanWrapper for the target bean
  5. */
  6. protected BeanWrapperImpl getBeanWrapperForPropertyPath(String propertyPath) {
  7. int pos = PropertyAccessorUtils.getFirstNestedPropertySeparatorIndex(propertyPath);
  8. // Handle nested properties recursively.
  9. if (pos > -1) {
  10. String nestedProperty = propertyPath.substring(0, pos);
  11. String nestedPath = propertyPath.substring(pos + 1);
  12. //递归获取最后一个属性
  13. BeanWrapperImpl nestedBw = getNestedBeanWrapper(nestedProperty);
  14. return nestedBw.getBeanWrapperForPropertyPath(nestedPath);
  15. }
  16. else {
  17. return this;
  18. }
  19. }

自己实现了个小例子:

  1. public class HelloWorld {
  2. private String msg = null;
  3. private Date date = null;
  4. public String getMsg() {
  5. return msg;
  6. }
  7. public void setMsg(String msg) {
  8. this.msg = msg;
  9. }
  10. public Date getDate() {
  11. return date;
  12. }
  13. public void setDate(Date date) {
  14. this.date = date;
  15. }
  16. }
  17. package com.sunkey.test;
  18. public class Pepole {
  19. private String name;
  20. private int sex;
  21. private HelloWorld helloWorld;
  22. public String getName() {
  23. return name;
  24. }
  25. public void setName(String name) {
  26. this.name = name;
  27. }
  28. public int getSex() {
  29. return sex;
  30. }
  31. public void setSex(int sex) {
  32. this.sex = sex;
  33. }
  34. public HelloWorld getHelloWorld() {
  35. return helloWorld;
  36. }
  37. public void setHelloWorld(HelloWorld helloWorld) {
  38. this.helloWorld = helloWorld;
  39. }
  40. }

测试代码:

    1. @Test
    2. public void testBeanWapper() throws InstantiationException, IllegalAccessException, ClassNotFoundException {
    3. Object obj = Class.forName("com.sunkey.test.HelloWorld").newInstance();
    4. BeanWrapper bw = new BeanWrapperImpl(obj);
    5. bw.setPropertyValue("msg", "HellowWorld");
    6. bw.setPropertyValue("date", new Date());
    7. System.out.println(bw.getPropertyValue("date") + "\n" + bw.getPropertyValue("msg"));
    8. }
    9. @Test
    10. public void testNestedBeanWapper() throws InstantiationException, IllegalAccessException, ClassNotFoundException {
    11. Object obj = Class.forName("com.sunkey.test.HelloWorld").newInstance();
    12. BeanWrapper bw = new BeanWrapperImpl(obj);
    13. bw.setPropertyValue("msg", "HellowWorld");
    14. bw.setPropertyValue("date", new Date());
    15. Object objP = Class.forName("com.sunkey.test.Pepole").newInstance();
    16. BeanWrapper pbw = new BeanWrapperImpl(objP);
    17. pbw.setPropertyValue("name", "jack");
    18. pbw.setPropertyValue("helloWorld", obj);
    19. System.out.println(pbw.getPropertyValue("name") + "\n" + pbw.getPropertyValue("helloWorld.msg"));
    20. pbw.setPropertyValue("helloWorld.msg", "HellowWorld修改过");
    21. System.out.println(pbw.getPropertyValue("name") + "\n" + pbw.getPropertyValue("helloWorld.msg")); }
上一篇:centos设置中文输入法无效的解决办法


下一篇:使用Ansible实现数据中心自动化运维管理