tapestry的URL形如/examples/app?service=page/Admin
能够保证有效运行的一个非常重要的原因是,用有状态的javabean代替无状态的servlet构建一个tapestry应用。page是有状态的,他只能在一个线程里为一个用户处理一个request,而一个servelet,没有用户的状态,可以在并发线程中为任何数量同时发生的request提供服务。使用有状态的page遇到的问题和使用数据库的连接遇到的问题非常相似。
engine
engine是每个tapestry应用的中心,它是一个负责支持和组织应用所有方面的对象,他把所有小的子系统绑在一起构成一个tapestry应用,它首先负责管理server端的状态,管理Visit对象及持久页面属性,他会被保存在session中。engine对象的service()方法,负责进来的request处理和把响应返回给客户端。 ApplicationServlet调用public boolean service(RequestContext context)方法figure 7.4,执行request处理,此服务不仅要进行很多的初始化工作,更重要的是它包括多级的异常捕捉、报告,任何未捕捉的异常会由异常页来呈现。过程如下:
1).调用AbstractEnginer的protected void setupForRequest(RequestContext context)方法,确保engine对象被设置,这个方法很重要,细节可参看API文档,在覆写类方法的子类中,必须首先第一句调用这个方法
2).调用自己的getService(String name)
3).new RequestCycle(IEngine engine, RequestContext requestContext, IEngineService service, IMonitor monitor)
4).调用IEngineService的service(IEngineServiceView engine, IRequestCycle cycle, ResponseOutputStream output)方法
5).调用RequestCycle对象的cleanup()
6).调用自己的cleanupAfterRequest(IRequestCycle cycle)方法
Engine service
Engine service是实现了IEngineService接口的对象,他包含很多创建和服务应用URL的方法,且全是在一个对象中,Engine service更象servlet,他能被很多线程共享,不能记录客户状态。tapeestry默认是有9个service,4个最长用的是home, page,direct和external,可见Table 7.3,大部分service有相对应的部件
IEngineService的service(IEngineServiceView engine, IRequestCycle cycle, ResponseOutputStream output)方法,IRequestCycle的一些方法调用等,各个service的调用是不相同,这些步之后各个service都要回调IEngine对象的renderResponse()方法,处理也是一样的。
home Service:
1).调用IRequestCycle的getPage(String name),返回home page
2).调用IRequestCycle的activate(IPage page)方法,此方法为request设置最终返回客户端显示的活动页面,活动页面典型的由service设置,但因为可能被替换要显示的页面,也会经常被validator方法pageValidate(PageEvent event)改变,这个方法的操作过程如下:它调用page对象的validate(IRequestCycle cycle)方法,(validate()方法用于基本的安全验证,这个方法实际上并不执行任何检查Figure7.06,page对象可以有多个PageValidateListener,The validate()方法调用每个validator对象的pageValidate()。最通常的方法是page对象自己实现PageValidateListener接口,会自动注册成为自己的validator。validator可以通过throw a PageRedirectException激活不同的页面,当PageRedirectException异常被扔出,由service处理的request过程将被中止,被异常指定的页面被激活并被立马呈现给客户端。)
3).service对象回调engine的renderResponse()方法,将使活动页面被呈现并响应给客户端
page service:
除了service对象和home service不一样,其他步骤相同
direct service:
DirectLink和Form部件使用这个服务,这两个部件都实现IDirect接口,当处理form的submit时,会首先执行一个rewind动作,之后执行form指定的listener动作。direct service能够检查session是否过期,DirectLink和Form部件的“stateful”属性,默认为“false”,设置为“true”就可以进行session检查,当呈现响应时,direct service会生成URL,形如:/examples/app?service=direct/1/Guess/select,URL中的1,就标明这需要检查session是否过期,
一旦session过期,用户就会看到Session已经过期的页面,默认的是个很简陋的,可以创建一个命名为“StaleSession”的page,来给用户提供一个更友好的界面。处理请求,DirectService的service()方法处理过程:
1).调用IRequestCycle的getPage(String name),返回page
2).调用IRequestCycle的activate(IPage page)方法,此方法为request设置最终返回客户端显示的活动页面,活动页面典型的由service设置,但因为可能被替换要显示的页面,也会经常被validator方法pageValidate(PageEvent event)改变,这个方法的操作过程如下:它调用page对象的validate(IRequestCycle cycle)方法,(validate()方法用于基本的安全验证,这个方法实际上并不执行任何检查,page对象可以有多个PageValidateListener,The validate()方法调用每个validator对象的pageValidate()。最通常的方法是page对象自己实现PageValidateListener接口,会自动注册成为自己的validator。validator可以通过throw a PageRedirectException激活不同的页面,当PageRedirectException异常被扔出,由service处理的request过程将被中止,被异常指定的页面被激活并被立马呈现给客户端。)
3).调用IPage的getNestedComponent(String path),返回一个IDirect对象
4).调用IDirect对象的isStateful(),如果为true,session过期检查将要发生,检查HttpSession过期,StaleSessionException异常将被服务扔出
5).调用IRequestCycle的setServiceParameters(Object[] parameters),由service调用,service参数被解开并存入request cycle的serviceParameters属性中
6).调用IDirect对象的trigger(IRequestCycle cycle),调用部件的listener方法,执行相应的action
7).service对象回调engine的renderResponse()方法,将使活动页面被呈现并响应给客户端
需要注意的是,各方法调用顺序很重要,validate()发生的比较早,那时还不能访问service参数,session检查发生在validate()之后
对于DirectLink部件,trigger()方法内部处理过程:
1).调用IActionListenerr的actionTriggered(IComponent component,IRequestCycle cycle)方法
1.1).通过反射机制调用listener方法
对于Form部件,trigger()方法执行,要执行rewind,内部过程Figure 7.12:
1).调用IRequestCycle的rewindForm(IForm form,String targetActionId)方法
1.1).调用page对象的beginPageRender(),触发适当的事件
1.2).回调IForm对象的rewind(IMarkupWriter writer,IRequestCycle cycle)
1.2.1).调用IForm对象的render(IMarkupWriter writer, IRequestCycle cycle)
1.2.2).调用监听方法等
1.3).调用page对象的endPageRender(),触发适当的事件
external Service:
1).调用IRequestCycle的getPage(String name),返回page
2).调用IRequestCycle的setServiceParameters(Object[] parameters),由service调用
3).调用IRequestCycle的activate(IPage page)方法,此方法为request设置最终返回客户端显示的活动页面,活动页面典型的由service设置,但因为可能被替换要显示的页面,也会经常被validator方法pageValidate(PageEvent event)改变,这个方法的操作过程如下:它调用page对象的validate(IRequestCycle cycle)方法,(validate()方法用于基本的安全验证,这个方法实际上并不执行任何检查,page对象可以有多个PageValidateListener,The validate()方法调用每个validator对象的pageValidate()。最通常的方法是page对象自己实现PageValidateListener接口,会自动注册成为自己的validator。validator可以通过throw a PageRedirectException激活不同的页面,当PageRedirectException异常被扔出,由service处理的request过程将被中止,被异常指定的页面被激活并被立马呈现给客户端。)
3).调用IExternalPage的activateExternalPage(Object[] parameters, IRequestCycle cycle)
4).service对象回调engine的renderResponse()方法,将使活动页面被呈现并响应给客户端
从池中获得一个页面
虽然采用池化技术,因为page众多,我认为,在最开始池中是没有page的,只有第一次访问生成一个完整page,直接返回给request使用,使用完毕page返回池中等待复用。
Figure 7.19 IRequestCycle的getPage(String name)方法返回一个page实例,这个实例被request cycle对象在整个request期间缓存,将来调用同名page名字的getPage()会返回相同实例,getPage()方法内部步骤是,
1).用IPageSource的getPage(IRequestCycle cycle, String pageName, IMonitor monitor)方法,IPageSource就是page池,他能实例化一个新的page实例,如果池中没有可用的page实例,page实例调用自己的attach(IEngine value)以把自己绑定到一个具体的engineh上,一直到request cycle结束,才解除绑定返回池中。
2).调用page的setRequestCycle(IRequestCycle cycle)
3).调用IEngine的getPageRecorder(String pageName,IRequestCycle cycle),IPageRecorder是一个对象,负责追踪page的持久页面属性变化的,当持久页面属性改变了,新值就会被记录在HttpSession中。page持久状态是特定于一个专门用户,完全和page实例分开的。IPageRecorder通过简单的通知机制被钩入page实例,IPageRecorder观察到持久属性改变,就会他作为一个命名的session属性把持久属性值安全的保存到HttpSession,各自的持久页面属性被作为独自的HttpSession属性保存
4).调用IPageRecorder的rollback(IPage page),将页面持久属性恢复到HttpSession属性保存的值
创建一个新Page实例
当请求page,而池中又没有实例可用,page source会利用PageLoader(是类不是IPageLoader接口)来创建一个page实例figure 7.20:
PageLoader不是threadsafe,PageSource要创建一个新的PageLoader实例为每一个要装载的页面,为解决多线程的问题
PageLoader的IPage loadPage(String name,
INamespace namespace,
IRequestCycle cycle,
IComponentSpecification specification)过程:
1).new一个page实例,java page class被实例化
2).初始化属性,page的初始属性被设置,包括page名字
3).page实例调用自己的attach(IEngine value)以把自己绑定到一个具体的engineh上
4).page包含的部件被递归创建,每一个部件被创建,page loader就会调用部件的finishLoad()方法,从BaseComponent类继承的部件在这时也会装载他的模板
5).page的finishLoad()方法被调用
一旦page的finishLoad()方法执行完毕,一个初始化和配置过的完整的page对象被返回给request cycle。对于页面和部件来说,当在page or component specification中对有的初始化不能被表达时,重载后的finishLoad()方法是个进行这种最终初始化的好地方,经常这类初始化要涉及page中的其他部件。finishLoad()有两个方法,public void finishLoad(
IRequestCycle cycle,
IPageLoader loader,
IComponentSpecification specification);
protected void finishLoad();
尽量重载protected的无参的finishLoad(),且无需先调用父类的方法,除非要使用到3个参数,才会重载public的,且当重载public的带3个参数的方法时,必须先调用父类的public的同名方法,一旦调用失败,就会造成加载page或component的模板失败,也会造成其他方面的影响
把Page实例返回池
在request结束时,response被发送回client之后,附着在request上的page必须被返回池,页面属性包括持久的临时的都必须被重置回初始值,以供其他用户的request使用。如果保留属性值不重置是非常危险的,因为所有的page实例是完全共享的,另一个request完全有可能从池中获得上一个用户使用过的page,所以信息就会暴露。IRequestCycle的cleanup()方法Figure 7.21,会释放所有他拥有的资源,就包括释放page回page source,过程如下:
1).调用IPageSource的releasePage(IPage page)方法
1.1).IPage的detach()方法
1.1.1).清除changeObserved属性,Clears the changeObserved property
1.1.2).调用所有注册的相关监听器的PageDetachListener.pageDetached(PageEvent)
1.1.3).调用org.apache.tapestry.AbstractPage.initialize()清除重置所有属性
1.1.4).Clears the engine, visit and requestCycle properties,the page's visit,engine,and requestCycle properties are reset to null.
子类可以覆写这个方法,但是必须在子类方法中调用父类的同名方法,通常是在方法的最后(以上detach()说明主要参见AbstractPage.detach()方法API文档)
实际上只要你用<property-specification>元素来声明持久或临时属性,就不必关心页面的清除工作, detach()或initialize()方法主要是早期版本中要顾及的。
页面呈现过程:
tapestry呈现的核心是IRender接口,只有一个方法:
public void render(IMarkupWriter writer, IRequestCycle cycle);
这个接口被希望参与页面呈现处理的所有对象所实现,他是IComponent的父接口,因此所有的部件都可以被呈现。IMarkupWriter接口一个很重要的工作,把所有xml保留字自动转换,如><等,不用程序来干预。BasePage类实现了 getResponseWriter(),源码为:
public IMarkupWriter getResponseWriter(OutputStream out) {
return new HTMLWriter(out, getOutputEncoding());
}
HTMLWriter的getContentType()方法默认会返回一个字符串“text/html; charset=UTF-8”,因此若要返回xml类型,应该在BasePage的子类中覆写getResponseWriter()方法,如下:
public IMarkupWriter getResponseWriter(OutputStream out) {
return new HTMLWriter("text/xml",getOutputEncoding(),out);
}
若要返回wml类型,应该在BasePage的子类中覆写getResponseWriter()方法,如下:
public IMarkupWriter getResponseWriter(OutputStream out) {
return new WMLWriter(out, getOutputEncoding());
}
engine对象的renderResponse()方法:
1).调用IRequestCycle对象的getPage(),得到page对象
2).调用page对象的public IMarkupWriter getResponseWriter(OutputStream out)方法,返回一个IMarkupWriter对象
3).调用IMarkupWriter对象的getContentType(),这个值用来设置HttpServletResponse的setContentType()方法
4).调用IRequestCycle对象的renderPage(IMarkupWriter writer),呈现指定的页面,应用应该总是用这个方法来呈现页面,而不是直接调用IRender.render(IMarkupWriter, IRequestCycle),因为在呈现之前cycle对象必须进行一些设置;
4.1).IRequestCycle对象的renderPage()调用page对象renderPage(IMarkupWriter writer,IRequestCycle cycle),被调用来呈现完整页面,这个方法应该只由IRequestCycle.renderPage(IMarkupWriter writer)调用,这个方法内执行呈现的具体过程如下Figure 7.18
4.1.1).调用PageRenderListener的pageBeginRender(org.apache.tapestry.event.PageEvent)方法
4.1.2).调用page对象的beginResponse(IMarkupWriter, IRequestCycle),这是最后一次机会可以修改持久属性
4.1.1).回调IRequestCycle的commitPageChanges()方法,这个方法会通知负责持久属性管理的page recorders进行相应的保存
4.1.1).调用page对象的render(IMarkupWriter, IRequestCycle),page开始呈现他的模板的内容,同样的递归呈现他包含的部件
4.1.1).调用PageRenderListener的pageEndRender(org.apache.tapestry.event.PageEvent) (this occurs even if a previous step throws an exception).
一旦呈现页面开始,持久页面属性就不能再被修改!!!
页面属性
声明的属性,持久的、临时的都可以有一个初始值,初始值或者是<property-specification>元素的"initial-value"属性的值,或者是<property-specification>元素体的内容,初始值是OGNL表达式,此表达式只被计算一次并且表达式的值被保存起来,是在page或component的finishLoad()方法被调用之后。表达式的值被用来给属性赋初值,当page被从request拆开时,为复用返回page池之前,此表达式的值被用来更新属性。初始值也可以不在<property-specification>元素指定,可以在finishLoad()方法内设置,finishLoad()方法调用完毕,tapestry框架会读这个属性,读出的值就将是属性的初值,会被保存为以后用,当page拆开时返回前,此保存的初值会被重新赋给属性,也就是说,无论是在initial-value或者是在finishLoad()中,都可指定初值,然后此初值被tapestry保存用于为属性恢复初值!
本文来源于IT民工 http://www.whitehouse.net.cn/blog/ , 原文地址:http://www.whitehouse.net.cn/blog/Article/113.html
分享到:
相关推荐
10. **Tapestry-internal-test**: 内部测试套件,用于验证Tapestry框架的内部工作原理和功能,这对于理解Tapestry的实现机制非常有帮助。 通过研究这些源码和API,开发者不仅可以掌握Tapestry的基本用法,还能深入...
- **内部架构**:深入解析Tapestry的内部架构,包括其核心模块、运行时机制等,有助于理解其运作原理。 - **组件生命周期**:详细介绍了Tapestry中组件的生命周期,包括初始化、渲染、事件处理等各个阶段。 - **高级...
总之,"tapestry-project-4.1.5"是一个宝贵的教育资源,它不仅展示了Tapestry 框架的结构和工作原理,还提供了动手实践的机会。通过对这个项目的学习和分析,开发者能够深入理解Tapestry 的组件化开发思想,掌握其...
学习如何配置缓存,并理解其工作原理。 2. **请求处理**:深入理解Tapestry的请求处理流程,包括页面选择、事件处理和响应生成。 3. **部署与性能监控**:了解如何在生产环境中部署Tapestry应用,并使用Tapestry...
本章节我们将通过分析书中提供的源代码,进一步探讨Tapestry4的核心概念、工作原理以及实际应用。 首先,我们来看"**MyTapestry4-quickStart**"这个项目。这是一个快速启动示例,它包含了创建一个基本Tapestry4应用...
### 七、Tapestry内部原理 **知识点:** - **请求处理流程**:分析Tapestry接收到HTTP请求后的处理流程。 - **页面生命周期**:深入了解页面从加载到渲染的全过程。 - **组件渲染机制**:探讨组件如何被渲染成HTML...
在提供的文件中,“Rowstron-Pastry.pdf”可能是对Pastry协议的详细描述,包括其设计原理、路由算法和性能分析,而“tapestry.pdf”则可能涵盖Tapestry框架的使用指南、最佳实践以及示例代码。阅读这些文档将有助于...
学习源码对于理解Tapestry的工作原理和开发技巧至关重要,开发者可以通过阅读源码来提升自己的技能。 “工具”标签则可能意味着这个压缩包包含了与Tapestry开发相关的辅助工具,如构建脚本、IDE配置文件等,这些...
源码分析可以帮助开发者深入理解Tapestry的工作原理,如何通过元数据驱动组件的创建和操作。 2. **tapestry-hibernate**:这个模块将Tapestry与Hibernate ORM(对象关系映射)工具集成,允许开发者使用Tapestry构建...
此“tapestry5整站源码”提供了完整的网站开发实例,帮助开发者深入理解Tapestry5的工作原理和应用实践。 Tapestry5的核心理念是将Web页面构建为一系列相互协作的组件,每个组件都是一个独立的Java类,具有自己的...
通过对这些源码的分析和实践,你将能更深入地理解Tapestry的内部工作原理,提高你的Web开发技能。同时,这些源码也可以作为你在开发过程中遇到问题时的参考,帮助你解决问题并优化你的应用。总之,《享受Web开发:...
4. **源码分析**:对于“源码”标签,我们可能要讨论如何阅读和理解Tapestry的源代码,以及如何利用源码学习Tapestry的工作原理,这对于高级开发者来说是提升技能的重要途径。 5. **开发工具**:Tapestry通常与其他...
在这个"tapestry示例源码"中,我们可以深入理解Tapestry的工作原理及其核心特性。 1. **Tapestry 框架简介** Tapestry 以其组件模型著称,它将HTML页面拆分为独立的、可重用的组件,每个组件都有自己的业务逻辑和...
该书详细介绍了如何使用Tapestry这一Java Web应用开发框架,通过实践操作来深化理解其工作原理与应用技巧。 ### 重要知识点解析 #### 1. **Tapestry框架概述** - **定义与背景**:Tapestry是一个用于构建高度交互...
通过这个小例子,我们可以深入了解Tapestry框架的核心特性和工作原理。 【描述】:“doc/readme” 描述中的"doc/readme"通常是指项目中的README文件,这是一个标准的文档,用于提供项目的基本信息、安装指南、使用...
通过这个“水果商店”案例,我们可以深入理解Tapestry4的工作原理,以及如何利用其组件化和事件驱动特性来构建复杂的Web应用。虽然现在Tapestry已经发展到了更高级的版本,但学习Tapestry4的基础仍对理解现代Web框架...
通过对这些源码的分析和研究,开发者不仅可以理解Tapestry5的工作原理,还能学习到如何有效地利用其特性构建高质量的Web应用。同时,这个源码库对于想要对Tapestry5进行扩展或定制的开发者来说,也是一份宝贵的参考...
分布式哈希表(DHT,Distributed Hash Table)是一种用于分布式系统中的数据存储技术,它通过将数据均匀地分布在大量的网络节点上,...通过阅读这些文件,你可以深入了解DHT的工作原理,并有可能实现自己的DHT系统。
3.3.5 Tapestry 48 3.3.6 Kademlia 50 3.3.7 小结 53 3.4 非结构化P2P网络的搜索技术 53 3.4.1 Flooding 54 3.4.2 Modified-BFS 55 3.4.3 Iterative Deepening 55 3.4.4 Random Walk 56 3.4.5 ...