Grails Service 层

Service 层


除了 Web 层 之外, Grails 还定义了service 层的概念。Grails 团队不赞成在controllers中嵌入核心的应用程序逻辑,因为这样并没有提升重用和清楚的关注点分离。
Grails中的Services在应用程序中被视为放置多数逻辑的地方 。从controllers脱离,负责处理通过重定向的请求流等等。
 
创建Service
你可以在终端窗口的项目根目录下运行 create-service 创建Service:
 
grails create-service simple
上面的示例将在grails-app/services/SimpleService.groovy位置创建一个Service。 service的名字按规约以 Service结尾。除此之外,service就是个普通的Groovy类:
 
class SimpleService {
}
1 声明式事务处理
Services一般涉及协调 domain 类之间的逻辑, , 因此常常涉及大范围的持久化操作。因为services性质,它们常常需要事物状态。你可以使用 withTransaction方法来编程事物,不过,这是重复性的,没有充分利用Spring强大的潜在事物抽象
Services允许启用事物,本质上是以声明的方式来声明service中的所有方法必须用于事物。默认情况下,所有services的事物都是可用的——禁用它,只需设置  transactional属性为 false:
 
class CountryService {
    static transactional = false
}


你也可以默认设置这个属性为 true 在以后改变它,或者清楚的表明这个服务是有意地用于事物。
 
警告: 依赖注入 是 唯一 声明事物工作的方式。你不能使用new操作符,像这样new 
BookService()获取事物服务
其结果是,所有的方法都被包含在事物中,当方法体中抛出异常时,自动回滚。事物的传播级别被默认设置为 PROPAGATION_REQUIRED.
2 服务作用域
默认情况下,存取服务方法是非同步的,所以无法阻止同步执行这些函数。事实上,因为服务是单例的,可以被同时使用,你必须非常小心服务中存储状态。或者采用容易(和更好的)途径并不在y service中存储状态。
你可以通过把service放置于特定的作用域来改变这样的行为:
prototype -一个新的service每次被注入到其他类时创建
request - 一个新的service在每次请求时创建
flash - 一个新的service只在当前或下个请求时创建
flow - 在web flows中, service将存在于flow的作用域
conversation - 在web flows中, service将存在于会话的作用域。根flow和它的子 flows
session - 一个service被创建用于session的作用域
singleton (默认) - 只有一个实例的service,任何时候都存在
 
假如你的service为flash, flow 或  conversation 作用域,它需要实现 java.io.Serializable 并只用于 Web Flow上下文
为了启用一个作用域,在你的类中添加一个静态scope属性,其值为上面所述的作用域之一:
 
static scope = "flow"


 
3 依赖注入与服务
依赖注入基础
Grails服务的一个重要方面是,有能力利用Spring 框架的依赖注入能力。 Grails支持 "依赖注入通过规约". 换句话说,你可以使用一个属性名表示的一个服务的类名,自动把他们注入到 controllers, tag libraries,等等。
作为示例,给定的服务名为BookService, 如果你像下面这样在controller中放置一个名为bookService 的属性:
 
class BookController {
   def bookService
   …
}


在这种情况下,Spring 容器将自动注入一个基于它自己配置作用域的服务实体。所有的依赖注入是通过名字的; Grails 不支持类型注入。你也可以像下面这样指定类型:
 
class AuthorService {
	BookService bookService
}


不过, 存在副作用,即在开发模式下BookService的改变会在加载时抛出一个错误。
 
依赖注入与服务
你可以使用相同的技术在一个服务中注入另一个服务。如果说,你的AuthorService需要一个  BookService, 可以像下面这样声明 AuthorService:
 
class AuthorService {
	def bookService
}
 


依赖注入与Domain类
你甚至可以在domain类中注入服务,这可以帮助开发出各种丰富的domain:
 
class Book {	
	…
	def bookService
	def buyBook() {
		bookService.buyBook(this)
	}
}


4 Using Services from Java
服务的强大在于它包含了可重用的逻辑,你可以使用来自其他类的服务,包括Java类。这里有一些方法让你重用来自Java的服务。简单的方法是把你的服务移动到grails-app/services目录下的一个包里。这是关键步骤,因为你不可能在Java中导入一个默认package 。作为示例, BookService 就是因为上面的原因,在下面Java中不能使用:
 
class BookService {
	void buyBook(Book book) {
		// logic
	}
}


不过, 把这个类放入一个package中便可修复,, 把这个类移动到  grails-app/services/bookstore子目录,然后,修改package 声明:
 
package bookstore
class BookService {
	void buyBook(Book book) {
		// logic
	}
}


package的替代是,定义个需要服务实现的接口:
 
package bookstore;
interface BookStore {
	void buyBook(Book book);
}


然后,服务:
 
class BookService implements bookstore.BookStore {
	void buyBook(Book b) {
		// logic
	}
}


后一种方法更熟悉, 在Java端,只需要接口的引用,而不需要实现类。无论哪种方式,这个练习的目的是,在编译时,让Java能够静态解决类(或接口)的使用。现在,这样便可在  src/java包内创建一个Java类,并提供了一个setter,在Spring中使用bean的类型和它的名字:
 
package bookstore;
// note: this is Java class
public class BookConsumer {
	private BookStore store;
	public void setBookStore(BookStore storeInstance) {
		this.store = storeInstance;
	}	
	…
}


这样一来,你可以在  grails-app/conf/spring/resources.xml中把这个Java当做Spring bean来配置 (更多详情查看 Grails and Spring):
 
<bean id="bookConsumer" class="bookstore.BookConsumer">
	<property name="bookStore" ref="bookService" />
</bean>


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值