Spring是我们学习Java不可避开的一个重要的框架知识点,而在spring中核心知识点就是IOC和AOP,而其主的IOC控制反转就是将Bean初始化加载到容器中,但是Bean又是如何加载到容器中的呢,这里我们可以使用spring的注解方式或者通过spring XML配置的方式来进行实现。
Spring注解方式减少了配置文件内容,更加便于我们对于项目的管理,并且使用注解可以大大的提高我们的开发效率!
下面我们介绍一下spring中常用的一些注解。
@component:标注一个普通的spring Bean类。@Repository:标注一个Dao组件类。@service:标注一个业务逻辑组件类。@Controller:标注一个控制器组件类。这些都是注解在平时的开发中使用比较高的一些注解。这些注解实质上属于同一类注解,用法相同,功能相同,区别在于标识的组件类型不同。@component可以代替@Repository、@Service、@Controller,因为这三个注解是被@Component标注的。代码如下
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Controller{
string value() default “”;
}
@Repository
public class HappyDaoImpl implements HappyDao{
private final static Logger LOGGER = LoggerFactory.getLogger(HappyDaoImpl .class);
public void club(){
//do something ,like drinking and singing
}
}
@Service(value="goodClubService")
//使用@Service注解不加value ,默认名称是clubService
public class ClubServiceImpl implements ClubService {
@Autowired
private ClubDao clubDao;
public void doHappy(){
//do some Happy
}
}
@Controller
public class HappyController {
@Autowired //下面进行讲解
private ClubService clubService;
// Control the people entering the Club
// do something
}
/*Controller相关的注解下面进行详细讲解,这里简单引入@Controller*/
备注解的java类当做Bean实例,Bean实例的名称默认是Bean类的首写小字母,其他部分不变。@Service也可以自定义Bean名称,但是必须是唯一的:2,尽量使用对应组件注解的类替换@Component注解,在spring未来的版本中,@Controller,@Service,@Resposity会携带更多语义。并且便于开发和维护!3,指定了某些类可以作为spring Bean类使用后,最好还需要让spring搜索指定路径,在spring配置文件中加入如下配置:
@Autowired:属于spring的org.springframework.beans.factory.annotation包下,可用于为类的属性、构造器、方法进行注值@Resource:不属于spring的注解,而是来自于JSR-250位于java.annotation包下,使用该annotation为目标bean指定协作者Bean。@postConstruct和@preDestory方法实现初始化和销毁bean之前进行的操作。
@Autowired
@Target({ElementType.CONSTRUCTOR, ElementType.FIELD, ElementType.METHOD,
ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Autowired {
boolean required() default true;
}
@Controller
public class HappyController {
@Autowired //默认依赖的ClubDao 对象(Bean)必须存在
//@Autowired(required = false) 改变默认方式
@Qualifier("goodClubService")
private ClubService clubService;
// Control the people entering the Club
// do something
}
@Target({TYPE, FIELD, METHOD})
@Retention(RUNTIME)
public @interface Resource {
String name() default "";
Class type() default java.lang.Object.class;
public class AnotationExp {
@Resource(name = "HappyClient")
private HappyClient happyClient;
@Resource(type = HappyPlayAno .class)
private HappyPlayAno happyPlayAno;
}
相同点:@Resource的作用相当于@Autowired,均可以标注在字段或属性的setter方法上。
不同点:提供方@Autowired是spring的注解,@Resource是javax.annotation注解,而是来自于JSR-250,J2EE提供,需要JDK1.6以上。
注入方式@Autowired只按照Type注入;@Resource默认按Name自动注入,也提供按照Type注入
属性:@Autowired注解可用于为类的属性、构造器、方法进行注值。默认情况下,器依赖的对象必须存在(bean可用),如果需要改变这种默认方式,可用设置其required属性为false。还有一个比较重要的点就是,@Autowired注解默认按照类型装配,如果容器中包含多个同一类型的Bean,那么启动容器时会报找不到指定类型bean的异常,解决办法时结合**@Qualifier**注解进行限定,指定注入的bean名称。
@Resource有两个重要的属性,name和type。name属性指定byName,如果没有指定name属性,当注解标注在字段上,即默认读取字段的名称作为bean名称寻找依赖对象,当注解标注在属性的setter方法上,即默认取属性名作为bean名称寻找依赖对象。需要注意的是,@Resource注解会回退到按类型装配。但一旦指定了name属性,就只能按名称装配了。
@Resource注解的使用性更为灵活,可指定名称,也可以指定类型;@Autowired注解进行装配容易抛出异常,特别是装配的bean类型有多个的时候,而注解的办法时需要增加@Qualifer进行限定。
[Spring中 @Autowired注解与@Resource注解的区别](http://qiangmzsx.blog.51cto.com/2052549/1359952)
注意点:使用@Resource也要注意添加配置文件到spring,如果没有配置component-scan
则一定要配置annotation-config
①:简单介绍
spring的官方团队说@Component可以替代@Configure注解,事实上我们看源码也可以发现看到,如下
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component //看这里!!!
public @interface Configuration {
String value() default "";
虽然说可以替代但是两个注解之间还是有区别的!
Bean注解只要用于方法上,有点类似于工厂干活,当使用@Bean注解,我们可以连续使用多种定义bean是用到的注解,譬如用@Qualifier注解定义工厂方法的名称,用@Scope注解定义该bean的作用域范围,譬如是singleton还是prototype等。
spring中新的Java配置支持的核心就是@Configuration注解的类。这些类主要包括@Bean注解的方法来为spring的IOC容器管理的对象定义实例,配置和初始化逻辑
使用@Configuration来注解类表示可以被spring的IOC容器所使用,作为bean定义的资源。
@Configuration
public class AppConfig {
@Bean
public MyService myService() {
return new MyServiceImpl();
}
}
这和spring的xml文件中的非常类似
@Bean注解扮演了和元素相同的角色。
②:举例说明@Component和@Configuration
@Configuration
public static class Config {
@Bean
public SimpleBean simpleBean() {
return new SimpleBean();
}
@Bean
public SimpleBeanConsumer simpleBeanConsumer() {
return new SimpleBeanConsumer(simpleBean());
}
}
@Component
public static class Config {
@Bean
public SimpleBean simpleBean() {
return new SimpleBean();
}
@Bean
public SimpleBeanConsumer simpleBeanConsumer() {
return new SimpleBeanConsumer(simpleBean());
}
}
第一个代码正常工作,正如预期的那样,SimpleBeanConsumer将会得到一个单例SimpleBean的链接。第二个配置是完全错误的,因为Spring会创建一个SimpleBean的单例bean,但是SimpleBeanConsumer将获得另一个SimpleBean实例(也就是相当于直接调用new SimpleBean() ,这个bean是不归Spring管理的),既new SimpleBean() 实例是Spring上下文控件之外的
使用@configuration,所以标记为@bean的方法将被包装成一个CGLIB包装器,他的工作方式就好像是这个方法的第一个调用,那么原始的主体将被执行,最终的对象在spring上下文中注册。所有进一步的调用只返回从上下文检索的bean。
在上面的第二个代码块中,新的SimpleBeanConsumer(simpleBean)只调用一个纯java方法。为了纠正第二个代码块,我们可以这样做
@Component
public static class Config {
@Autowired
SimpleBean simpleBean;
@Bean
public SimpleBean simpleBean() {
return new SimpleBean();
}
@Bean
public SimpleBeanConsumer simpleBeanConsumer() {
return new SimpleBeanConsumer(simpleBean);
}
}
Spring @Configuration vs @Component
基本概念:@Configuration 和@Bean
@Controller
public class HappyController {
//do something
...
}
spring MVC 使用@Controller定义控制器,他还允许自动检测定义在类路径下的组件(配置文件中配置扫描路径)并自动注册。
@Controller
@RequestMapping("/happy")
public class HappyController {
@Autowired
private HappyService happyService;
@RequestMapping(/hello/*)
public void sayHello(){
//请求为 /happy/hello/* 都会进入这个方法!
//例如:/happy/hello/123 /happy/hello/adb
//可以通过get/post 请求
}
@RequestMapping(value="/haha",method=RequestMethod.GET)
public void sayHaHa(){
//只能通过get请求
}
...
}
@RequestMapping既可以作用在类级别,也可以作用在方法级别。当他定义在类级别时,标明该控制器处理所有的请求都被映射到/favsoft路径下。@RequestMapping中可以使用method属性标记其所接受的方法类型,如果不指定方法类型的话,可以使用HTTP GET/POST方法请求数据,但是一旦指定方法类型,就只能使用该类型获取数据。
@RequestMapping("/happy")
public String sayHappy(
@RequestParam(value = "name", required = false) String name,
@RequestParam(value = "age", required = true) String age) {
//age参数必须传 ,name可传可不传
...
}
@RequestMapping(value="/happy/{dayid}",method=RequestMethod.GET)
public String findPet(@PathVariable String dayid, Model mode) {
//使用@PathVariable注解绑定 {dayid} 到String dayid
}
@PathVariable中的参数可以是任意的简单类型,如int, long, Date等等。Spring会自动将其转换成合适的类型或者抛出 TypeMismatchException异常。当然,我们也可以注册支持额外的数据类型。
@PathVariable支持使用正则表达式,这就决定了它的超强大属性,它能在路径模板中使用占位符,可以设定特定的前缀匹配,后缀匹配等自定义格式。
@RequestMapping(value = "/something", method = RequestMethod.PUT)
public void handle(@RequestBody String body,@RequestBody User user){
//可以绑定自定义的对象类型
}
@ResponseBody : @ResponseBody与@RequestBody类似,它的作用是将返回类型直接输入到HTTP response body中。
@RequestMapping(value = "/happy", method =RequestMethod.POST)
@ResponseBody
public String helloWorld() {
return "Hello World";//返回String类型
}
@RestController :控制器实现了REST的API,只为服务于JSON,XML或其它自定义的类型内容,@RestController用来创建REST类型的控制器,与@Controller类型。@RestController就是这样一种类型,它避免了你重复的写@RequestMapping与@ResponseBody。
@ModelAttribute :@ModelAttribute可以作用在方法或方法参数上,当它作用在方法上时,标明该方法的目的是添加一个或多个模型属性(model attributes)。
该方法支持与@RequestMapping一样的参数类型,但并不能直接映射成请求。控制器中的@ModelAttribute方法会在@RequestMapping方法调用之前而调用。
@ModelAttribute方法有两种风格:一种是添加隐形属性并返回它。另一种是该方法接受一个模型并添加任意数量的模型属性。用户可以根据自己的需要选择对应的风格。
在处理dao层或service层的事务操作时,譬如删除失败时的回滚操作。使用**@Transactional** 作为注解,但是需要在配置文件激活
@Service
public class CompanyServiceImpl implements CompanyService {
@Autowired
private CompanyDAO companyDAO;
@Transactional(propagation = Propagation.REQUIRED, readOnly = false, rollbackFor = Exception.class)
public int deleteByName(String name) {
int result = companyDAO.deleteByName(name);
return result;
}
...
}
事务的传播机制和隔离机制比较重要!
一图学习 Spring事务传播性
readOnly : 事务的读写属性,取true或者false,true为只读、默认为false
rollbackFor : 回滚策略,当遇到指定异常时回滚。譬如上例遇到异常就回滚
timeout (补充的) : 设置超时时间,单位为秒
isolation : 设置事务隔离级别,枚举类型,一共五种
透彻的掌握 Spring 中@transactional 的使用 Spring事务配置及事务的传播性与隔离级别详解
原文链接:https://blog.csdn.net/u010648555/article/details/76299467