热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

SpringBoot动态注入及操作Bean

写在前面:2020年面试必备的Java后端进阶面试题总结了一份复习指南在Github上,内容详细,图文并茂,有需要学习的朋友可以Star一下! GitHub地址:https://github.co

写在前面:2020年面试必备的Java后端进阶面试题总结了一份复习指南在Github上,内容详细,图文并茂,有需要学习的朋友可以Star一下!
GitHub地址:https://github.com/abel-max/Java-Study-Note/tree/master

在Spring中我们通过getBean来获得对象,但这些对象都是事先定义好的,如果需要操作事先未定义的Bean就需要动态注入、修改和删除Bean

思路
在Spring中,BeanFactory负责管理Bean,具体来说是DefaultListableBeanFactory,我们只需要获取到当前上下文中的BeanFactory,就能执行其中的注册Bean的方法registerBeanDefinition,注册Bean时需要Bean的信息,Spring提供了BeanDefinitionBuilder.genericBeanDefinition()方法来构建BeanDefinition,用这些方法就可以实现Bean的动态注入

代码实现
测试服务类TestService
创建一个测试类来测试Bean的操作,这个类上没有加注解,Spring在加载时不会自动注入

package com.mantis.bean.service;
/**
 * @Description:
 * @author: wei.wang
 * @since: 2020/9/17 12:53
 * @history: 1.2020/9/17 created by wei.wang
 */
public class TestService {
    private String name;
    private String id;
    public String getId() {
        return id;
    }    public void setId(String id) {
        this.id = id;
    }    public String getName() {
        return name;
    }    public void setName(String name) {
        this.name = name;
    }    public void print() {
        System.out.println("获取bean,name=" + name + ",id=" + id);
    }    @Override
    public String toString() {
        return "TestService{" +
                "name='" + name + ''' +
                ", id='" + id + ''' +
                '}';
    }}

SpringContextUtil类
操作Bean的Util类,其中上线文ApplicationContext在SpringBoot启动时设置,也可以直接使用Autowired注入或者根据具体项目具体实现

package com.mantis.bean.util;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
import org.springframework.context.ApplicationContext;
/**
 * @Description:
 * @author: wei.wang
 * @since: 2020/9/17 12:52
 * @history: 1.2020/9/17 created by wei.wang
 */
public class SpringContextUtil {
    private static ApplicationContext applicationContext;
    //获取上下文
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }
    //设置上下文
    public static void setApplicationContext(ApplicationContext applicationContext) {
        SpringContextUtil.applicatiOnContext= applicationContext;
    }
    //通过名字获取上下文中的bean
    public static Object getBean(String name) {
        try {
            return applicationContext.getBean(name);
        } catch (NoSuchBeanDefinitionException ex) {
            return null;
        }
    }
    //通过类型获取上下文中的bean
    public static Object getBean(Class> requiredType) {
        return applicationContext.getBean(requiredType);
    }
}

启动类SpringBootBeanApplication
在启动类上设置上下文

package com.mantis.bean;
import com.mantis.bean.util.SpringContextUtil;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ApplicationContext;
@SpringBootApplication
public class SpringBootBeanApplication {
    public static void main(String[] args) {
        ApplicationContext app = SpringApplication.run(SpringBootBeanApplication.class, args);
        SpringContextUtil.setApplicationContext(app);    }}
注入Bean
向Spring上下文中注入Bean,注入前先检查Bean是否已经存在

/**
     * 注册Bean     *     * @return
     */    @GetMapping("/bean/register/{beanName}")
    public String registerBean(@PathVariable String beanName) {        //获取context        ConfigurableApplicationContext cOnfigurableApplicationContext= (ConfigurableApplicationContext) SpringContextUtil.getApplicationContext();        //获取BeanFactory        DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) configurableApplicationContext.getBeanFactory();        //创建bean信息.        BeanDefinitionBuilder beanDefinitiOnBuilder= BeanDefinitionBuilder.genericBeanDefinition(TestService.class);        beanDefinitionBuilder.addPropertyValue("id", "1");
        beanDefinitionBuilder.addPropertyValue("name", "张三");
        //判断Bean是否已经注册        Object beanObject = SpringContextUtil.getBean(beanName);        if (beanObject != null) {
            System.out.println(String.format("Bean %s 已注册", beanName));
        } else {
            //动态注册bean.            defaultListableBeanFactory.registerBeanDefinition(beanName, beanDefinitionBuilder.getBeanDefinition());            //获取动态注册的bean.            beanObject = SpringContextUtil.getBean(beanName);            if (beanObject != null) {
                System.out.println(String.format("Bean %s 注册成功", beanName));
                return beanObject.toString();
            } else {
                return "register Bean Error";
            }        }        return "register Bean Success";
    }

