spring学习1

1、bean实例化过程:bean实例化的基本流程,会将xml配置的bean信息封装成一个beandefinition对象,,所有的beandefinition会存储到一个beandefinition的map中,spring框架对该map进行遍历,并通过反射的方式创建bean实例对象,创建好的bean对象存放在singletonObjects的map集合中当使用getbean方法时,最终从该map集合中取出bean实例对象

2、spring的后置处理器:是spring开发的对外重要扩展点,支持动态注册beandefinition,动态修改beandefinition

BeanFactoryPostProcessor:在beandefinitionmap填充完毕,bean实例化之前进行调用,比如下面的手动注册bean,需要把自己实现的类配置到spring的xml中管理,可以通过自定义注解实现bean注册

java 复制代码
package com.example.demo10.processor;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.beans.factory.support.RootBeanDefinition;

public class myprocessor implements BeanFactoryPostProcessor {

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        System.out.println("beandifinition填充完毕后执行该方法");
        //注册beanDefinition
        BeanDefinition beanDefinition = new RootBeanDefinition();
        beanDefinition.setBeanClassName("com.example.demo10.domain.Impl.PersonImpl");
        //强转成
        DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) beanFactory;
        defaultListableBeanFactory.registerBeanDefinition("personDao",beanDefinition);
    }
}

@Test
    public void test8(){
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationcontext.xml");
        Object p = applicationContext.getBean(Person.class);
        System.out.println(p);
        applicationContext.close();
    }

BeanPostProcessor:在每个bean实例化之后,填充到单例池singletonObjects之前进行调用

java 复制代码
package com.example.demo10.processor;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;

public class MyBeanProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        //可以在这里给创建的bean设置参数,打印日志等,即对bean的增强
        System.out.println(beanName+"postProcessBeforeInitialization");
        return bean;
    }
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println(beanName+"postProcessBeforeInitialization");
        return bean;
    }
}
java 复制代码
package com.example.demo10.processor;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Date;

public class TimeProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {

        return bean;
    }

    /**
     * 通过动态代理,对现有的bean进行增强
     * 动态代理的实现依赖于 java.lang.reflect.Proxy 类和 InvocationHandler 接口。
     * 通过动态代理,可以在不修改原始类的情况下,为方法添加额外的逻辑,例如日志记录、权限校验等。
     * @param bean the new bean instance
     * @param beanName the name of the bean
     * @return
     * @throws BeansException
     */
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        Object beanProxy = Proxy.newProxyInstance(
                bean.getClass().getClassLoader(),
                bean.getClass().getInterfaces(),
                (proxy, method, args) ->{
                    //输出开始时间
                    System.out.println("方法:"+method.getName()+"-开始时间:"+new Date());
                    //执行目标方法
                    Object res = method.invoke(bean,args);
                    //输出结束时间
                    System.out.println("方法:"+method.getName()+"-结束时间:"+new Date());
                        return null;
                     }
        );

        //返回存入到单例池中
        return beanProxy;
    }
}

3、spring三级缓存解决循环引用问题,假如创建Abean,需要注入Bbean,创建Bbean需要注入Abean,spring会将没有注入参数的bean包装为一个factory并且存储到singletonfactory中,当创建Abean,会去三级缓存中找有没有Bbean,没有则去创建Bbean,Bbean创建时会去三级缓存中找有没有Abean,Abean此时在singletonfactory,会将Abean将singletonfactory移除,放入二级缓存earlysingletonObjects中,此时Bbean可以创建,当Bbean和Abean都被创建后,放入一级缓存singletonObjects中

4、spring常用注解:

java 复制代码
/**
 * @Component 注册类到spring容器,其bean name为userdao
 * @Scope 设置是单例还是多例
 * @PostConstruct:标记初始化方法
 * @PreDestroy:标记销毁方法
 * @Repository:标记dao层的类
 * @Service:标记service层的类
 * @Controller:标记web层的类
 * @Value:标记注入值给属性,通过配置文件读取
 * @Autowired:根据类型注入,先去匹配类型,类型有多个,再去匹配name
 * @Bean:将方法返回的bean注册到spring里,适用于调用返回第三方类
 * @Profile("test")  :标注属于哪个环境,比如当前这个只属于test环境,只有test环境被激活,才能使用该bean
 */
//@Component
@Repository
@Scope("singleton")
public class ToyDaoImpl implements ToyDao {

    @Value("S{toyname}")
    private String toyname;


    public String getToyname() {
        return toyname;
    }

    public ToyDaoImpl(){
        System.out.println("构造方法创建对象...");
    }

    @PostConstruct
    public void init(){
        System.out.println("初始化方法...");
    }

    @PreDestroy
    public void destory(){
        System.out.println("销毁方法...");
    }
}

5、配置类

java 复制代码
package com.example.demo10.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;
import org.springframework.context.annotation.PropertySource;

import javax.sql.DataSource;

/**
 * spring的配置类,用来代替spring的配置文件
 */

@Configuration  //标注当前类是一个配置类,把当前的类放入spring容器管理
@ComponentScan(basePackages = {"com.example.demo10"})  //标注扫描基本包,把该包里的类交给spring管理
@PropertySource("classpath:kv.properties")  //加载配置文件
@Import(DataSource.class)
public class SpringConfig {
}

6、@Component会将扫描到的所有类,通过dosacn注册到beandefinitionmap中,执行bean的生命周期,最终生成实例bean

相关推荐
测试开发Kevin18 小时前
小tip:换行符CRLF 和 LF 的区别以及二者在实际项目中的影响
java·开发语言·python
笨手笨脚の18 小时前
Redis: Thread limit exceeded replacing blocked worker
java·redis·forkjoin·thread limit
Lenyiin18 小时前
Linux 基础IO
java·linux·服务器
经年未远18 小时前
vue3中实现耳机和扬声器切换方案
javascript·学习·vue
Hill_HUIL19 小时前
学习日志22-静态路由
网络·学习
松☆19 小时前
Dart 核心语法精讲:从空安全到流程控制(3)
android·java·开发语言
编码者卢布19 小时前
【App Service】Java应用上传文件功能部署在App Service Windows上报错 413 Payload Too Large
java·开发语言·windows
q行19 小时前
Spring概述(含单例设计模式和工厂设计模式)
java·spring
詩不诉卿20 小时前
zephyr学习之自定义外部module记录
学习
浮游本尊20 小时前
React 18.x 学习计划 - 第十三天:部署与DevOps实践
学习·react.js·状态模式