【Spring】IOC容器注解汇总,你想要的都在这儿了!!

public class CustomCondition implements Condition {

/****

@param context

  • @param metadata

  • @return

*/

@Override

public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {

//判断容器中是否有CustomAspect的组件

return context.getBeanFactory().containsBean(“customAspect”);

}

}

public class MainConfig {

@Bean

public CustomAspect customAspect() {

return new CustomAspect();

}

@Bean

@Conditional(value = CustomCondition.class)

public CustomLog customLog() {

return new CustomLog();

}

}

向IOC 容器添加组件


(1)通过@CompentScan +@Controller @Service @Respository @compent。适用场景: 针对我们自己写的组件可以通过该方式来进行加载到容器中。

(2)通过@Bean的方式来导入组件(适用于导入第三方组件的类)

(3)通过@Import来导入组件 (导入组件的id为全类名路径)

@Configuration

@Import(value = {Person.class})

public class MainConfig {

}

通过@Import 的ImportSeletor类实现组件的导入 (导入组件的id为全类名路径)

public class CustomImportSelector implements ImportSelector {

@Override

public String[] selectImports(AnnotationMetadata importingClassMetadata) {

return new String[]{“com.binghe.spring”};

}

}

Configuration

@Import(value = {Person.class}

public class MainConfig {

}

通过@Import的 ImportBeanDefinitionRegister导入组件 (可以指定bean的名称)

public class DogBeanDefinitionRegister implements ImportBeanDefinitionRegistrar {

@Override

public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {

//创建一个bean定义对象

RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(Dog.class);

//把bean定义对象导入到容器中

registry.registerBeanDefinition(“dog”,rootBeanDefinition);

}

}

@Configuration

@Import(value = {Person.class, Car.class, CustomImportSelector.class, DogBeanDefinitionRegister.class})

public class MainConfig {

}

通过实现FacotryBean接口来实现注册 组件

public class CarFactoryBean implements FactoryBean {

@Override

public Car getObject() throws Exception {

return new Car();

}

@Override

public Class<?> getObjectType() {

return Car.class;

}

@Override

public boolean isSingleton() {

return true;

}

}

Bean的初始化与销毁


指定bean的初始化方法和bean的销毁方法

由容器管理Bean的生命周期,我们可以通过自己指定bean的初始化方法和bean的销毁方法

@Configuration

public class MainConfig {

//指定了bean的生命周期的初始化方法和销毁方法.@Bean(initMethod = “init”,destroyMethod = “destroy”)

public Car car() {

return new Car();

}

}

针对单实例bean的话,容器启动的时候,bean的对象就创建了,而且容器销毁的时候,也会调用Bean的销毁方法

针对多实例bean的话,容器启动的时候,bean是不会被创建的而是在获取bean的时候被创建,而且bean的销毁不受IOC容器的管理

通过 InitializingBean和DisposableBean实现

通过 InitializingBean和DisposableBean个接口实现bean的初始化以及销毁方法

@Component

public class Person implements InitializingBean,DisposableBean {

public Person() {

System.out.println(“Person的构造方法”);

}

@Override

public void destroy() throws Exception {

System.out.println("DisposableBean的destroy()方法 ");

}

@Override

public void afterPropertiesSet() throws Exception {

System.out.println(“InitializingBean的 afterPropertiesSet方法”);

}

}

通过JSR250规范

通过JSR250规范 提供的注解@PostConstruct 和@ProDestory标注的方法

@Component

public class Book {

public Book() {

System.out.println(“book 的构造方法”);

}

@PostConstruct

public void init() {

System.out.println(“book 的PostConstruct标志的方法”);

}

@PreDestroy

public void destory() {

System.out.println(“book 的PreDestory标注的方法”);

}

}

通过BeanPostProcessor实现

通过Spring的BeanPostProcessor的 bean的后置处理器会拦截所有bean创建过程

  • postProcessBeforeInitialization 在init方法之前调用

  • postProcessAfterInitialization 在init方法之后调用

@Component

public class CustomBeanPostProcessor implements BeanPostProcessor {

@Override

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

System.out.println(“CustomBeanPostProcessor…postProcessBeforeInitialization:”+beanName);

return bean;

}

@Override

public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {

System.out.println(“CustomBeanPostProcessor…postProcessAfterInitialization:”+beanName);

return bean;

}

}

BeanPostProcessor的执行时机

populateBean(beanName, mbd, instanceWrapper)

initializeBean{

applyBeanPostProcessorsBeforeInitialization()

invokeInitMethods{

isInitializingBean.afterPropertiesSet()

自定义的init方法

}

applyBeanPostProcessorsAfterInitialization()方法

}

通过@Value +@PropertySource来给组件赋值

