随机名言

Spring注解驱动



作为复习把,来一次总结,方便以后SpringBoot的翻看


1.导入组件

@Component:普通组件

@Service:service层

@Repository:dao层

@Controller:controller层(默认id为类名的小写驼峰)

@Import:快速导入外部组件(默认id为全限定类名)

@ImportResource(locations = {"classpath:application.xml"}):将以前的xml配置文件读取并应用

FactoryBean:实现接口,用@Bean导入容器,导入的是工厂产生的类


2.java配置类相关注解

@Configuration:声明为配置类

@Bean:返回值注册进容器

@ComponentScan:扫描注册组件

@WishlyConfiguration:@Configuration与@ComponentScan的组合注解


3.@Bean的属性支持

@Bean(initMethod = "",destroyMethod = "")

@Scope 设置Spring容器如何新建Bean实例(方法上,得有@Bean)

@Lazy:针对单例实现懒加载

@PostConstruct:由JSR-250提供,bean的initMethod

@PreDestory:由JSR-250提供,等价于bean的destroyMethod


4.注入bean的注解

@Autowired:由Spring提供,按类型装入

@Qualifier:指定组件id,而不是按照类型注入

@Primary:指定自动装配的首选

@Inject:由JSR-330提供

@Resource:由JSR-250提供,与@Autowired区别于按byName装入


5.@PropertySource和@Value注解

@Value("Howl")
String name;

@Value("#{systemProperties['os.name']}")
String osName;

@Value("#{ T(java.lang.Math).random() * 100 }") 
String randomNumber;

@Value("#{domeClass.name}")
String name;

@Value("classpath:com/hgs/hello/test.txt")
String Resource file;

@Value("http://www.cznovel.com")
Resource url;

// 使用外部配置文件,前提把配置文件加载到环境变量中
Value("${book.name}")
String bookName;

@PropertySource 放在配置类上:读取properties外部配置文件K/V保存到运行的环境变量中,用${}取出
@PropertySource("classpath:com/hgs/hello/test/test.propertie")


6.切面(AOP)相关注解

@Aspect:声明一个切面类

@Before:前置通知

@After:后置通知

@AfterReturning:返回通知

@AfterThrowing:异常通知

@Around:在方法执行之前与之后执行

@PointCut:声明切点 在java配置类中使用@EnableAspectJAutoProxy注解开启Spring对AspectJ代理的支持(类上)


7.环境切换

@Profile:通过设定配置环境

@Conditional:实现Condition接口,从而决定该bean是否被实例化


8.异步相关

@EnableAsync:配置类中,通过此注解开启对异步任务的支持

@Async:在实际执行的bean方法使用该注解来申明其是一个异步任务(方法上或类上所有的方法都将异步,需要@EnableAsync开启异步任务)


9.定时任务相关

@EnableScheduling:在配置类上使用,开启计划任务的支持

@Scheduled:来申明这是一个任务,包括cron,fixDelay,fixRate等类型,使用cron表达式:每10秒执行


10.@Enable*注解说明

这些注解主要用来开启对xxx的支持。

@EnableAspectJAutoProxy:开启对AspectJ自动代理的支持

@EnableAsync:开启异步方法的支持

@EnableScheduling:开启计划任务的支持

@EnableWebMvc:开启Web MVC的配置支持

@EnableConfigurationProperties:开启对@ConfigurationProperties注解配置Bean的支持

@EnableJpaRepositories:开启对SpringData JPA Repository的支持

@EnableTransactionManagement:开启注解式事务的支持

@EnableCaching:开启注解式的缓存支持


11.测试相关注解

@RunWith:运行器,Spring中通常用于对JUnit的支持
@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration:用来加载配置ApplicationContext,其中classes属性用来加载配置类
@ContextConfiguration(classes={TestConfig.class})


12.补充

@EventListener:添加监听器,当然要注册进容器


13.SpringMVC部分

@EnableWebMvc:在配置类中开启Web MVC的配置支持

@Controller:声明该类为SpringMVC中的Controller

@ResponseBody:响应返回字符串

@RestController:该注解为一个组合注解,相当于@Controller和@ResponseBody的组合

@RequestMapping:用于映射Web请求,包括访问路径和参数(类或方法上)
@GetMapping
@PostMapping

@RequestBody:请求发送json数据时,而参数在请求体中,而不是在url。那么该注解补充get方法只获取url参数的缺点,调用setter方法映射进对象中

@PathVariable:用于接收路径参数,resful风格

@CookieValue:放方法参数中,将映射到参数上

@RequestParam:绑定请求中name属性相同的变量(使用了反射)

@RestControllerAdvice
@ControllerAdvice:该注解将对于控制器的全局配置放置在同一个位置。注解了@Controller的类的方法可使用@ExceptionHandler、@InitBinder、@ModelAttribute注解到方法上, 这对所有注解了@RequestMapping的控制器内的方法有效。

结合下面的注解可实现:

@ExceptionHandler:用于全局处理控制器里的异常

@InitBinder:用来设置WebDataBinder,WebDataBinder用来自动绑定前台请求参数到Model中。

