开发基于注解形式的spring
SpringIOC容器的2种形式:
(1)xml配置文件:applicationContext.xml;
存bean:<bean>
取bean:
ApplicationContext context=new ClassPathXmlApplicationContext("applicationContext.xml");
(2)注解:带有@Configuration注解的类(配置类)
存bean:@Bean+方法的返回值
//配置类,相当于applicationContext.xml @Configuration public class MyConfig { @Bean //id=方法名(myStudent) public Student myStudent(){ Student student=new Student(2,"fg",34); return student; } }
取bean:
ApplicationContext context=new AnnotationConfigApplicationContext(MyConfig.class); Student myStudent = (Student) context.getBean("myStudent");
注意:两种形式获取的IOC是独立的
注解形式向IOC容器存放bean详解:
1.必须有@Configuration
2.形式:
2.1 三层组件(Controller、Service、Dao):
(1)将三层组件分别加注解@Controller、@Service、@Repository等价于@Commponent
(2)纳入扫描器
a.xml配置
<context:component-scan base-package="org.ghl.controller"></context:component-scan>
b.注解形式
component-scan只对三层组件负责。
给扫描器指定规则:
过滤类型:FilterType(ANNOTATION, ASSIGNABLE_TYPE, CUSTOM)
ANNOTATION:三层注解类型@Controller、@Service、@Repository等价于@Commponent
排除: @ComponentScan(value = "org.ghl",excludeFilters = {@ComponentScan.Filter(type = FilterType.ANNOTATION ,value = {Service.class,Repository.class})}) 包含:(有默认行为,可以通过useDefaultFilters禁止) @ComponentScan(value = "org.ghl",includeFilters = {@ComponentScan.Filter(type = FilterType.ANNOTATION ,value = {Service.class,Repository.class})},useDefaultFilters = false)
ASSIGNABLE_TYPE:指具体的类。
@ComponentScan(value = "org.ghl",includeFilters = {@ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE ,value = {StudentController.class})},useDefaultFilters = false)
区分:ANNOTATION:Service.class指标有@Service的所有类;
ASSIGNABLE_TYPE:指具体的类。
CUSTOM:自定义:自己定义规则
@ComponentScan(value = "org.ghl",includeFilters = {@ComponentScan.Filter(type = FilterType.CUSTOM ,value = {MyFilter.class})},useDefaultFilters = false)
public class MyFilter implements TypeFilter { @Override public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException { ClassMetadata classMetadata = metadataReader.getClassMetadata(); //拿到扫描器value = "org.ghl"包中的所有标有三层组件注解类的名字 String className = classMetadata.getClassName(); //只过滤出和student相关的三层组件 if (className.contains("Student")){ return true; //表示包含 } return false; //表示排除 } }
2.2 非三层组件(Student.clss/转换器等):
(1)@Bean+方法的返回值,id的默认值为方法名,也可以通过@Bean("stu")修改。
(2)import/FactoryBean
bean的作用域
(@Scope("singleton"))scope="singleton":单例
scope="prototype":原型、多实例。
执行的时机(产生bean的时机):
singleton:容器在初始化时,就创建对象,且只创建一次; 也支持延迟加载:在第一次使用时,创建对象。在config中加入@Lazy。
prototype:容器在初始化时,不创建对象,在每次使用时(每次从容器获取对象时),再创建对象。
条件注解
可以让某一个Bean在某些条件下加入IOC容器。
(1)准备bean;
(2)增加条件bean:给每个bean设置条件,必须实现Condition接口。
(3)根据条件加入IOC容器
回顾给IOC加入Bean的方法:
注解:全部在@Configuration配置中设置:
三层组件:扫描器+三层注解
非三层组件:(1)@Bean+返回值
(2)@import
(3)FactoryBean(工厂Bean)
@import使用:
(1)直接编写到@Import中;
@Import({Apple.class,Banana.class})
(2)自定义ImportSelector接口的实现类,通过selectimports方法实现(方法的返回值就是要纳入IOC容器的Bean)。并告知程序自己编写的实现类。
public class MyImportSelector implements ImportSelector { @Override public String[] selectImports(AnnotationMetadata annotationMetadata) { return new String[]{"org.ghl.entity.Apple","org.ghl.entity.Banana"}; //方法的返回值就是要纳入IOC容器的Bean } }
@Import({MyImportSelector.class})
(3)编写ImporBeanDefinitionRegistrar接口的实现类并重写方法。
public class MyImporBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar { @Override public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry) { //BeanDefinition beanDefinition=new RootBeanDefinition(Orange.class); BeanDefinition beanDefinition=new RootBeanDefinition("org.ghl.entity.Orange"); beanDefinitionRegistry.registerBeanDefinition("myorange",beanDefinition); } }
@Import({MyImporBeanDefinitionRegistrar.class})
FactoryBean(工厂Bean)
1.写实现类和重写方法;
public class MyFactoryBean implements FactoryBean{ @Override public Object getObject() throws Exception { return new Apple(); } @Override public Class<?> getObjectType() { return Apple.class; } @Override public boolean isSingleton() { return true; } }
2.注册到@Bean中
@Bean public FactoryBean<Apple> myFactoryBean(){ return new MyFactoryBean(); }
注意:需要通过&区分获取的对象是哪一个。不加&,获取的是最内部真实的apple,如果加&,获取的是FactoryBean。