@Controller ,标注在类级别,用于标注一个Portlet Handler,控制器类可以是任意的类。
----
@Autowired ,标注在构造器,setter方法,或者字段上,用于让Spring根据类型来注入依赖,所以如果在Spring上下文中如果有某个类的多个bean实例,则会抛出异常,此时我们应该配合@Qualifier 一起使用。
比如我们有
- <bean id="bookservice1" class="foo.bar.service.BookService/>
- <bean id="bookservice2" class="foo.bar.service.BookService"/>
- <bean id="bookservice3" class="foo.bar.service.BookService"/>
我们想让我们的Controller注入bookservice1 ,则必须联合使用@Autowired和@Qualifier:
- @Controller(value="addBookController")
- public class AddBookController {
- @Autowired
- @Qualifier("bookservice1")
- private BookService bookService;
- ...
- }
-----
@Resource,标注在类,字段,setter方法,但是不可以标注在构造器上,用于让Spring根据名字来注入依赖。
- @Controller(value="addBookController")
- public class AddBookController {
- @Resource(name="myBookService")
- private BookService bookService;
- ...
- }
在Portlet Controller的内部,我们还可以用以下的注解:
@RequestMapping,标注在类和方法上,Portlet常用的映射策略是:将一种Portlet模式映射到控制器类上,然后将请求参数映射到各个处理器方法上。
当用于类级时,它可以将对portlet的请求根据当前portlet的mode和请求参数来映射到Handler上,例如:
- @Controller(value="someController")
- @RequestMapping(value="VIEW",
- params="action=showSomething")
- public class SomeController {
- public String showSomething(...) {
- return "addBookForm";
- }
- public void doSomething(...) {
- }
- }
当用于方法级时,它可以将对portlet的请求根据当前portlet的mode和请求参数映射到Handler的某个方法上,例如:
- @Controller(value="someController")
- public class SomeController {
- @RequestMapping(value="VIEW",
- params="action=showSomething")
- public String showSomething(...) {
- }
- @RequestMapping(value="ACTION", params="action=doSomeWork")
- public void doSomething(...) {
- }
- }
注意,类级别的@RequestMapping的模式和方法级别的@RequestMapping的模式必须匹配,否则会产生错误。
也可以根据方法的签名来让Spring自动判断某个方法是render处理方法还是action处理方法,比如:
- @Controller(value="someController")
- @RequestMapping(value="VIEW")
- public class SomeController {
- ...
-
//下面的处理方法是一个render请求的处理方法 - @RequestMapping(params="action=showSomething")
- public String showSomething(Model model) {
- }
- //下面的处理方法是一个action请求的处理方法
- @RequestMapping(params="action=doSomeWork")
- public void doSomething(ActionRequest request) {
- }
- }
action请求的处理方法总是返回void
----
为了让请求类型映射更加清晰,我们不用@RequestMapping,而是用更加具体的标注比如 @RenderMapping ,@ActionMapping:
----
@RequestMapping
@RenderMapping:标注在方法上,用于将一个render类型的portlet 请求映射到处理器的render方法上,其中value也可以不指定,如果不指定则表示渲染到NOMAL+MAXIMIZED+MINIMIZED上。
- @Controller(value="someController")
- @RequestMapping(value="VIEW")
- public class SomeController {
- @RenderMapping(value="NORMAL",
- params="action=showSomething")
- public String showSomething(Model model) {
- }
- }
注意,@RenderMapping必须和@RequestMapping联合使用,因为@RequestMapping可以指定Portlet Mode,而@RenderMapping无法指定Portlet Mode.
-----
@ActionMapping:
@ActionMapping标注在方法上,用于将一个action类型的portlet请求映射到处理器的action方法上,其中value也可以不指定。
- @Controller(value="someController")
- @RequestMapping("VIEW")
- public class SomeController {
- @ActionMapping(value="doSomeWork",
- params={"work=laundry","time=enough"})
- public String doSomething(ActionRequest request) {
- ...
- }
- ...
- }
注意,@ActionMapping必须和@RequestMapping联合使用,因为@RequestMapping可以指定Portlet Mode,而@ActionMapping无法指定Portlet Mode.
----
@RequestParam 是方法参数级的注解,用于绑定请求参数到方法参数
- @RequestMapping
- public void advancedSearch(
- @RequestParam("queryStr") String queryStr,
- @RequestParam("showFlag") String showFlag,
- @RequestParam("totalnumber") String totalNumber,
- @RequestParam("upType") String upType,
- @RequestParam("jmesareq") String jmesaReq,
- @RequestParam("isExportOper") String isExportOper,
- RenderResponse response, final RenderRequest request,
- ModelMap model) {
- // get query structure and query type from page
- List<Long> cd_ids = new ArrayList<Long>();
- if(StringUtils.equals("invoke", jmesaReq)){
- cd_ids = (List<Long>)request.getSession().getAttribute(Constants.RESULT_IDS);
- }
- ....
- }
-----
@ModelAttribute ,标注在方法或者方法参数中,用于从Model对象中添加或者移除属性。
如果用于方法级别,那么方法的返回值被添加到Model对象中,value属性表明存放到Model中的key.例如:
- @Controller(value="bookController")
- @RequestMapping("VIEW")
- public class BookController {
- @Autowired
- @Qualifier("myBookService")
- private BookService bookService;
- ...
- @ModelAttribute(value="books")
- public List<Book> getBooks() {
- return bookService.getBooks(isbnNumber);
- }
- ...
- }
注意,被@ModelAttribute标注的方法的调用时间总是早于被@RenderMapping或者@ActionMapping所标注的方法.
注意,@ModelAttribute标注也必须和@RequestMapping联合使用。
如果用于方法参数级别,那么它用于绑定模型中的属性到方法参数,例如:
- @RequestMapping(value = "VIEW")
- @Controller(value="addBookController")
- public class AddBookController {
- @Autowired
- @Qualifier("myBookService")
- private BookService bookService;
- ...
- @ActionMapping(params = "myaction=addBook")
- public void addBook(@ModelAttribute(value="book")
- Book book, ...) {
- ...
- bookService.addBook(book);
- }
- }
-----
@SessionAttributes ,标注在类级别,用于让模型的属性存储于控制器的会话状态中。直到Session结束(详细说来是调用了SessionStatus的setComplete()方法)。这样做的好处就是可以在一个作用域更大的区域上共享对象,用法是将@SessionAttributes和@ModelAttribute 联合使用,并且他们的value值设成一样。例如:
- @Controller
- @RequestMapping(value="VIEW")
- @SessionAttributes(value="book")
- public class EditBookController {
- @Autowired
- @Qualifier("myBookService")
- private BookService bookService;
- @ModelAttribute("book")
- public Book getBook(@RequestParam Long isbnNumber) {
- return bookService.getBook(isbnNumber);
- }
- @ActionMapping(params="myaction=editBook")
- public void editBook
- (@ModelAttribute("book")Book book,...){
- ...
- bookService.editBook(book);
- ...
- }
- }
- @RenderMapping(params="myaction=editBookForm")
- public String showEditBookForm(){
- return "editBookForm";
- }
- }
对于共享PortletSession上相同类型的多个属性,也可以用如下的写法:
- @SessionAttributes(types=XXXClass.class)
这样所有XXClass类型的属性都会被设置到PortletSession域上。
-----
@InitBinder 标注在方法级,为什么要用这个注解呢?
因为请求类型的参数都是String类型的, 所以要让某个属性编辑器(比如自定义的属性编辑器),格式化器或者校验器来处理某个请求参数,就必须用WebDataBinder进行绑定,而@InitBinder就负责初始化WebDataBinder,比如:
- @Controller
- public class MyFormController {
- @InitBinder
- public void initBinder(WebDataBinder binder) {
- SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
- dateFormat.setLenient(false);
- binder.registerCustomEditor(Date.class, new CustomDateEditor(dateFormat, false));
- }
- …
- }
这个例子就是Date类型绑定到自定义的Date编辑器。