@ModelAttribute:本来的作用是绑定键值对到Model里,在@ControllerAdvice中是让全局的@RequestMapping都能获得在此处设置的键值对。


14.springboot注解

@SpringBootApplication

@EnableAutoConfiguration

@SpringBootConfiguration

@EnableConfigurationProperties:开启下面的功能

@ConfigurationProperties(prefix="person"):需要下面的依赖
其是@Value的升级版。注解类上,宽松批量注入属性,不像@Value一个个来书写注解

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-configuration-processor</artifactId>
</dependency>




1. 组件注册

  • 包扫描+组件注解:@Component
  • @Bean:导入第三方包里面的组件
  • @Import:快速给容器导入一个组件
  • FactoryBean:实现接口注入容器

1.1 配置类上的基本注解

@Configuration:声明当前类为配置类

@Bean:声明方法的返回值为一个bean,id默认是方法名

@ComponentScan:用于对Component进行扫描,里面有包含与排除规则

@Configuration
@ComponentScan(value = {"com.howl.springannotation.controller","com.howl.springannotation.service"},
        excludeFilters = {
        @ComponentScan.Filter(type = FilterType.ANNOTATION,classes = Controller.class)
})
public class MainConfig {

    @Bean(value = "person01")
    public Person person(){
        return new Person("Howl",20);
    }
}

1.2 @Bean的属性

@Scope:设置Bean实例的作用域,默认单例(IOC容器启动会调用方法创建对象)

@Lazy:针对单实例实现懒加载

@Configuration
public class MainConfig2 {

    @Bean
    @Scope(value = "singleton")
    @Lazy
    public Person person(){
        System.out.println("person was created");
        return new Person("Howl",20);
    }
}


1.3 @Conditional

满足条件才注册bean,其接收Condition类的数组。而Condition是个接口,需要我们去实现。

@Conditional可在方法上,也可在类上

public class MyCondition implements Condition {

    /**
     * @param context:判断条件能使用的上下文环境
     * @param metadata:标注了@Conditional的注释信息
     * 参数是使用该类的地方帮我们传进去的
     * @return
     */
    @Override
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
//        context.getClassLoader();
//        context.getBeanFactory();
//        context.getClass();
//        context.getRegistry();
//        String os = context.getEnvironment().getProperty("os.name");
//		  return os.contains("Windows");
        return false;
    }
}
@Configuration
public class MainConfig {

    @Bean
    @Conditional({MyCondition.class})
    public Person person01(){
        return new Person("Howl",20);
    }
}


1.4 Import、ImportSelector、Registrar、FactoryBean

  • @Import(value = {Person.class,User.class}):直接写类,id默认全类名
  • @Import(value = {MyImportSelector.class}):写ImportSelector实现类,id默认全类名
  • @Import(value = {MyImportBeanDefinitionRegistrar.class}):手动注册bean
  • FactoryBean:实现该接口,用@Bean注解导入该类,那么就会将其内部的类也加入容器。获取工厂可加前缀&
public class MyImportSelector implements ImportSelector {

    /**
     * @param importingClassMetadata:可获取注解类的所有注解信息
     * @return:返回值就是要导入的组件
     */
    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        return new String[]{"com.howl.springannotation.bean.Student","com.howl.springannotation.bean.Worker"};
    }
}
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {

    /**
     * @param importingClassMetadata:当前类的注解信息
     * @param registry:BeanDefinition的注册类
     */
    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        boolean teacher = registry.containsBeanDefinition("Teacher");
        if (!teacher) {
            RootBeanDefinition root = new RootBeanDefinition(Teacher.class);
            registry.registerBeanDefinition("Teacher", root);
        }
    }
}
public class PersonFactoryBean implements FactoryBean<Person> {

    // 返回一个对象会添加到容器中
    @Override
    public Person getObject() throws Exception {
        return new Person();
    }

    @Override
    public Class<?> getObjectType() {
        return Person.class;
    }

    @Override
    public boolean isSingleton() {
        returnjava true;
    }
}
@Configuration
@Import(value = {User.class, MyImportSelector.class, MyImportBeanDefinitionRegistrar.class})
public class MainConfig4 {

    @Bean
    public PersonFactoryBean personFactoryBean() {
        return new PersonFactoryBean();
    }
}


2. 属性赋值

  • @Value:
    • 基本数值
    • SpEL #{}
    • 配置文件 ${}
public class Person {

    @Value("Howl")
    private String name;
    @Value("${person.age}")
    private Integer age;

    // 。。。。
}
@PropertySource(value = "classpath:person.properties")
@Configuration
public class MainConfig5 {

    @Bean
    public Person person(){
        return new Person();
    }

}


3. 自动装配

  • @Autowired:用AutowiredAnnotationBeanPostProcssor完成自动注入的
    • 放属性上
    • 放setter(方法)上:调用方法完成赋值,方法的参数从容器中获取
    • 放构造器上:默认的组件会调用无参构造器创建对象再进行初始化赋值等操作;若当前类只有一个有参构造器,那么@Autowired可以省略的(前提没有默认,否则首选默认)
    • 配置@Bean,那么方法参数可以自动注入而不用@Autowired
  • XXXAware:自定义组件想要使用Spring容器底层的组件(ApplicationContext,BeanFactory),只需实现接口即可,在创建对象的时候,会调用接口规定的方法。其实也是setter方法参数自动注入