public class Person {

//通过普通的方式

@Value(“独孤”)

private String firstName;

//spel方式来赋值

@Value(“#{28-8}”)

private Integer age;

通过读取外部配置文件的值

@Value(“${person.lastName}”)

private String lastName;

}

@Configuration

@PropertySource(value = {“classpath:person.properties”}) //指定外部文件的位置

public class MainConfig {

@Bean

public Person person() {

return new Person();

}

}

自动装配


@AutoWired的使用

自动注入

@Repository

public class CustomDao {

}

@Service

public class CustomService {

@Autowired

private CustomDao customDao;

综上:

(1)自动装配首先时按照类型进行装配,若在IOC容器中发现了多个相同类型的组件,那么就按照 属性名称来进行装配

@Autowired

private CustomDao customDao;

比如,我容器中有二个CustomDao类型的组件 一个叫CustomDao 一个叫CustomDao2那么我们通过@AutoWired 来修饰的属性名称时CustomDao,那么拿就加载容器的CustomDao组件,若属性名称为tulignDao2 那么他就加载的时CustomDao2组件

(2)假设我们需要指定特定的组件来进行装配,我们可以通过使用@Qualifier(“CustomDao”)来指定装配的组件

或者在配置类上的@Bean加上@Primary注解

@Autowired

@Qualifier(“CustomDao”)

private CustomDao customDao2

(3)假设我们容器中即没有CustomDao 和CustomDao2,那么在装配的时候就会抛出异常

No qualifying bean of type ‘com.binghhe.spring.dao.CustomDao’ available

若我们想不抛异常 ,我们需要指定 required为false的时候可以了

@Autowired(required = false)

@Qualifier(“customDao”)

private CustomDao CustomDao2;

(4)@Resource(JSR250规范)

功能和@AutoWired的功能差不多一样,但是不支持@Primary 和@Qualifier的支持

(5)@InJect(JSR330规范)

需要导入jar包依赖,功能和支持@Primary功能 ,但是没有Require=false的功能

javax.inject

javax.inject

1

(6)使用@Autowired 可以标注在方法上

  • 标注在set方法上

//@Autowired

public void setCustomLog(CustomLog customLog) {

this.customLog = customLog;

}

  • 标注在构造方法上

@Autowired

public CustomAspect(CustomLog customLog) {

this.customLog = customLog;

}

标注在配置类上的入参中(可以不写)

@Bean

public CustomAspect CustomAspect(@Autowired CustomLog customLog) {

CustomAspect customAspect = new CustomAspect(customLog);

return ustomAspect;

}

XXXAwarce接口


我们自己的组件 需要使用spring ioc的底层组件的时候,比如 ApplicationContext等我们可以通过实现XXXAware接口来实现

@Component

public class CustomCompent implements ApplicationContextAware,BeanNameAware {

private ApplicationContext applicationContext;

@Override

public void setBeanName(String name) {

System.out.println(“current bean name is :【”+name+“】”);

}

@Override

public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {

this.applicationContext = applicationContext;

}

}

@Profile注解


通过@Profile注解 来根据环境来激活标识不同的Bean

  • @Profile标识在类上,那么只有当前环境匹配,整个配置类才会生效

  • @Profile标识在Bean上 ,那么只有当前环境的Bean才会被激活

  • 没有标志为@Profile的bean 不管在什么环境都可以被激活

@Configuration

@PropertySource(value = {“classpath:ds.properties”})

public class MainConfig implements EmbeddedValueResolverAware {

@Value(“${ds.username}”)

private String userName;

@Value(“${ds.password}”)

private String password;

private String jdbcUrl;

private String classDriver;

@Override

public void setEmbeddedValueResolver(StringValueResolver resolver) {

this.jdbcUrl = resolver.resolveStringValue(“${ds.jdbcUrl}”);

this.classDriver = resolver.resolveStringValue(“${ds.classDriver}”);

}

@Bean

@Profile(value = “test”)

public DataSource testDs() {

return buliderDataSource(new DruidDataSource());

}

@Bean

@Profile(value = “dev”)

public DataSource devDs() {

最后

无论是哪家公司,都很重视基础,大厂更加重视技术的深度和广度,面试是一个双向选择的过程,不要抱着畏惧的心态去面试,不利于自己的发挥。同时看中的应该不止薪资,还要看你是不是真的喜欢这家公司,是不是能真的得到锻炼。

针对以上面试技术点,我在这里也做一些分享,希望能更好的帮助到大家。

e String password;

private String jdbcUrl;

private String classDriver;

@Override

public void setEmbeddedValueResolver(StringValueResolver resolver) {

this.jdbcUrl = resolver.resolveStringValue(“${ds.jdbcUrl}”);

this.classDriver = resolver.resolveStringValue(“${ds.classDriver}”);

}

@Bean

@Profile(value = “test”)

public DataSource testDs() {

return buliderDataSource(new DruidDataSource());

}

@Bean

@Profile(value = “dev”)

public DataSource devDs() {

最后

无论是哪家公司,都很重视基础,大厂更加重视技术的深度和广度,面试是一个双向选择的过程,不要抱着畏惧的心态去面试,不利于自己的发挥。同时看中的应该不止薪资,还要看你是不是真的喜欢这家公司,是不是能真的得到锻炼。

针对以上面试技术点,我在这里也做一些分享,希望能更好的帮助到大家。

[外链图片转存中…(img-o5pC8xLn-1720084098682)]

[外链图片转存中…(img-YiWAN0F0-1720084098683)]

[外链图片转存中…(img-qkMpTSub-1720084098684)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值