修改Bean
从SpringContext中获取到指定Bean的Class对象,获取到之后就可以根据字段进行赋值

/**
     * 动态修改Bean     *     * @return
     */    @GetMapping("/bean/update/{beanName}")
    public String update(@PathVariable String beanName) {        ApplicationContext applicatiOnContext= SpringContextUtil.getApplicationContext();        String[] beans = applicationContext.getBeanDefinitionNames();        for (String bean : beans) {
            // 拿到bean的Class对象            Class> beanType = applicationContext.getType(bean);            if (beanType == null) {
                continue;            }            // 拿到当前bean类型的所有字段            Field[] declaredFields = beanType.getDeclaredFields();            if (beanName.equals(bean)) {
                for (Field field : declaredFields) {
                    // 从spring容器中拿到这个具体的bean对象                    Object beanObject = applicationContext.getBean(bean);                    // 当前字段设置新的值                    try {                        String fieldName = field.getName();                        if ("name".equals(fieldName)) {
                            setFieldData(field, beanObject, "AL113A5");
                        } else if ("id".equals(fieldName)) {
                            setFieldData(field, beanObject, "12");
                        }                    } catch (Exception e) {                        e.printStackTrace();                    }                }            }        }        return "update Bean Success";
    }    private void setFieldData(Field field, Object bean, String data) throws Exception {        field.setAccessible(true);
        Class> type = field.getType();
        if (type.equals(String.class)) {
            field.set(bean, data);        } else if (type.equals(Integer.class)) {
            field.set(bean, Integer.valueOf(data));        } else if (type.equals(Long.class)) {
            field.set(bean, Long.valueOf(data));        } else if (type.equals(Double.class)) {
            field.set(bean, Double.valueOf(data));        } else if (type.equals(Short.class)) {
            field.set(bean, Short.valueOf(data));        } else if (type.equals(Byte.class)) {
            field.set(bean, Byte.valueOf(data));        } else if (type.equals(Boolean.class)) {
            field.set(bean, Boolean.valueOf(data));        } else if (type.equals(Date.class)) {
            field.set(bean, new Date(Long.parseLong(data)));        }    }

移除Bean
将Bean从Spring上下文中移除

/**
     * 移除Bean
     *
     * @return
     */
    @GetMapping("/bean/remove/{beanName}")
    public String removeBeanDefinition(@PathVariable String beanName) {
        Object beanObject = SpringContextUtil.getBean(beanName);
        if (beanObject == null) {
            System.out.println(String.format("Bean %s 不存在", beanName));
            return "remove Error";
        }        //获取context.
        ConfigurableApplicationContext cOnfigurableApplicationContext= (ConfigurableApplicationContext) SpringContextUtil.getApplicationContext();
        //获取BeanFactory
        DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) configurableApplicationContext.getBeanFactory();
        defaultListableBeanFactory.removeBeanDefinition(beanName);
        System.out.println(String.format("Bean %s 已移除", beanName));
        return "remove Success";
    }

测试
写一个操作Bean的方法

/**
     * 操作Bean
     *
     * @return
     */
    @GetMapping("/bean/print/{beanName}")
    public String print(@PathVariable String beanName) {
        Object beanObject = SpringContextUtil.getBean(beanName);
        if (beanObject != null) {
            ((TestService) beanObject).print();            return beanObject.toString();
        } else {
            System.out.println(String.format("Bean %s 不存在", beanName));
        }        return String.format("Bean %s 不存在", beanName);
    }

注册
调用方法

localhost:9000/bean/register/TestService

结果

执行操作Bean的方法,返回结果如下

TestService{name=’张三’, id=’1′}

修改
调用方法

localhost:9000/bean/update/TestService

结果

执行操作Bean的方法,返回结果如下

TestService{name=’AL113A5′, id=’12’}

移除
调用方法

localhost:9000/bean/remove/TestService

结果

执行操作Bean的方法,返回结果如下

Bean TestService 不存在

来源:https://www.tuicool.com/articles/JzaIraN


推荐阅读
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 标题: ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • SpringBoot uri统一权限管理的实现方法及步骤详解
    本文详细介绍了SpringBoot中实现uri统一权限管理的方法,包括表结构定义、自动统计URI并自动删除脏数据、程序启动加载等步骤。通过该方法可以提高系统的安全性,实现对系统任意接口的权限拦截验证。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
author-avatar
手机用户2502901265_642
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有