@Autowired
BookService bookService;

@Autowired
public void setBookService(BookService bookService) {
    this.bookService = bookService;
}

@Autowired
public BookController(BookService bookService) {
    this.bookService = bookService;
}

@Bean
public BookController bookController(BookService bookService){
    return new BookController(bookService);
}


4. 环境切换

默认是default标识,但是下面代码三步执行,第二部就是加载配置文件,没有设置环境,所以得手动做这三步

AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(MainConfig7.class)

public AnnotationConfigApplicationContext(Class<?>... componentClasses) {
   this();
   register(componentClasses);		// 加载配置文件
   refresh();
}
AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext();
ac.getEnvironment().setActiveProfiles("dev");
ac.register(MainConfig7.class);
ac.refresh();


5. Bean生命周期

bean创建-----初始化-----销毁

一. 构造

  • 单实例:容器启动时创建(调用构造器)
  • 多实例:每次获取时创建(调用构造器)

二.初始化

  • 对象创建完成,并复赋值好后,调用初始化方法

三.销毁

  • 单实例:容器关闭时调用
  • 多实例:容器不管理这个bean,GC来回收

指定初始化和销毁方法:

  • @Bean(initMethod = "init",destroyMethod = "destory")

让Bean实现接口:

  • InitializingBean
  • DisposableBean

JSR-250提供注解

  • @PostConstruct
  • @PreDestroy

后置处理器:对每一个bean都有效

  • BeanPostProcessor
public class Worker implements InitializingBean, DisposableBean {

    public Worker() {
        System.out.println("worker constructor");
    }

    @PostConstruct
    public void init(){
        System.out.println("worker init");
    }

    @PreDestroy
    public void destory(){
        System.out.println("worker destory");
    }

    @Override
    public void destroy() throws Exception {
        System.out.println("接口的销毁方法");
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        System.out.println("接口的初始化方法");
    }
}
public class MyBeanProcessor implements BeanPostProcessor {
    /**
     * @param bean:创建的实例
     * @param beanName:实例的名字
     * @return :返回需要用到的实例,类似于@Bean
     * @throws BeansException
     */
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println(bean + beanName + "---------内容呢额-----------");
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println(bean + beanName + "-----------内容呢额---------");
        return bean;
    }
}

6. AOP

被切入的方法,切面类,各种通知、原理是动态代理

切面类:

@Aspect
public class LogAspect {

    @AfterReturning(value = "execution(public void com.howl..*(..))", returning = "result")
    public int after(JoinPoint joinPoint, Object result) {
        System.out.println("前置通知" + joinPoint.getSignature().getName() + result);
        return 10;
    }
}

配置类:重点在于开启注解

@EnableAspectJAutoProxy  // 开启切面注解
@Configuration
public class AopConfig {

    @Bean  // 注意Bean上的@Autowired是自动注入,可以不写的
    public LogAspect logAspect(JoinPoint joinPoint){
        return new LogAspect();
    }

    @Bean
    public TestService testService(){
        return new TestService();
    }
}


7. 声明式事务

// 1. 导入相关依赖
// 2. 注册配置数据源
// 3. 给需要事务的方法或类标注@Transactional
// 4. 开启基于注解的事务管理功能 @EnableTransactionManagement
// 5. 注册Spring平台的事务管理器来控制事务,重点他要管理数据源,才能管理每一条连接,才能管理事务
@EnableTransactionManagement
@Configuration
public class DbConfig {

    @Bean
    public DataSource dataSource(){
        DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setUsername("root");
        dataSource.setPassword("root");
        dataSource.setDriverClassName("com.mysql.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://localhost:3306/test");
        return dataSource;
    }

    // 可以放参数上,那样会自动注入Bean对象
    // 或调用方法,配置类中的方法调用会被认为是依赖注入
    @Bean
    public PlatformTransactionManager transactionManager(DataSource dataSource){
        return new DataSourceTransactionManager(dataSource());
    }
}


8. 监听器

用在方法上,当然这个类要加入进容器才行

public class test {

    @EventListener(classes = {ApplicationEvent.class})
    public void evenListener(ApplicationEvent event){
        System.out.println("注解下的监听器");
    }
}


9. @ConfigurationProperties的使用

  • 编写properties文件
  • 编写properties类,类上加注解来宽松匹配
  • 配置类上开启宽松匹配功能,并导入properties文件进环境
student.name=howl
student.age=20
@ConfigurationProperties(prefix = "student")
// @Component
public class StudentProperties {

    private String name;
    private int age;

	// 利用setter方法注入
}
@EnableConfigurationProperties
@PropertySource("classpath:application.properties")
@Configuration
public class MainConfig10 {

    @Bean
    public Student student(){
        return new Student();
    }
}


posted @ 2020-06-02 21:53  Howlet  阅读(340)  评论(0编辑  收藏  举报

Copyright © By Howl