??xml version="1.0" encoding="utf-8" standalone="yes"?>久久九九亚洲精品,国产亚洲av人片在线观看,亚洲va久久久噜噜噜久久天堂http://www.tkk7.com/hulizhong/archive/2009/12/16/306196.html二胡二胡Wed, 16 Dec 2009 09:51:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/12/16/306196.htmlhttp://www.tkk7.com/hulizhong/comments/306196.htmlhttp://www.tkk7.com/hulizhong/archive/2009/12/16/306196.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/306196.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/306196.html

关于跨域名问题还是问题么Q这斚w的解军_践非常多Q今天我旧话重提把我所知道的通过几个应用场景来分别ȝ一?/p>

先说明一点:我说的某某域名在您的控制下的意思是q个域名下的|页由您来负责开发内部的JavaScript
场景一Q将bbs.xxx.com的页面用iframe嵌入到www.xxx.com的中Q如何在iframe内外使用js通信
一U域名都是xxx.com q个域名一定是在您的控制下Q所以你只要在两个页面中同时升域名卛_
在父H口和iframe内部分别加上js语句Qdocument.domain=”xxx.com”;
之后2个页面就{于在同一域名下,通过window.parent oIframe.contentDocument可以相互访问,q行无障的JS通信
在新、淘宝等很多面都能扑ֈq样的语句。不qdocument.domain不可以随便指定,只能向上升Q从bbs.xxx.com升到yyy.com肯定会出?/p>

场景?/span>Q将www.yyy.com的页面用iframe嵌入到www.xxx.com的中Q两个域名都在您的控制下Q如何在iframe内外q行一定的数据交流
你可以通过怺改变hash值的方式来进行一些数据的通信

q里的实现基于如下技术要点:
1、父H口通过改变子窗口的src中的hash值把一部分信息传入Q如果src只有hash部分改变Q那么子H口是不会重新蝲入的?br /> 2、子H口可以重写父窗口的location.hrefQ但是注意这里子H口无法d而只能重写location.href所以要求前提是您控制两个域名,知道当前父窗口的location.href是什么ƈ写在子窗口内Q这样通过parent.location.href = “已知的父H口的href”+”#”+hash。这LH口只有hash改变也不会重载?br /> 3、上面两步分别做C两个H口之间的无h数据通知Q那么下面的来说如何感知数据变化。标准中没有相关规定Q所以当前的L览器遇到location.hash变化都不会触发Q何javaScript事gQ也是说您要自己写监听函数来监视loaction.hash的值的变化。做法是通过setTimeout或者setInterval来写一个监听函数每20-100ms查看一下hash是否变化Q如果变化了驱动jsҎ新的数据做想做的事情?/p>

q种实现的一些分析:
1、信息通道是双向的Q当然会兼容单向Q如果只是父H口向子H口通知数据Q只需要子H口写hash监听Q反之亦然?br /> 2、局限性也是颇大,因ؓq种通信的前提是双方知道Ҏ的location.href。如果父H口带有动态的location.search也就是查询参敎ͼ那么子窗口的处理上就比较困难Q需要把父窗口的location.search作ؓ传递信息的一部分告知子窗口?br /> 3、另外的困扰会有览器带l你QIE之外的浏览器遇到hash的改变会记录历史Q这样你在处理前q后退的时候会非常头疼

场景?/span>Q将www.yyy.com的页面用iframe嵌入到www.xxx.com的中Q只有被嵌入的yyy.com在您的控制下Q如何在iframe内外q行一定的交流
真实场景Qgoogle adsence的一个需求,你希望google发现您的面不能匚w出相x非常好的按点击付费q告Ӟ你希望google的广告iframe能够隐藏?br /> google的广告iframe在google域下昄不能把自己隐藏掉Q那么怎么办呢Q?br /> 1、google会提供给你一个html面
2、您这个页面放|在您的域名下,q告诉google它的位置
3、当google发现没有很好的广告时Q会子H口的loaction重定向到您的那个面下,q样您的面因ؓ同域名就可以讉K爉面来隐藏自己?br /> 是不是很巧的ҎQ?/p>

场景?/span>Q您是内容发布商Q如何改造接口,让其他域名下的页面可以从览器端出发获得您的数据
我们知道ajax的xmlHttpRequest()说到底是一个无hh服务器数据的辅助工具Q但是xmlHttpRequestq不能跨域名h数据Q在某些情况下成了极大的限制?br /> 但是我们如果通过其他方式完成无刷新请求数据不也可以么Q我们用DomҎ操作动态JS脚本h来做qg事?br />     //创徏一个脚本节?br />     var oScript = document.createElement(’script’);
    //指定脚本src src可以指向L域名
    //注意src不再指向静态jsQ而是带着查询参数指向一个动态脚本广播服务?br />     oScript.src = “http://yyy.com/query.php?”+yourQueryString;   
    //如果指定了charset 同时q可以解决xmlHttpRequest另一大困?q问题                                                                                                                           
    //oScript.charset = “utf-8″;
    //通过Dom操作把这个新的节点加入到文档当中                                 
    document.getElementsByTagName(”head”)[0].appendChild(oScript);

q样只要query.php的输出是可执行的javaScript脚本Q比如:djsCallBackQ{jsondata}Q?
当他从服务器q回后就会自动执?你可以方便的用json方式来做数据传递了?br /> 要注意,您的脚本h最好带上时间戳Q避免浏览器~存造成取回数据实时性下降?/p>

如果您是数据提供者,您可以要求数据烦取者在查询参数中提供回调函数名Q比如query.php?callback=myDataHandler&key=…?
q样您就可以Ҏ参数来提供给他myDataHandler({jsondata}),q样不同的数据烦取者都会得到自定义的正的异步回调?/p>

场景?/span>Q通过后端E序语言Qؓ了跨域名而做各自的后台数据抓取{化服务,比如php curlQYAHOO  CHINA NCP是用这U方案?/p>

场景?/span>Q通过flash proxyQ因为flash的跨域调用可以通过crossdomain.xml和security.allowdomain(’*')文g实现Q而js又可以和flashq行通信Q所以js完全可以借用flash

               实现js跨域通信?/p>

 
ȝȝQ?br /> W一U场景,相应的处理办法有q非常好的效果,可以说完全解决了问题?br /> W二U场景,相应的处理办法具有一定的跨域数据交流功效Q具有相当大的局限,q不适合在复杂业务流E中应用Q实际上我也实也没看到q基于此的大规模应用?br /> W三U场景,相应的处理办法比较y妙,虽然redirect之后׃q你什么事了,但如果你是google一样面向众多域名的内容提供商,也是个不错的解决思\?br /> W四U场景,相应的处理办法非常强大,ҎAjax可以看到Q跨域名没问题,无刷新没问题Q本w又是异步的QJSON比xml快的多,同时解决q问题Q只是请求都是Get方式的,不能做Post方式的请求。多一U武器自然可以从定w择了?/p>

W五U场景,处理很方便,也很实用?/p>

W六U场景,需要一定的flash基础哈,作用当然非常强大?/p>

二胡 2009-12-16 17:51 发表评论
]]>
l节处见功夫---cookie的应用和处理http://www.tkk7.com/hulizhong/archive/2009/12/08/305166.html二胡二胡Tue, 08 Dec 2009 07:03:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/12/08/305166.htmlhttp://www.tkk7.com/hulizhong/comments/305166.htmlhttp://www.tkk7.com/hulizhong/archive/2009/12/08/305166.html#Feedback4http://www.tkk7.com/hulizhong/comments/commentRss/305166.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/305166.html        cookie在web开发中应用的比较多!览器默认的接受cookie?但是如果览器禁止了cookie,会出C么情况了?
        在此我测试了如下pȝ:gmail,163邮箱,126邮箱,tom邮箱
        我在览器禁用cookie的情况下,登陆上述4邮箱!
        Gmail:
        正确输入用户名密码后,pȝl出如下提示:
        

    163邮箱:

 正确输入用户名密码后,pȝl出如下提示:让我搞不清楚是用户名密码错误q是其它错误!



    126邮箱
    
    正确输入用户名密码后,pȝl出如下提示:让我搞不清楚是用户名密码错误q是其它错误!

    
    tom邮箱

    正确输入用户名密码后,pȝl出如下提示:居然提示我是"非法h"

 

        从上面测试中感受C同的用户体验.对大多数做web开发的人来?判断览器是否支持Cookieq不是什么难?隄是对l节的处?
        也许我们与优U的品差距最大的是对l节的处?
        
        ? 我测试用的是IE6.0



二胡 2009-12-08 15:03 发表评论
]]>
?回忆我们l典的开发工?/title><link>http://www.tkk7.com/hulizhong/archive/2009/10/20/299010.html</link><dc:creator>二胡</dc:creator><author>二胡</author><pubDate>Tue, 20 Oct 2009 05:04:00 GMT</pubDate><guid>http://www.tkk7.com/hulizhong/archive/2009/10/20/299010.html</guid><wfw:comment>http://www.tkk7.com/hulizhong/comments/299010.html</wfw:comment><comments>http://www.tkk7.com/hulizhong/archive/2009/10/20/299010.html#Feedback</comments><slash:comments>1</slash:comments><wfw:commentRss>http://www.tkk7.com/hulizhong/comments/commentRss/299010.html</wfw:commentRss><trackback:ping>http://www.tkk7.com/hulizhong/services/trackbacks/299010.html</trackback:ping><description><![CDATA[?http://blog.csdn.net/gaoweipeng/archive/2009/10/18/4695780.aspx<br /> qx的开发如果我们能有些得心应手的开发工P好比是如虎ȝ。会大大的提高我们的开发效率。Visual Studio 自不必说Q通过此文Q和大家回忆下除此之外的l典的开发工P同时希望能把q些l典的工具介l给新手Q相信对qx的开发会有很大的帮助?br /> <br /> <img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong style="font-size: 12pt;">Internet Explorer Developer Toolbar</strong><br style="font-size: 12pt;" /> <span style="color: red;">?/span>Q微软发布了Internet Explorer Developer Toolbar最新版.该品让开发h员能够深入探索和理解Web面,帮助开发者更好地创徏Web应用.安装后可以在IE中快速分析网늚软g.该工h可集成在IEH口,或以动H口形式存在.<br /> <img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/ittoolbar.gif" height="274" width="1273" /><br /> IE Developer ToolbarҎ如?<br /> <p>览和修改Web늚文档对象模型(DOM).通过多种技术方式定位、选定Web上的特定元?止或激zIE讄.查看HTML对象的类名? ID,以及cM链接路径、tab序、快捷键{细?描绘表格、单元格、图片或选定标签的轮?昄囄象素、大、\径、替代文字等.x重定义浏览器 H口大小?00x600或自定义大小.清空览器缓存和cookie,被清除项可从所有对象或l定域中选择.直接讉K兌W3C规范参考、IE开发组 blog或其他来?Q显C计时标尺,帮助寚w对象. .... <br /> </p> <p><span style="color: red; font-family: 宋体;">下蝲地址及相兌料:</span><a target="_blank"><br /> http://www.microsoft.com/downloads/details.aspx?FamilyID=e59c3964-672d-4511-bb3e-2d5e1db91038&DisplayLang=en#Overview</a></p> <p><br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong style="font-size: 12pt;">HttpWatch</strong><br style="font-size: 12pt;" /> <span style="color: red;">?/span>Q强大的<u>|页数据分析工具</u>。集成在Internet Explorer工具栏。包括网|要。Cookies理。缓存管理。消息头发?接受。字W查询。POST 数据和目录管理功能。报告输出?/p> <p>安装完HttpWatch后,׃在浏览器中找CQ?br /> <img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/httpwatch2.gif" border="0" height="178" width="404" /><br /> ȝ面:<br /> <img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/httpwatch.gif" border="0" height="326" width="927" /> <br /> <span style="color: red;">下蝲地址Q?/span><a target="_blank">http://www.crsky.com/soft/3455.html</a></p> <p><span style="color: #ff0000;">?/span><span style="color: #ff0000;">用介l:</span><a target="_blank">http://www.cnblogs.com/mayingbao/archive/2007/11/30/978530.html</a> </p> <p><br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong style="font-size: 12pt;">Fiddler2</strong></p> <p><span style="color: red;">?/span>QFiddler2是一?u>|络调试代理</u>Q用来监本地计? 机和Internet之间所有的HTTP通讯。可以监所有的HTTP通讯Q设|断点,q且可以修改到进入到本地计算机的数据和从本地计算机出ȝ数据 Q就是可以伪造数据)。Fiddler包含一个JScript .NET 事g脚本子系l(event-based scripting subsystemQ,可以使用M一U?Net语言扩展。该软g是免费的Q支持多U浏览器Q包括Internet ExplorerQMozilla FirefoxQOpera和其它一些浏览器?/p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/fiddle2.gif" height="381" width="790" /><br /> </p> <p>从Fiddler官方|站上可以看见原版的英文介绍Q?a title="http://www.fiddler2.com/fiddler2/" target="_blank">http://www.fiddler2.com/fiddler2/</a><br /> <span style="color: red;">下蝲地址Q?/span><a title="http://www.fiddler2.com/fiddler2/" target="_blank">http://www.fiddler2.com/fiddler2/</a></p> <p><span style="color: red;">使用介绍Q?/span><a target="_blank">http://blog.csdn.net/lihongzhai/archive/2009/09/14/4551035.aspx</a> </p> <p><br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong style="font-size: 12pt;">NUnit </strong></p> <p>NUnit是一?u>单元试框架</u>Q专门针对于。NET来写?其实在前面有JUnit(Java)QCPPUnit(C++)Q他们都是xUnit的一员。最初它是从JUnit而来。现在的版本?.2.接下来我所用的都是Zq个版本?/p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/NUnit.1.jpg" height="315" width="554" /></p> <p><span style="color: red;">下蝲地址Q?/span><a target="_blank">http://www.nunit.org/index.php</a></p> <p><span style="color: red;">使用介绍Q?/span><a target="_blank">http://www.uml.org.cn/net/200702273.asp</a> </p> <p><br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong><span style="font-size: 12pt;">PowerDesigner</span></strong><br /> </p> <p><span style="color: red;">介:</span>PowerDesigner是Sybase公司的CASE工具集,使用它可以方便地对管理信息系l进行分析设计,它几乎包括了<u>数据库模型设?/u>的全q程?br /> 利用PowerDesigner可以制作数据程图、概忉|据模型、物理数据模型,可以生成多种客户端开发工L应用E序Q还可ؓ数据仓库制作l构模型Q也能对团队设计模型q行控制?br /> </p> <p>主要功能QDataArchitectQ这是一个强大的数据库设计工P使用DataArchitect 可利用实?关系图ؓ一个信息系l。创?概念数据模型"QCDMQConceptual Data ModelQ。ƈ且可ҎCDM 产生Z某一特定数据库管理系l(例如QSybase System 11Q的"物理数据模型"-PDM(Physical Data Model)。还可优化PDMQ生ؓ特定DBMS 创徏数据库的SQL 语句q可以文件Ş式存储以便在其他时刻q行q些SQL 语句创徏数据库。另外,DataArchitectq可Ҏ已存在的数据库反向生?br /> PDMQCDM 及创建数据库的SQL脚本?br /> ProcessAnalystQ这部分用于创徏功能模型和数据流图,创徏"处理层次关系"?br /> AppModelerQؓ客户/服务器应用程序创建应用模型?br /> ODBC AdministratorQ此部分用来理pȝ的各U数据源?br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/PD_03_440.jpg" height="539" width="752" /><br /> </p> <p><span style="color: red;">下蝲地址Q?/span><a target="_blank">http://www.baidu.com/s?tn=sitehao123&bs=PowerDesigner&f=3&wd=powerdesigner%CF%C2%D4%D8&oq=powerdesigner&rsp=0</a> Q?<br /> </p> <p><span style="color: red;">使用介绍Q?</span><a target="_blank">http://www.cnblogs.com/yxonline/archive/2007/04.html</a> <br /> </p> <p><br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /> <strong style="font-size: 12pt;">Reflector</strong></p> <p><span style="color: red;">介:</span>Visual Studio内置的ILDASM成ؓ最?u>挖掘E序集的上佳工具</u>? 但自从Reflector出现后,ILDASM相Ş见拙。因为,Reflector能提供更多的E序集信息。Reflector可以?NETE序集中? 中间语言反编译成C#或者Visual Basic代码。除了能IL转换为C#或Visual Basic以外QReflectorq能够提供程序集中类及其成员的概要信息、提供查看程序集中IL的能力以及提供对W三Ҏ件的支持?img alt="" src="http://www.cnblogs.com/gaoweipeng/admin/file:///C:/DOCUME%7E1/ADMINI%7E1/LOCALS%7E1/Temp/moz-screenshot.png" /></p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/reflector.JPG" height="465" width="778" /><br /> </p> <p><span style="color: red;">下蝲地址</span><span style="color: red;">Q?</span><a target="_blank">/Files/gaoweipeng/Reflector.rar</a></p> <p><span style="color: red;">使用介绍及其他相关下载:</span><a target="_blank">http://www.cnblogs.com/zzy2740/archive/2005/09/20/240216.html</a></p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong style="font-size: 12pt;">1st JavaScript Editor</strong></p> <p><span style="color: red;">介:</span> 1st JavaScript Editor 是一Ƒּ大的<u>JavaScript开发、校验和调试工具</u>Q? 它简单易用,不论你是初学者或者是专业的开发h士,都可以轻松上手!同时它又是完的Ajax (Asynchronous JavaScript and XML),CSS, HTML, DOM DHTML开发工P它提供了单而且实用的功能:丰富的代码编辑功?JavaScript, HTML, CSS, VBScript, PHP ,ASP(Net))Q语法高亮,内置预览功能Q提供了完整的HTML 标记, HTML 属? HTML 事g, JavaScript 事g和JavaScript 函数{完整的代码库,同时有着贴心的代码自动补全功能?br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/jsEdit.JPG" height="470" width="698" /><br /> </p> <p>官网及下载地址Q?span style="font-size: 10.5pt; font-family: 'Times New Roman';"><a target="_blank">http://yaldex.com/</a></span><br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/193608/o_comment.gif" border="0" height="24" width="32" /><strong style="font-size: 12pt;">Regulator</strong></p> <p><span style="color: red;">?/span>Q用Reglator可以方便的验?u>正则表达式的执行l果</u>Q带有智能感知功能。帮助文档是一个很好的学习正则的入门教E,也可以当作参考手册。基本上全了,同时q行学习和实c还有生成c# or vb.net代码{附加功能?cM的Y件现在很多:RegexBuddy QRegexTester。等<br /> </p> <p><img alt="" src="http://images.cnblogs.com/cnblogs_com/gaoweipeng/regulator_main_1245339666.png" height="424" width="682" /></p> <p> <span style="color: red;">下蝲地址Q?/span><a title="" target="_blank">http://sourceforge.net/projects/regulator/</a></p> <p> <span style="color: red;">使用介绍Q?/span><a target="_blank">http://www.ctochina.net/topic/ShowArticle/112.html</a></p> <p> 结Q上面的开发工具都是我qx喜欢用的Q希望通过此文的介l,Ҏ有用过的朋友带来帮助。也希望园子中的朋友?strong>U一U自己qx常用的开发工?/strong>Q分享些更实用,方便的开发工P</p> <br /> <br /><img src ="http://www.tkk7.com/hulizhong/aggbug/299010.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/hulizhong/" target="_blank">二胡</a> 2009-10-20 13:04 <a href="http://www.tkk7.com/hulizhong/archive/2009/10/20/299010.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>?HTTP状态码QHTTP Status codesQ简?http://www.tkk7.com/hulizhong/archive/2009/05/06/269205.html二胡二胡Wed, 06 May 2009 03:38:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/05/06/269205.htmlhttp://www.tkk7.com/hulizhong/comments/269205.htmlhttp://www.tkk7.com/hulizhong/archive/2009/05/06/269205.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/269205.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/269205.html
q来L朋友咨询cPanel 的Awstats?#8220;HTTP错误码(HTTP Error codesQ?#8221;的含义,以及是否需要关注和处理?br /> 关于Awstatsh看《CP How-ToQ如何用cPanel查看站点l计数据QawstatsQ?文章地址Q?http://bbs.netpu.net/viewthread.php tid=694 其实q是一个误会,在这里它应该?#8220;HTTP状态码QHTTP Status codesQ?#8221;?br /> OKQ既然是状态码Q那么就可能有正和错误两种状态了Q至不全是错误了,大大的松口气吧)。那么这些代码都代表什么含义呢Q到底哪些是错误状态,哪些是正状态呢Q不要急,下边我冒充内行ؓ大家做一个简单的介绍?br /> HTTP与Status codes
HTTP可能大家都熟悉,是文本传输协议。浏览器通过HTTP与WEB Server通讯Q也有一些其它Y件比如IM使用HTTP协议传递数据)Q把我们的请求(HTTP RequestQ传递给服务器,服务器响应这个请求,q回应答QHTTP ResponseQ以及我们需要的数据。大致就是这个样子了?br /> 如果 我们h啥,服务器就q回啥,是乎׃需要HTTP Status codes了。但是事情往往不是那么单。比如我们请求一个网页面,可是服务器不存在q个面Q或者这个页面被转移到其它地方,或者服务器止我们查看 q个面{等Q所以ؓ了便于浏览器处理q些正确与不正确的情况,HTTP用Status codes来表C求(HTTP RequestQ在服务器端被处理的情况。Status codes通过应答QHTTP ResponseQ返回给览器,览器根据这个作相应的处理?br /> HTTP Status codes的分c?br /> 既然有正和错误的状态,HTTP定义两大cȝ态码是不是就可以了?人家制订协议的可是专Ӟ不象我是冒充的)Q想得比我们要周全,要长q。HTTP 1.1中定义了5大类Status codesQ分别是Q?br /> Informational 意义Q信?范围Q?XX
Successful 意义Q成?范围Q?XX
Redirection 意义Q重定向 范围Q?XX
Client Error 意义Q客L错误 范围Q?XX
Server Error 意义Q服务器错误 范围Q?XX
您看看h家想得多周到啊,真专家就是真专家?br /> 常见HTTP Status codes?br /> 下面单介l一下我们经常碰到的HTTP Status codes?br /> 也许是我孤陋寡闻Q常遇到的HTTP Status codes那么几个,见笑啦?br /> Successful 200 - OKQOK q个是最常见的啦Q也许我们不会直接看刎ͼ但是如果您用一些抓包工P大多数http应答中都有这个)。意义很单,是说服务器收到q理解客L的请 求而且正常处理了?206 - Partial ContentQ部分内宏V?q个也经常发生。很Ҏ让大家发c? 通俗点说是如果客户端请求文档(囑փQ文本,声音{等Q的部分内容Q服务器正常处理Q那么就q回206。大致意思就是它h的时候,除了指定h的内 容,q指定了偏移量以及长度?部分内容Q没搞错吧?呵呵没搞错,现在很多览器以及Y件支持断点箋传就是靠q个的。呵呵,以后看到206不要怕了?br /> Redirection 301 - Moved PermanentlyQ永久移动? q个状态码是指客户端要h的资源已l被怹的{Ud一个新的地方了。这个应{(HTTP ResponseQ里边同时包含了资源的新地址。它告诉客户端,如果下次q想要这个资源,那么q新的地址d?302 FoundQ旉定向? q个状态码是指客户端要h的资源时放C个新地方了。同P应答中也包含了资源的新地址?307 - Temporary RedirectQ旉定向。(如果不去实现协议或者做相关开发,我们大致理解它很302差不多就可以啦) 有一谈重定向的文章《从Business.com遭封?02重定向》很有意?我们转蝲了一份,地址Q?http://bbs.netpu.net/viewthread.php tid=810 Client Error 400 - Bad RequestQ错误请?是h的语法错误,服务器无法理解?401 – UnauthorizedQ未授权 当服务器q回401 CodeQ就是告诉说客户端访问指定资源以前,必须通过服务器的授权?403 – ForbiddenQ禁止访?是不允许访问某些资源?404 - Not FoundQ找不到 找不到客Lh的内?br /> Server Error 500 - Internal Server Error 服务器内部错误?br /> l束?br /> 写懒Q所以就单单写这么多啦。没有啥大用处。如果能消除一两位关于q方面朋友的疑虑Q就以令我ƣ慰了。既然之前都说过是冒充内行,所以有错漏之处在所隑օQ还望大家不吝赐教?br /> 需要深入研I这斚w内容的朋友,千万不要看这文章,以免被此文误对{请学习官方的协议内宏V?官方的资料地址:
http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html




二胡 2009-05-06 11:38 发表评论
]]>
?利用JExcelApi来动态生成excel文档 http://www.tkk7.com/hulizhong/archive/2009/04/28/267825.html二胡二胡Tue, 28 Apr 2009 01:02:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/04/28/267825.htmlhttp://www.tkk7.com/hulizhong/comments/267825.htmlhttp://www.tkk7.com/hulizhong/archive/2009/04/28/267825.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/267825.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/267825.html

首先Q请?a >http://www.andykhan.com/jexcelapi/index.html下蝲java excel apiQ主上同时有比较详l的介绍。最新版本ؓ2.4.3Q同时也可以刎ͼhttp://www.andykhan.com/jexcelapi/jexcelapi_2_4_3.tar.gz下蝲到该最新版的APIQ由于该目是开源的Q所以下载的文g中已l包含了源代码,同样的,文g中也有javadocQ大家在开发中可以参考javadoc?/p>

下蝲完毕后,我们需要把文g中的jxl.jar加入C的开发classpath中?br /> 下图是现在要生的excel截图Q?br /> http://blog.csdn.net/beming/gallery/image/3437.aspx

代码如下Q?/p>

   File excel = new File("d:/aming.xls");
   if(!excel.exists()){
    excel.createNewFile();
   }   
   WritableWorkbook wwb = Workbook.createWorkbook(excel);
   WritableSheet ws = wwb.createSheet("testexcel",0);
   Label lable = null;
   
   //对中文的支持非常?br />    lable = new Label(0,0,"我的中国?);
   ws.addCell(lable);
   
   //可以定义模板格式化你的cell
   WritableFont wf = new WritableFont(WritableFont.ARIAL,10,WritableFont.NO_BOLD,false,UnderlineStyle.NO_UNDERLINE, Colour.BLACK);
   WritableCellFormat wcf = new WritableCellFormat(wf);
   wcf.setBackground(Colour.WHITE);
   lable = new Label(0,1,"fdsl",wcf);
   ws.addCell(lable);
   
   wf = new WritableFont(WritableFont.TIMES,18,WritableFont.BOLD,true);
   wcf = new WritableCellFormat(wf);
   lable = new Label(0,2,"aming",wcf);
   ws.addCell(lable);
   
   //cell的类型同样可以定义ؓ数字cd
   Number nb = new Number(0,3,21.4321321);
   ws.addCell(nb);
   
   //支持格式化你的数字串
   NumberFormat nf = new NumberFormat("#.###");
   wcf = new WritableCellFormat(nf);
   nb = new Number(0,4,21.43254354354354,wcf);
   ws.addCell(nb);

   //cell的类型可以ؓbooleancd
   Boolean bl = new Boolean(0,5,true);
   ws.addCell(bl);

   //cell的类型同样可以ؓ日期Q时?br />    DateTime dt = new DateTime(0,6,new Date());
   ws.addCell(dt);

   //q且可以很好格式化你的日期格?br />    DateFormat df = new DateFormat("MM dd yyyy hh:mm:ss");
   wcf = new WritableCellFormat(df);
   dt = new DateTime(0,7,new Date(),wcf);
   ws.addCell(dt);
   
   //开始写文g?br />    wwb.write();
   wwb.close();

上面的下载地址无法打开.
下蝲请到:http://prdownloads.sourceforge.net/jexcelapi





二胡 2009-04-28 09:02 发表评论
]]>
?web.xml配置详细说明(?http://www.tkk7.com/hulizhong/archive/2009/04/24/267404.html二胡二胡Fri, 24 Apr 2009 09:17:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/04/24/267404.htmlhttp://www.tkk7.com/hulizhong/comments/267404.htmlhttp://www.tkk7.com/hulizhong/archive/2009/04/24/267404.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/267404.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/267404.html

5.2 分配JSP初始化参?
lJSP面提供初始化参数在三个斚w不同于给servlet提供初始化参数?
1Q用jsp-file而不是servlet-class。因此,WEB-INF/web.xml文g的servlet元素如下所C:

Java代码
  1. <servlet>   
  2. <servlet-name>PageName</servlet-name>   
  3. <jsp-file>/RealPage.jsp</jsp-file>   
  4. <init-param>   
  5. <param-name>...</param-name>   
  6. <param-value>...</param-value>   
  7. </init-param>   
  8. ...   
  9. </servlet>  

2) 几乎L分配一个明的URL模式。对servletQ一般相应地使用以http://host/webAppPrefix/servlet/ 开始的~省URL。只需CQ用注册名而不是原名称卛_。这对于JSP面在技术上也是合法的。例如,在上面给出的例子中,可用URL http://host/webAppPrefix/servlet/PageName 讉KRealPage.jsp的对初始化参数具有访问权的版本。但在用于JSP面Ӟ许多用户g不喜Ƣ应用常规的servlet的URL。此外,如果 JSP面位于服务器ؓ其提供了目录清单的目录中Q如Q一个既没有index.html也没有index.jsp文g的目录)Q则用户可能会连接到? JSP面Q单dQ从而意外地ȀzL初始化的面。因此,好的办法是用url-patternQ?.3节)JSP面的原URL与注册的 servlet名相兌。这P客户机可使用JSP面的普通名Uͼ但仍然激zd制的版本。例如,l定来自目1的servlet定义Q可使用下面? servlet-mapping定义Q?/p>

Java代码
  1. <servlet-mapping>   
  2. <servlet-name>PageName</servlet-name>   
  3. <url-pattern>/RealPage.jsp</url-pattern>   
  4. </servlet-mapping>  

3QJSP用jspInit而不是init。自动从JSP面建立的servlet或许已经使用了intiҎ。因此,使用JSP声明提供一个initҎ是不合法的,必须制定jspInitҎ?
? 了说明初始化JSP面的过E,E序清单5-9l出了一个名为InitPage.jsp的JSP面Q它包含一个jspInitҎ且放|于 deployDemo Web应用层次l构的顶层。一般,http://host/deployDemo/InitPage.jsp 形式的URL激zL面的不h初始化参数访问权的版本,从而将对firstName和emailAddress变量昄null。但是, web.xml文gQ程序清?-10Q分配了一个注册名Q然后将该注册名与URL模式/InitPage.jsp相关联?

E序清单5-9 InitPage.jsp

Java代码
  1. <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">   
  2. <HTML>   
  3. <HEAD><TITLE>JSP Init Test</TITLE></HEAD>   
  4. <BODY BGCOLOR="#FDF5E6">   
  5. <H2>Init Parameters:</H2>   
  6. <UL>   
  7. <LI>First name: <%= firstName %>   
  8. <LI>Email address: <%= emailAddress %>   
  9. </UL>   
  10. </BODY></HTML>   
  11. <%!   
  12. private String firstName, emailAddress;   
  13.   
  14. public void jspInit() {   
  15. ServletConfig config = getServletConfig();   
  16. firstName = config.getInitParameter("firstName");   
  17. emailAddress = config.getInitParameter("emailAddress");   
  18. }   
  19. %>  



E序清单5-10 web.xmlQ说明JSP面的init参数的摘录)

Java代码
  1. <?xml version="1.0" encoding="ISO-8859-1"?>   
  2. <!DOCTYPE web-app   
  3. PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"  
  4. "http://java.sun.com/dtd/web-app_2_3.dtd">   
  5.   
  6. <web-app>   
  7. <!-- ... -->   
  8. <servlet>   
  9. <servlet-name>InitPage</servlet-name>   
  10. <jsp-file>/InitPage.jsp</jsp-file>   
  11. <init-param>   
  12. <param-name>firstName</param-name>   
  13. <param-value>Bill</param-value>   
  14. </init-param>   
  15. <init-param>   
  16. <param-name>emailAddress</param-name>   
  17. <param-value>gates@oracle.com</param-value>   
  18. </init-param>   
  19. </servlet>   
  20. <!-- ... -->   
  21. <servlet-mapping>   
  22. <servlet-name> InitPage</servlet-name>   
  23. <url-pattern>/InitPage.jsp</url-pattern>   
  24. </servlet-mapping>   
  25. <!-- ... -->   
  26. </web-app>  



5.3 提供应用范围内的初始化参?
一般,对单个地servlet或JSP面分配初始化参数。指定的servlet? JSP面利用ServletConfig的getInitParameterҎdq些参数。但是,在某些情形下Q希望提供可׃Q意servlet? JSP面借助ServletContext的getInitParameterҎd的系l范围内的初始化参数?
可利用context-param元素声明q些pȝ范围内的初始化倹{context-param元素应该包含param-name、param-value以及可选的description子元素,如下所C:
<context-param>
<param-name>support-email</param-name>
<param-value>blackhole@mycompany.com</param-value>
</context-param>
? 回忆一下,Z保证可移植性,web.xml内的元素必须以正的ơ序声明。但q里应该注意Qcontext-param元素必须出现L与文档有关的? 素(icon、display-name或descriptionQ之后及filter、filter-mapping、listener? servlet元素之前?
5.4 在服务器启动时装载servlet
假如servlet或JSP面有一个要花很长时间执行的 init QservletQ或jspInitQJSPQ方法。例如,假如init或jspInitҎ从某个数据库或ResourceBundle查找产量。这U? 情况下,在第一个客hh时装载servlet的缺省行为将对第一个客h产生较长旉的gq。因此,可利用servlet的load-on- startup元素规定服务器在W一ơ启动时装蝲servlet。下面是一个例子?/p>

Java代码
  1. <servlet>   
  2. <servlet-name> … </servlet-name>   
  3. <servlet-class> … </servlet-class> <!-- or jsp-file -->   
  4. <load-on-startup/>   
  5. </servlet>  

可以为此元素体提供一个整数而不是用一个空的load-on-startup。想法是服务器应该在装蝲较大数目的servlet或JSP面之前 装蝲较少数目的servlet或JSP面。例如,下面的servlet(攄在Web应用的WEB-INF目录下的web.xml文g中的web- app元素内)指C服务器首先装蝲和初始化SearchServletQ然后装载和初始化由位于Web应用的result目录中的index.jsp? 件生的 servlet?/p>

Java代码
  1. <servlet>   
  2. <servlet-name>Search</servlet-name>   
  3. <servlet-class>myPackage.SearchServlet</servlet-class> <!-- or jsp-file -->   
  4. <load-on-startup>1</load-on-startup>   
  5. </servlet>   
  6. <servlet>   
  7. <servlet-name>Results</servlet-name>   
  8. <servlet-class>/results/index.jsp</servlet-class> <!-- or jsp-file -->   
  9. <load-on-startup>2</load-on-startup>   
  10. </servlet>  


6 声明qo?/span>

servlet版本2.3引入了过滤器的概c虽然所有支持servlet API版本2.3的服务器都支持过滤器Q但Z使用与过滤器有关的元素,必须在web.xml中用版?.3的DTD?
q? 滤器可截取和修改q入一个servlet或JSP面的请求或从一个servlet或JSP面发出的相应。在执行一个servlet或JSP面之前Q? 必须执行W一个相关的qo器的doFilterҎ。在该过滤器对其FilterChain对象调用doFilterӞ执行链中的下一个过滤器。如果没 有其他过滤器Qservlet或JSP面被执行。过滤器h对到来的ServletRequest对象的全部访问权Q因此,它们可以查看客户机名、查? 到来的cookie{。ؓ了访问servlet或JSP面的输出,qo器可响应对象包裹在一个替w对象(stand-in objectQ中Q比方说把输出篏加到一个缓冲区。在调用FilterChain对象的doFilterҎ之后Q过滤器可检查缓冲区Q如有必要,对? q行修改Q然后传送到客户机?
例如Q程序清?-11帝国难以了一个简单的qo器,只要讉K相关的servlet或JSP面Q它截取请求ƈ在标准输Z打印一个报告(开发过E中在桌面系l上q行Ӟ大多数服务器都可以用这个过滤器Q?

E序清单5-11 ReportFilter.java

Java代码
  1. package moreservlets;   
  2.   
  3. import java.io.*;   
  4. import javax.servlet.*;   
  5. import javax.servlet.http.*;   
  6. import java.util.*;   
  7.   
  8. /** Simple filter that prints a report on the standard output
  9. * whenever the associated servlet or JSP page is accessed.
  10. * <P>
  11. * Taken from More Servlets and JavaServer Pages
  12. * from Prentice Hall and Sun Microsystems Press,
  13. * http://www.moreservlets.com/.
  14. * © 2002 Marty Hall; may be freely used or adapted.
  15. */  
  16.   
  17. public class ReportFilter implements Filter {   
  18. public void doFilter(ServletRequest request,   
  19. ServletResponse response,   
  20. FilterChain chain)   
  21. throws ServletException, IOException {   
  22. HttpServletRequest req = (HttpServletRequest)request;   
  23. System.out.println(req.getRemoteHost() +   
  24. " tried to access " +   
  25. req.getRequestURL() +   
  26. " on " + new Date() + ".");   
  27. chain.doFilter(request,response);   
  28. }   
  29.   
  30. public void init(FilterConfig config)   
  31. throws ServletException {   
  32. }   
  33.   
  34. public void destroy() {}   
  35. }  

一旦徏立了一个过滤器Q可以在web.xml中利用filter元素以及filter-nameQQ意名Uͼ、file-classQ完全限定的c? 名)和(可选的Qinit-params子元素声明它。请注意Q元素在web.xml的web-app元素中出现的ơ序不是L的;允许服务器(但不是必 需的)强制所需的次序,q且实际中有些服务器也是q样做的。但q里要注意,所有filter元素必须出现在Q意filter-mapping元素之前Q? filter-mapping元素又必d现在所有servlet或servlet-mapping元素之前?
例如Q给定上q的ReportFilterc,可在web.xml中作Z面的filter声明。它把名UReporter与实际的cReportFilterQ位于moreservletsE序包中Q相兌?/p>

Java代码
  1. <filter>   
  2. <filter-name>Reporter</filter-name>   
  3. <filter-class>moresevlets.ReportFilter</filter-class>   
  4. </filter>  

一旦命名了一个过滤器Q可利用filter-mapping元素把它与一个或多个servlet或JSP面相关联。关于此工作有两种选择?
? 先,可用filter-name和servlet-name子元素把此过滤器与一个特定的servlet名(此servlet名必ȝ后在相同? web.xml文g中用servlet元素声明Q关联。例如,下面的程序片断指C系l只要利用一个定制的URL讉K名ؓSomeServletName 的servlet或JSP面Q就q行名ؓReporter的过滤器?/p>

Java代码
  1. <filter-mapping>   
  2. <filter-name>Reporter</filter-name>   
  3. <servlet-name>SomeServletName</servlet-name>   
  4. </filter-mapping>  

其次Q可利用filter-name和url-pattern子元素将qo器与一lservlet、JSP面或静态内容相兌。例如,盔R的程序片D|C系l只要访问Web应用中的LURLQ就q行名ؓReporter的过滤器?/p>

Java代码
  1. <filter-mapping>   
  2. <filter-name>Reporter</filter-name>   
  3. <url-pattern>/*</url-pattern>   
  4. </filter-mapping>  

例如Q程序清?-12l出了将ReportFilterqo器与名ؓPageName的servlet相关联的web.xml文g的一部分。名? PageName依次又与一个名为TestPage.jsp的JSP面以及以模式http: //host/webAppPrefix/UrlTest2/ 开头的URL相关联。TestPage.jsp的源代码已经JSP面命名的谈论在前面??分配名称和定制的URL"中给出。事实上Q程序清?- 12中的servlet和servlet-name从该节原封不动地拿q来的。给定这些web.xml,可看C面的标准输出形式的调试报告(换行? ZҎ阅读Q?
audit.irs.gov tried to access
http://mycompany.com/deployDemo/UrlTest2/business/tax-plan.html
on Tue Dec 25 13:12:29 EDT 2001.

E序清单5-12 Web.xmlQ说明filter用法的摘录)

Java代码
  1. <?xml version="1.0" encoding="ISO-8859-1"?>   
  2. <!DOCTYPE web-app   
  3. PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"  
  4. "http://java.sun.com/dtd/web-app_2_3.dtd">   
  5.   
  6. <web-app>   
  7. <filter>   
  8. <filter-name>Reporter</filter-name>   
  9. <filter-class>moresevlets.ReportFilter</filter-class>   
  10. </filter>   
  11. <!-- ... -->   
  12. <filter-mapping>   
  13. <filter-name>Reporter</filter-name>   
  14. <servlet-name>PageName</servlet-name>   
  15. </filter-mapping>   
  16. <!-- ... -->   
  17. <servlet>   
  18. <servlet-name>PageName</servlet-name>   
  19. <jsp-file>/RealPage.jsp</jsp-file>   
  20. </servlet>   
  21. <!-- ... -->   
  22. <servlet-mapping>   
  23. <servlet-name> PageName </servlet-name>   
  24. <url-pattern>/UrlTest2/*</url-pattern>   
  25. </servlet-mapping>   
  26. <!-- ... -->   
  27. </web-app>  


7 指定Ƣ迎?假如用户提供了一个像http: //host/webAppPrefix/directoryName/ q样的包含一个目录名但没有包含文件名的URLQ会发生什么事情呢Q用戯得到一个目录表Q一个错误?q是标准文g的内容?如果得到标准文g内容Q是 index.html、index.jsp、default.html、default.htm或别的什么东西呢Q? Welcome-file-list 元素及其辅助的welcome-file元素解决了这个模p的问题。例如,下面的web.xmlҎ出,如果一个URLl出一个目录名但未l出文g名,? 务器应该首先试用index.jspQ然后再试用index.html。如果两者都没有扑ֈQ则l果有赖于所用的服务器(如一个目录列表)? 虽然许多服务器缺省遵循这U行为,但不一定必这栗因此,明确C用welcom-file-list保证可移植性是一U良好的习惯?

8 指定处理错误的页? 现在我了解到Q你在开发servlet和JSP面时从不会犯错误,而且你的所有页面是那样的清晎ͼ一般的E序员都不会被它们的搞糊涂。但是,是hM? 错误的,用户可能会提供不合规定的参数Q用不正确的URL或者不能提供必需的表单字D倹{除此之外,其它开发h员可能不那么l心Q他们应该有些工h? 服自q不?error-page元素是用来克服q些问题的。它有两个可能的子元素,分别是:error-code和exception- type。第一个子元素error-code指出在给定的HTTP错误代码出现时用的URL。第二个子元素excpetion-type指出在出现某? l定的Java异常但未捕捉到时使用的URL。error-code和exception-type都利用location元素指出相应的URL。此 URL必须?开始。location所指出的位|处的页面可通过查找HttpServletRequest对象的两个专门的属性来讉K关于错误的信息, q两个属性分别是Qjavax.servlet.error.status_code和javax.servlet.error.message? 可回忆一下,在web.xml内以正确的次序声明web-app的子元素很重要。这里只要记住,error-page出现在web.xml文g的末N q,servlet、servlet-name和welcome-file-list之后卛_?

8.1 error-code元素 Z更好C解error-code元素的|可考虑一下如果不正确地输入文件名Q大多数站点会作Z么反映。这样做一般会出现一?04错误信息Q它? CZ能找到该文gQ但几乎没提供更多有用的信息。另一斚wQ可以试一下在www.microsoft.com、www.ibm.com 处或者特别是在www.bea.com 处输出未知的文g名。这是会得出有用的消息,q些消息提供可选择的位|,以便查找感兴的面。提供这h用的错误面对于Web应用来说是很有h值得? 事实上rm-error-page子元素)。由form-login-pagel出的HTML表单必须h一个j_security_check? ACTION属性、一个名为j_username的用户名文本字段以及一个名为j_password的口令字Dc? 例如Q程序清?-19指示服务器用基于表单的验证。Web应用的顶层目录中的一个名为login.jsp的页面将攉用户名和口oQƈ且失败的登陆? q同目录中名ؓlogin-error.jsp的页面报告?E序清单5-19 web.xmlQ说明login-config的摘录)

9.2 限制对Web资源的访?现在Q可以指C服务器使用何种验证Ҏ了?了不P"你说道,"除非我能指定一个来收到保护? URLQ否则没有多大用处?没错。指些URLq说明他们应该得CU保护正是security-constriaint元素的用途。此元素? web.xml中应该出现在login-config的紧前面。它包含是个可能的子元素Q分别是Qweb-resource-collection? auth-constraint、user-data-constraint和display-name。下面各节对它们进行介l?l web-resource-collection 此元素确定应该保护的资源。所有security-constraint元素都必d含至一个web-resource-collectionV此? 素由一个给ZQ意标识名U的web-resource-name元素、一个确定应该保护的URL的url-pattern元素、一个指出此保护所适用? HTTP命oQGET、POST{,~省为所有方法)的http-method元素和一个提供资料的可选description元素l成。例如,下面? Web-resource-collection(在security-constratint元素内)指出Web应用的proprietary目录? 所有文档应该受C护?重要的是应该注意刎ͼurl-pattern仅适用于直接访问这些资源的客户机。特别是Q它不适合于通过MVC体系l构利用 RequestDispatcher来访问的面Q或者不适合于利用类似jsp:forward的手D|讉K的页面。这U不匀U如果利用得当的话很有好 处。例如,servlet可利用MVC体系l构查找数据Q把它放到bean中,发送请求到从bean中提取数据的JSP面q显C它。我们希望保证决不直 接访问受保护的JSP面Q而只是通过建立该页面将使用的bean的servlet来访问它。url-pattern和auth-contraint元素 可通过声明不允怓Q何用L接访问JSP面来提供这U保证。但是,q种不匀U的行ؓ可能让开发h员放松警惕,使他们偶然对应受保护的资源提供不受限制的 讉K?l auth-constraint 管web-resource-collention元素质出了哪些URL应该受到保护Q但是auth-constraint元素却指出哪些用户应该具? 受保护资源的讉K权。此元素应该包含一个或多个标识h讉K权限的用L别role- name元素Q以及包含(可选)一个描q角色的description元素。例如,下面web.xml中的security-constraint元素? 门规定只有指定ؓAdministrator或Big KahunaQ或两者)的用户具有指定资源的讉K权? 重要的是认识刎ͼ到此为止Q这个过E的可移植部分结束了。服务器怎样定哪些用户处于M角色以及它怎样存放用户的口令,完全有赖于具体的pȝ? 例如QTomcat使用install_dir/conf/tomcat-users.xml用户名与角色名和口令相兌Q正如下面例子中所C,它指? 用户joeQ口令bigshotQ和janeQ口令enajQ属于administrator和kahuna角色?l user-data-constraint q个可选的元素指出在访问相兌源时使用M传输层保护。它必须包含一个transport-guarantee子元素(合法gؓNONE? INTEGRAL或CONFIDENTIALQ,q且可选地包含一个description元素。transport-guarantee为NONE值将 Ҏ用的通讯协议不加限制。INTEGRALDC数据必M一U防止截取它的h阅读它的方式传送。虽然原理上Qƈ且在未来的HTTP版本中)Q在 INTEGRAL和CONFIDENTIAL之间可能会有差别Q但在当前实践中Q他们都只是单地要求用SSL。例如,下面指示服务器只允许对相兌源做 HTTPSq接Q?l display-name security-constraint的这个很用的子元素给予可能由GUI工具使用的安全约束项一个名U?9.3 分配角色? q今为止Q讨论已l集中到完全由容器(服务器)处理的安全问题之上了。但servlet以及JSP面也能够处理它们自q安全问题? 例如Q容器可能允许用户从bigwig或bigcheese角色讉K一个显CZh员额外紧贴的面Q但只允许bigwig用户修改此页面的参数。完成这 U更l致的控制的一U常见方法是调用HttpServletRequset的isUserInRoleҎQƈ据此修改讉K?Servlet? security-role-ref子元素提供出现在服务器专用口令文件中的安全角色名的一个别名。例如,假如~写了一个调? request.isUserInRoleQ?boss"Q的servletQ但后来该servlet被用在了一个其口o文g调用角色manager而不 是boss的服务器中。下面的E序D该servlet能够使用q两个名UC的Q何一个? 也可以在web-app内利用security-role元素提供出现在role-name元素中的所有安全角色的一个全局列表。分别地生命角色佉KU? IDEҎ处理安全信息?10 控制会话时 如果某个会话在一定的旉内未被访问,服务器可把它扔掉以节U内存。可利用HttpSession的setMaxInactiveIntervalҎ? 接设|个别会话对象的时倹{如果不采用q种ҎQ则~省的超时值由具体的服务器军_。但可利用session-config和session- timeout元素来给Z个适用于所有服务器的明的时倹{超时值的单位为分钟,因此Q下面的例子讄~省会话时gؓ三个时Q?80分钟Q?/p>

11 Web应用的文档化 来多的开发环境开始提供servlet和JSP的直接支持。例子有Borland Jbuilder Enterprise Edition、Macromedia UltraDev、Allaire JRun StudioQ写此文Ӟ已被Macromedia收购Q以及IBM VisuaAge for Java{? 大量的web.xml元素不仅是ؓ服务器设计的Q而且q是为可视开发环境设计的。它们包括icon、display-name和discription {? 可回忆一下,在web.xml内以适当地次序声明web-app子元素很重要。不q,q里只要Cicon、display-name? description是web.xml的web-app元素内的前三个合法元素即可?l icon icon元素指出GUI工具可用来代表Web应用的一个和两个囑փ文g。可利用small-icon元素指定一q?6 x 16的GIF或JPEG囑փQ用large-icon元素指定一q?2 x 32的图像。下面D一个例子: l display-name display-name元素提供GUI工具可能会用来标记此Web应用的一个名U。下面是个例子?l description description元素提供解释性文本,如下所C: 12 兌文g与MIMEcd 服务器一般都h一U让Web站点理员将文g扩展名与媒体相关联的Ҏ。例如,会自动l予名ؓmom.jpg的文件一个image/jpeg? MIME cd。但是,假如你的Web应用h几个不寻常的文gQ你希望保证它们在发送到客户机时分配为某UMIMEcd。mime-mapping元素Q具? extension和mime-type子元素)可提供这U保证。例如,下面的代码指C服务器application/x-fubar的MIMEcd? 配给所有以.fool尾的文件? 或许Q你的Web应用希望重蝲QoverrideQ标准的映射。例如,下面的代码将告诉服务器在发送到客户机时指定.ps文g作ؓU文? Qtext/plainQ而不是作为PostScriptQapplication/postscriptQ?/p>

13 定位TLD JSP taglib元素h一个必要的uri属性,它给Z个TLDQTag Library DescriptorQ文件相对于Web应用的根的位|。TLD文g的实际名U在发布新的标签库版本时可能会改变,但我们希望避免更Ҏ有现有JSP? 面。此外,可能q希望用保持taglib元素的简l性的一个简短的uri。这是部v描述W文件的taglib元素z场的所在了。Taglib包含? 个子元素Qtaglib-uri和taglib-location。taglib-uri元素应该与用于JSP taglib元素的uri属性的东西相匹配。Taglib-location元素l出TLD文g的实际位|。例如,假如你将文gchart-tags- 1.3beta.tld攑֜WebApp/WEB-INF/tlds中。现在,假如web.xml在web-app元素内包含下列内宏V? l出q个说明后,JSP面可通过下面的简化Ş式用标{ֺ?14 指定应用事g监听E序 应用事g监听器程序是建立或修改servlet环境或会话对象时通知的类。它们是servlet规范的版?.3中的新内宏V这里只单地说明用来? Web应用注册一个监听程序的web.xml的用法? 注册一个监听程序涉及在web.xml的web-app元素内放|一个listener元素。在listener元素内,listener-class? 素列出监听程序的完整的限定类名,如下所C: <listener> 虽然listener元素的结构很单,但请不要忘记Q必L地l出web-app元素内的子元素的ơ序。listener元素位于所有的 servlet 元素之前以及所有filter-mapping元素之后。此外,因ؓ应用生存期监听程序是serlvet规范?.3版本中的新内容,所以必M? web.xml DTD?.3版本Q而不?.2版本? 例如Q程序清?-20l出一个名为ContextReporter的简单的监听E序Q只要Web应用的Servlet-Context建立Q如装蝲 Web应用Q或消除Q如服务器关闭)Ӟ它就在标准输Z昄一条消息。程序清?-21l出此监听程序注册所需要的web.xml文g的一部分? E序清单5-20 ContextReporterjava E序清单5-21 web.xmlQ声明一个监听程序的摘录Q?15 J2EE元素 本节描述用作J2EE环境l成部分的Web应用的web.xml元素。这里将提供一个简明的介绍Q详l内容可以参? http://java.sun.com/j2ee/j2ee-1_3-fr-spec.pdf的Java 2 Plantform Enterprise Edition版本1.3规范的第5章?l distributable distributable 元素指出QWeb应用是以q样的方式编E的Q即Q支持集的服务器可安全地在多个服务器上分布Web应用。例如,一个可分布的应用必d使用 Serializable对象作ؓ其HttpSession对象的属性,而且必须避免用实例变量(字段Q来实现持箋性。distributable元素? 接出现在discription元素之后Qƈ且不包含子元素或数据Q它只是一个如下的标志?l resource-env-ref resource -env-ref元素声明一个与某个资源有关的管理对象。此元素׃个可选的description元素、一个resource-env-ref- name元素Q一个相对于java:comp/env环境的JNDI名)以及一个resource-env-type元素Q指定资源类型的完全限定? c)Q如下所C: l env-entry env -entry元素声明Web应用的环境项。它׃个可选的description元素、一个env-entry-name元素Q一个相对于java: comp/env环境JNDI名)、一个env-entry-value元素Q项|以及一个env-entry-type元素Qjava.langE序 包中一个类型的完全限定cdQjava.lang.Boolean、java.lang.String{)l成。下面是一个例子: l ejb-ref ejb -ref元素声明对一个EJB的主目录的应用。它׃个可选的description元素、一个ejb-ref-name元素Q相对于java: comp/env的EJB应用Q、一个ejb-ref-type元素Qbean的类型,Entity或SessionQ、一个home元素Qbean的主 目录接口的完全限定名Q、一个remote元素Qbean的远E接口的完全限定名)以及一个可选的ejb-link元素Q当前bean链接的另一? bean的名Uͼl成?l ejb-local-ref ejb-local-ref元素声明一个EJB的本C目录的引用。除了用local-home代替home外,此元素具有与ejb-ref元素相同的属 性ƈ以相同的方式使用?/p>



二胡 2009-04-24 17:17 发表评论
]]>
? web.xml配置详细说明(?http://www.tkk7.com/hulizhong/archive/2009/04/24/267403.html二胡二胡Fri, 24 Apr 2009 09:15:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/04/24/267403.htmlhttp://www.tkk7.com/hulizhong/comments/267403.htmlhttp://www.tkk7.com/hulizhong/archive/2009/04/24/267403.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/267403.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/267403.html
3.2 定义定制的URL
大多数服务器h一个缺省的serlvet URLQ?
http://host /webAppPrefix/servlet/packageName.ServletName。虽然在开发中使用q个URL很方便,但是我们常常会希? 另一个URL用于部v。例如,可能会需要一个出现在Web应用层的URLQ如Qhttp: //host/webAppPrefix/AnynameQ,q且在此URL中没有servletV位于顶层的URL化了相对URL的用。此外,? 许多开发h员来_层URL看上L更长更麻烦的~省URL更简短?
事实上,有时需要用定制的URL。比如,你可能想关闭~省URL映射Q以便更好地强制实施安全限制或防止用h外地讉K无初始化参数的servlet。如果你止了缺省的URLQ那么你怎样讉Kservlet呢?q时只有使用定制的URL了?
? 了分配一个定制的URLQ可使用servlet-mapping元素及其servlet-name和url-pattern子元素。Servlet- name元素提供了一个Q意名Uͼ可利用此名称引用相应的servletQurl-pattern描述了相对于Web应用的根目录的URL。url- pattern元素的值必M斜杠Q?Qv始?
下面l出一个简单的web.xml摘录Q它允许使用URL http://host/webAppPrefix/UrlTest而不是http://host/webAppPrefix/servlet/Test?
http: //host/webAppPrefix/servlet/moreservlets.TestServlet。请注意Q仍焉要XML头? DOCTYPE声明以及web-app闭元素。此外,可回忆一下,XML元素出现地次序不是随意的。特别是Q需要把所有servlet元素攑֜所? servlet-mapping元素之前?
Java代码
  1. <servlet>   
  2. <servlet-name>Test</servlet-name>   
  3. <servlet-class>moreservlets.TestServlet</servlet-class>   
  4. </servlet>   
  5. <!-- ... -->   
  6. <servlet-mapping>   
  7. <servlet-name>Test</servlet-name>   
  8. <url-pattern>/UrlTest</url-pattern>   
  9. </servlet-mapping>  

URL模式q可以包含通配W。例如,下面的小E序指示服务器发送所有以Web应用的URL前缀开始,?.aspl束的请求到名ؓBashMS的servlet?/p>

Java代码
  1. <servlet>   
  2. <servlet-name>BashMS</servlet-name>   
  3. <servlet-class>msUtils.ASPTranslator</servlet-class>   
  4. </servlet>   
  5. <!-- ... -->   
  6. <servlet-mapping>   
  7. <servlet-name>BashMS</servlet-name>   
  8. <url-pattern>/*.asp</url-pattern>   
  9. </servlet-mapping>  


3.3 命名JSP面
因ؓJSP面要{换成sevletQ自然希望就像命名servlet一样命名JSP面。毕竟,JSP 面可能会从初始化参数、安全设|或定制的URL中受益,正如普通的serlvet那样。虽然JSP面的后台实际上是servletq句话是正确的,? 存在一个关键的猜疑Q即Q你不知道JSP面的实际类名(因ؓpȝ自己挑选这个名字)。因此,Z命名JSP面Q可jsp-file元素替换? servlet-calss元素Q如下所C:

Java代码
  1. <servlet>   
  2. <servlet-name>Test</servlet-name>   
  3. <jsp-file>/TestPage.jsp</jsp-file>   
  4. </servlet>  

命名JSP面的原因与命名servlet的原因完全相同:即ؓ了提供一个与定制讄Q如Q初始化参数和安全设|)一起用的名称Qƈ且,以便能更 Ҏz? JSP面的URLQ比方说Q以便多个URL通过相同面得以处理Q或者从URL中去?jsp扩展名)。但是,在设|初始化参数Ӟ应该注意QJSP? 面是利用jspInitҎQ而不是initҎd初始化参数的?
例如Q程序清?-3l出一个名为TestPage.jsp的简单JSP 面Q它的工作只是打印出用来Ȁzd的URL的本地部分。TestPage.jsp攄在deployDemo应用的顶层。程序清?-4l出了用来分? 一个注册名PageNameQ然后将此注册名与http://host/webAppPrefix/UrlTest2/anything 形式的URL相关联的web.xml文gQ即QdeployDemo/WEB-INF/web.xmlQ的一部分?

E序清单5-3 TestPage.jsp

Java代码
  1. <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">   
  2. <HTML>   
  3. <HEAD>   
  4. <TITLE>   
  5. JSP Test Page   
  6. </TITLE>   
  7. </HEAD>   
  8. <BODY BGCOLOR="#FDF5E6">   
  9. <H2>URI: <%= request.getRequestURI() %></H2>   
  10. </BODY>   
  11. </HTML>  



E序清单5-4 web.xmlQ说明JSP命名的摘录Q?/p>

Java代码
  1. <?xml version="1.0" encoding="ISO-8859-1"?>   
  2. <!DOCTYPE web-app   
  3. PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"  
  4. "http://java.sun.com/dtd/web-app_2_3.dtd">   
  5.   
  6. <web-app>   
  7. <!-- ... -->   
  8. <servlet>   
  9. <servlet-name>PageName</servlet-name>   
  10. <jsp-file>/TestPage.jsp</jsp-file>   
  11. </servlet>   
  12. <!-- ... -->   
  13. <servlet-mapping>   
  14. <servlet-name> PageName </servlet-name>   
  15. <url-pattern>/UrlTest2/*</url-pattern>   
  16. </servlet-mapping>   
  17. <!-- ... -->   
  18. </web-app>  



4 止Ȁzdservlet

对servlet或JSP面建立定制URL的一个原因是Q这样做可以注册? initQservletQ或jspInitQJSP面Q方法中d得初始化参数。但是,初始化参数只在是利用定制URL模式或注册名讉K servlet或JSP面时可以用,用缺省URL http://host/webAppPrefix/servlet/ServletName 讉K时不能用。因此,你可能会希望关闭~省URLQ这样就不会有h意外地调用初始化servlet了。这个过E有时称为禁止激zdservletQ因? 多数服务器具有一个用~省的servlet URL注册的标准servletQƈȀzȝ省的URL应用的实际servlet?
有两U禁止此~省URL的主要方法:
l 在每个Web应用中重新映?servlet/模式?
l 全局关闭Ȁzdservlet?
? 要的是应该注意到Q虽焉新映每个Web应用中的/servlet/模式比彻底禁止激zservlet所做的工作更多Q但重新映射可以用一U完全可UL 的方式来完成。相反,全局止Ȁzdservlet完全是针对具体机器的Q事实上有的服务器(如ServletExecQ没有这L选择。下面的讨论Ҏ 个Web应用重新映射/servlet/ URL模式的策略。后面提供在Tomcat中全局止Ȁzdservlet的详l内宏V?
4.1 重新映射/servlet/URL模式
? 一个特定的Web应用中禁止以http://host/webAppPrefix/servlet/ 开始的URL的处理非常简单。所需做的事情是建立一个错误消息servletQƈ使用前一节讨论的url-pattern元素所有匹配请求{向该 servlet。只要简单地使用Q?
<url-pattern>/servlet/*</url-pattern>
作ؓservlet-mapping元素中的模式卛_?
例如Q程序清?-5l出了将SorryServlet servletQ程序清?-6Q与所有以http://host/webAppPrefix/servlet/ 开头的URL相关联的部v描述W文件的一部分?

E序清单5-5 web.xmlQ说明JSP命名的摘录Q?/p>

Java代码
  1. <?xml version="1.0" encoding="ISO-8859-1"?>   
  2. <!DOCTYPE web-app   
  3. PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"  
  4. "http://java.sun.com/dtd/web-app_2_3.dtd">   
  5.   
  6. <web-app>   
  7. <!-- ... -->   
  8. <servlet>   
  9. <servlet-name>Sorry</servlet-name>   
  10. <servlet-class>moreservlets.SorryServlet</servlet-class>   
  11. </servlet>   
  12. <!-- ... -->   
  13. <servlet-mapping>   
  14. <servlet-name> Sorry </servlet-name>   
  15. <url-pattern>/servlet/*</url-pattern>   
  16. </servlet-mapping>   
  17. <!-- ... -->   
  18. </web-app>  



E序清单5-6 SorryServlet.java

Java代码
  1. package moreservlets;   
  2.   
  3. import java.io.*;   
  4. import javax.servlet.*;   
  5. import javax.servlet.http.*;   
  6.   
  7. /** Simple servlet used to give error messages to
  8. * users who try to access default servlet URLs
  9. * (i.e., http://host/webAppPrefix/servlet/ServletName)
  10. * in Web applications that have disabled this
  11. * behavior.
  12. * <P>
  13. * Taken from More Servlets and JavaServer Pages
  14. * from Prentice Hall and Sun Microsystems Press,
  15. * http://www.moreservlets.com/.
  16. * © 2002 Marty Hall; may be freely used or adapted.
  17. */  
  18.   
  19. public class SorryServlet extends HttpServlet {   
  20. public void doGet(HttpServletRequest request,   
  21. HttpServletResponse response)   
  22. throws ServletException, IOException {   
  23. response.setContentType("text/html");   
  24. PrintWriter out = response.getWriter();   
  25. String title = "Invoker Servlet Disabled.";   
  26. out.println(ServletUtilities.headWithTitle(title) +   
  27. "<BODY BGCOLOR=\"#FDF5E6\">\n" +   
  28. "<H2>" + title + "</H2>\n" +   
  29. "Sorry, access to servlets by means of\n" +   
  30. "URLs that begin with\n" +   
  31. "http://host/webAppPrefix/servlet/\n" +   
  32. "has been disabled.\n" +   
  33. "</BODY></HTML>");   
  34. }   
  35.   
  36. public void doPost(HttpServletRequest request,   
  37. HttpServletResponse response)   
  38. throws ServletException, IOException {   
  39. doGet(request, response);   
  40. }   
  41. }  


4.2 全局止ȀzdQTomcat
Tomcat 4中用来关闭缺省URL的方法与Tomcat 3中所用的很不相同。下面介l这两种ҎQ?
1Q禁止激zdQ?Tomcat 4
Tomcat 4用与前面相同的方法关闭激zdservletQ即利用web.xml中的url-mapping元素q行关闭。不同之处在于Tomcat使用了放? install_dir/conf中的一个服务器专用的全局web.xml文gQ而前面用的是存攑֜每个Web应用的WEB-INF目录中的标准 web.xml文g?
因此Qؓ了在Tomcat 4中关闭激zdservletQ只需在install_dir/conf/web.xml中简单地注释?servlet/* URL映射即可,如下所C:

Java代码
  1. <!--   
  2. <servlet-mapping>   
  3. <servlet-name>invoker</servlet-name>   
  4. <url-pattern>/servlet/*</url-pattern>   
  5. </servlet-mapping>   
  6. -->  

再次提醒Q应该注意这个项是位于存攑֜install_dir/conf的Tomcat专用的web.xml文g中的Q此文g不是存放在每个Web应用的WEB-INF目录中的标准web.xml?
2Q禁止激zdQTomcat3
在Apache Tomcat的版?中,通过在install_dir/conf/server.xml中注释出InvokerInterceptor全局止~省 servlet URL。例如,下面是禁止用缺省servlet URL的server.xml文g的一部分?/p>

Java代码
  1. <!--   
  2. <RequsetInterceptor   
  3. className="org.apache.tomcat.request.InvokerInterceptor"  
  4. debug="0" prefix="/servlet/" />   
  5. -->  


5 初始化和预装载servlet与JSP面

q里讨论控制servlet和JSP面的启动行为的Ҏ。特别是Q说明了怎样分配初始化参C及怎样更改服务器生存期中装载servlet和JSP面的时刅R?
5.1 分配servlet初始化参?
? 用init-param元素向servlet提供初始化参敎ͼinit-param元素hparam-name和param-value子元素。例如, 在下面的例子中,如果initServlet servlet是利用它的注册名QInitTestQ访问的Q它能够从其方法中调用getServletConfig(). getInitParameter("param1")获得"Value 1"Q调用getServletConfig().getInitParameter("param2")获得"2"?/p>

Java代码
  1. <servlet>   
  2. <servlet-name>InitTest</servlet-name>   
  3. <servlet-class>moreservlets.InitServlet</servlet-class>   
  4. <init-param>   
  5. <param-name>param1</param-name>   
  6. <param-value>value1</param-value>   
  7. </init-param>   
  8. <init-param>   
  9. <param-name>param2</param-name>   
  10. <param-value>2</param-value>   
  11. </init-param>   
  12. </servlet>  

在涉及初始化参数Ӟ有几炚w要注意:
l q回倹{GetInitParameter的返回值L一个String。因此,在前一个例子中Q可对param2使用Integer.parseInt获得一个int?
l JSP中的初始化。JSP面使用jspInit而不是init。JSP面q需要用jsp-file元素代替servlet-class?
l ~省URL。初始化参数只在通过它们的注册名或与它们注册名相关的定制URL模式讉KServlet时可以用。因此,在这个例子中Qparam1? param2初始化参数将能够在用URL http://host/webAppPrefix/servlet/InitTest 时可用,但在使用URL http://host/webAppPrefix/servlet/myPackage.InitServlet 时不能用?
例如Q程序清?-7l出一个名为InitServlet的简单servletQ它使用initҎ讄firstName和emailAddress字段。程序清?-8l出分配名称InitTestlservlet的web.xml文g?
E序清单5-7 InitServlet.java

Java代码
  1. package moreservlets;   
  2.   
  3. import java.io.*;   
  4. import javax.servlet.*;   
  5. import javax.servlet.http.*;   
  6.   
  7. /** Simple servlet used to illustrate servlet
  8. * initialization parameters.
  9. * <P>
  10. * Taken from More Servlets and JavaServer Pages
  11. * from Prentice Hall and Sun Microsystems Press,
  12. * http://www.moreservlets.com/.
  13. * © 2002 Marty Hall; may be freely used or adapted.
  14. */  
  15.   
  16. public class InitServlet extends HttpServlet {   
  17. private String firstName, emailAddress;   
  18.   
  19. public void init() {   
  20. ServletConfig config = getServletConfig();   
  21. firstName = config.getInitParameter("firstName");   
  22. emailAddress = config.getInitParameter("emailAddress");   
  23. }   
  24.   
  25. public void doGet(HttpServletRequest request,   
  26. HttpServletResponse response)   
  27. throws ServletException, IOException {   
  28. response.setContentType("text/html");   
  29. PrintWriter out = response.getWriter();   
  30. String uri = request.getRequestURI();   
  31. out.println(ServletUtilities.headWithTitle("Init Servlet") +   
  32. "<BODY BGCOLOR=\"#FDF5E6\">\n" +   
  33. "<H2>Init Parameters:</H2>\n" +   
  34. "<UL>\n" +   
  35. "<LI>First name: " + firstName + "\n" +   
  36. "<LI>Email address: " + emailAddress + "\n" +   
  37. "</UL>\n" +   
  38. "</BODY></HTML>");   
  39. }   
  40. }  


E序清单5-8 web.xmlQ说明初始化参数的摘录)

Java代码
  1. <?xml version="1.0" encoding="ISO-8859-1"?>   
  2. <!DOCTYPE web-app   
  3. PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"  
  4. "http://java.sun.com/dtd/web-app_2_3.dtd">   
  5.   
  6. <web-app>   
  7. <!-- ... -->   
  8. <servlet>   
  9. <servlet-name>InitTest</servlet-name>   
  10. <servlet-class>moreservlets.InitServlet</servlet-class>   
  11. <init-param>   
  12. <param-name>firstName</param-name>   
  13. <param-value>Larry</param-value>   
  14. </init-param>   
  15. <init-param>   
  16. <param-name>emailAddress</param-name>   
  17. <param-value>Ellison@Microsoft.com</param-value>   




二胡 2009-04-24 17:15 发表评论
]]>
? web.xml配置详细说明(?http://www.tkk7.com/hulizhong/archive/2009/04/24/267402.html二胡二胡Fri, 24 Apr 2009 09:13:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/04/24/267402.htmlhttp://www.tkk7.com/hulizhong/comments/267402.htmlhttp://www.tkk7.com/hulizhong/archive/2009/04/24/267402.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/267402.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/267402.html

转自http://elevenet.javaeye.com/blog/67862

q篇文章q是不错的,有些老,有些地址q是直接译的。不是很准确?/strong>

关键? J2EE

1 定义头和根元?

部v描述W文件就像所有XML文g一P必须以一个XML头开始。这个头声明可以使用的XML版本q给出文件的字符~码?
DOCYTPE声明必须立即出现在此头之后。这个声明告诉服务器适用的servlet规范的版本(?.2?.3Qƈ指定理此文件其余部分内容的语法的DTD(Document Type DefinitionQ文档类型定??
所有部|描q符文g的顶层(根)元素为web-app。请注意QXML元素不像HTMLQ他们是大小写敏感的。因此,web-App和WEB-APP都是不合法的Qweb-app必须用小写?

2 部v描述W文件内的元素次?

XML 元素不仅是大写敏感的,而且它们q对出现在其他元素中的次序敏感。例如,XML头必L文g中的W一,DOCTYPE声明必须是第二项Q而web- app元素必须是第三项。在web-app元素内,元素的次序也很重要。服务器不一定强制要求这U次序,但它们允许(实际上有些服务器是q样做的Q完? 拒绝执行含有ơ序不正的元素的Web应用。这表示使用非标准元素次序的web.xml文g是不可移植的?
下面的列表给Z所有可直接出现在web-app元素内的合法元素所必需的次序。例如,此列表说明servlet元素必须出现在所有servlet-mapping元素之前。请注意Q所有这些元素都是可选的。因此,可以省略掉某一元素Q但不能把它放于不正的位置?
l icon                          icon 元素指出IDE和GUI工具用来表示Web应用的一个和两个囑փ文g的位|?
l display-name           display-name元素提供GUI工具可能会用来标记这个特定的Web应用的一个名U?
l description               description元素l出与此有关的说明性文本?
l context-param          context-param元素声明应用范围内的初始化参数?
l filter                         qo器元素将一个名字与一个实现javax.servlet.Filter接口的类相关联?
l filter-mapping          一旦命名了一个过滤器Q就要利用filter-mapping元素把它与一个或多个servlet或JSP 面相关联?
l listener                    servlet API的版?.3增加了对事g监听E序的支持,事g监听E序在徏立、修改和删除会话或servlet环境时得到通知。Listener元素指出事g监听E序cR?
l servlet                    在向servlet或JSP面制定初始化参数或定制URLӞ必须首先命名servlet或JSP面。Servlet元素是用来完成此项d的?
l servlet-mapping      服务器一般ؓservlet提供一个缺省的URLQ?a href="http://host/webAppPrefix/servlet/ServletName">http://host/webAppPrefix/servlet/ServletName。但是,常常会更改这个URLQ以便servlet可以讉K初始化参数或更容易地处理相对URL。在更改~省URLӞ使用servlet-mapping元素?
l session-config         如果某个会话在一定时间内未被讉KQ服务器可以抛弃它以节省内存。可通过使用HttpSession的setMaxInactiveIntervalҎ 明确讄单个会话对象的超时|或者可利用session-config元素制定~省时倹{?
l mime-mapping         如果Web应用h惛_Ҏ的文Ӟ希望能保证给他们分配特定的MIMEcdQ则mime-mapping元素提供q种保证?
l welcom-file-list         welcome-file-list元素指示服务器在收到引用一个目录名而不是文件名的URLӞ使用哪个文g?
l error-page               error-page元素使得在返回特定HTTP状态代码时Q或者特定类型的异常被抛出时Q能够制定将要显C的面?
l taglib                        taglib元素Ҏ记库描述W文ӞTag Libraryu Descriptor fileQ指定别名。此功能使你能够更改TLD文g的位|,而不用编辑用这些文件的JSP面?
l resource-env-ref      resource-env-ref元素声明与资源相关的一个管理对象?
l resource-ref             resource-ref元素声明一个资源工厂用的外部资源?
l security-constraint    security-constraint元素制定应该保护的URL。它与login-config元素联合使用
l login-config 用login-config元素来指定服务器应该怎样l试图访问受保护面的用h权。它与sercurity-constraint元素联合使用?
l security-role              security-role元素l出安全角色的一个列表,q些角色出现在servlet元素内的security-role-ref元素的role- name子元素中。分别地声明角色可高IDE处理安全信息更ؓҎ?
l env-entry                  env-entry元素声明Web应用的环境项?
l ejb-ref                      ejb-ref元素声明一个EJB的主目录的引用?
l ejb-local-ref              ejb-local-ref元素声明一个EJB的本C目录的应用?

3 分配名称和定制的UL

在web.xml中完成的一个最常见的Q务是对servlet或JSP面l出名称和定制的URL。用servlet元素分配名称Q用servlet-mapping元素定制的URL与刚分配的名U相兌?
3.1 分配名称
? 了提供初始化参数Q对servlet或JSP面定义一个定制URL或分配一个安全角Ԍ必须首先lservlet或JSP面一个名U。可通过 servlet元素分配一个名U。最常见的格式包括servlet-name和servlet-class子元素(在web-app元素内)Q如下所C:
Java代码
  1. <servlet>   
  2. <servlet-name>Test</servlet-name>   
  3. <servlet-class>moreservlets.TestServlet</servlet-class>   
  4. </servlet>   
q? 表示位于WEB-INF/classes/moreservlets/TestServlet的servlet已经得到了注册名Test。给 servlet一个名U具有两个主要的含义。首先,初始化参数、定制的URL模式以及其他定制通过此注册名而不是类名引用此servlet。其?可在 URL而不是类名中使用此名U。因此,利用刚才l出的定义,URL http://host/webAppPrefix/servlet/Test 可用?http://host/webAppPrefix/servlet/moreservlets.TestServlet 的场所?
? CQXML元素不仅是大写敏感的,而且定义它们的次序也很重要。例如,web-app元素内所有servlet元素必须位于所有servlet- mapping元素Q下一节介绍Q之前,而且q要位于5.6节和5.11节讨论的与过滤器或文档相关的元素Q如果有的话Q之前。类似地Qservlet 的servlet-name子元素也必须出现在servlet-class之前?.2?部v描述W文件内的元素次?详l介l这U必需的次序?
? 如,E序清单5-1l出了一个名为TestServlet的简单servletQ它ȝ在moreservletsE序包中。因为此servlet是扎? 在一个名为deployDemo的目录中的Web应用的组成部分,所以TestServlet.class攑֜deployDemo/WEB- INF/classes/moreservlets中。程序清?-2l出放|在deployDemo/WEB-INF/内的web.xml文g的一? 分。此web.xml文g使用servlet-name和servlet-class元素名UTest与TestServlet.class相关联。图 5-1和图5-2分别昄利用~省URL和注册名调用TestServlet时的l果?

E序清单5-1 TestServlet.java
Java代码
  1. package moreservlets;   
  2.   
  3. import java.io.*;   
  4. import javax.servlet.*;   
  5. import javax.servlet.http.*;   
  6.   
  7. /** Simple servlet used to illustrate servlet naming
  8. * and custom URLs.
  9. * <P>
  10. * Taken from More Servlets and JavaServer Pages
  11. * from Prentice Hall and Sun Microsystems Press,
  12. * http://www.moreservlets.com/.
  13. * © 2002 Marty Hall; may be freely used or adapted.
  14. */  
  15.   
  16. public class TestServlet extends HttpServlet {   
  17. public void doGet(HttpServletRequest request,   
  18. HttpServletResponse response)   
  19. throws ServletException, IOException {   
  20. response.setContentType("text/html");   
  21. PrintWriter out = response.getWriter();   
  22. String uri = request.getRequestURI();   
  23. out.println(ServletUtilities.headWithTitle("Test Servlet") +   
  24. "<BODY BGCOLOR=\"#FDF5E6\">\n" +   
  25. "<H2>URI: " + uri + "</H2>\n" +   
  26. "</BODY></HTML>");   
  27. }   
  28. }  

E序清单5-2 web.xmlQ说明servlet名称的摘录)
Java代码
  1. <?xml version="1.0" encoding="ISO-8859-1"?>   
  2. <!DOCTYPE web-app   
  3. PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"  
  4. "http://java.sun.com/dtd/web-app_2_3.dtd">   
  5.   
  6. <web-app>   
  7. <!-- … -->   
  8. <servlet>   
  9. <servlet-name>Test</servlet-name>   
  10. <servlet-class>moreservlets.TestServlet</servlet-class>   
  11. </servlet>   
  12. <!-- … -->   
  13. </web-app>  




二胡 2009-04-24 17:13 发表评论
]]>
?Java Web中的入R及单实?/title><link>http://www.tkk7.com/hulizhong/archive/2009/04/17/266178.html</link><dc:creator>二胡</dc:creator><author>二胡</author><pubDate>Fri, 17 Apr 2009 08:42:00 GMT</pubDate><guid>http://www.tkk7.com/hulizhong/archive/2009/04/17/266178.html</guid><wfw:comment>http://www.tkk7.com/hulizhong/comments/266178.html</wfw:comment><comments>http://www.tkk7.com/hulizhong/archive/2009/04/17/266178.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.tkk7.com/hulizhong/comments/commentRss/266178.html</wfw:commentRss><trackback:ping>http://www.tkk7.com/hulizhong/services/trackbacks/266178.html</trackback:ping><description><![CDATA[?http://blog.csdn.net/zhaowei001/archive/2007/12/29/2001800.aspx<br /> <br /> 作者:<a >EasyJF开源团?/a> 大<br /> <br /> <strong>一、简?/strong><br /> <br /> 在Java Web应用E中Q特别是|站开发中Q我们有时候需要ؓ应用E序增加一个入侉|程序来防止恶意h的功能,防止非法用户不断的往Web应用中重复发送数 据。当Ӟ入R可以用很多Ҏ实现Q包括Y件、硬仉火墙Q入侉|的{略也很多。在q里我们主要介绍的是Java Web应用E序中通过软g的方式实现简单的入R及防M? <p>  该方法的实现原理很简单,是用户讉KWebpȝ时记录每个用L信息Q然后进行对照,q根据设定的{略(比如Q?U钟h面10?判断用户是否属于恶意h?/p> <p> 我们的入侉|程序应该放到所有Java WebE序的执行前Q也卌发现用户是恶意刷新就不再l箋执行Java Web中的其它部分内容Q否则就会失M意义。这需要以插g的方式把入R的E序|入Java Web应用中,使得每次用户讉KJava WebQ都先要到这个入侉|程序中报一ơ到Q符合规则才能放行?/p> <p>  Java Web应用大致分ؓ两种Q一U纯JSP(+Java Bean)方式Q一U是Z框架(如Struts、EasyJWeb{?的。第一U方式的Java Web可以通过Java Servlet中的Filter接口实现Q也卛_C个Filter接口Q在其doFilterҎ中插入入侉|程序,然后再web.xml中作单的 配置卛_。在Z框架的Web应用中,׃所有应用都有一个入口,因此可以把入侉|的E序直接插入框架入口引擎中,使框架本w支持入侉|功能。当? 也可以通过实现Filter接口来实现?/p> <p>  在EasyJWeb框架中,已经|入了简单入侉|的E序Q因此,q里我们以EasyJWeb框架Z,介绍具体的实现方法及源码Q完整的代码可以在EasyJWeb源码中找到?/p> <p>  在基于EasyJWeb的Java Web应用??a >http://www.easyjf.com/bbs/</a>)Q默认情况下你只要连l刷新页面次数过多,即会弹出如下的错误:</p> <p>  EasyJWeb框架友情提示!:-): <br /> <font color="#ff0000">  您对面的刷新太?L?0U后再刷新页面! <br /> 详细h?a target="_blank">http://www.easyjf.com</a> </font></p> <p><br /> <strong>二、用戯问信息记录UserConnect.javac?/strong>  <br /> <br /> q个cL一个简单的Java BeanQ主要代表用L信息Q包括用户名、IP、第一ơ访问时间、最后登录时间、登录次数、用L态等。全?br /> <br /> 代码如下Q?br /> <br /> package com.easyjf.web;</p> <p>import java.util.Date;<br /> /**<br /> *<br /> * </p> <p>Title:用户验证信息</p> <br /> * <p>Description:记录用户d信息,判断用户d情况</p> <br /> * <p>Copyright: Copyright (c) 2006</p> <br /> * <p>Company: <a >www.easyjf.com</a></p> ><br /> * @author 蔡世?br /> * @version 1.0<br /> */<br /> public class UserConnect {<br /> private String userName;<br /> private String ip;<br /> private Date firstFailureTime;<br /> private Date lastLoginTime;<br /> private int failureTimes;//用户dp|ơ数<br /> private int status=0;//用户状?表示正常,-1表示锁定<br /> public int getFailureTimes() {<br />  return failureTimes;<br /> }<br /> public void setFailureTimes(int failureTimes) {<br />  this.failureTimes = failureTimes;<br /> }<br /> public Date getFirstFailureTime() {<br />  return firstFailureTime;<br /> } <p>public void setFirstFailureTime(Date firstFailureTime) {<br />  this.firstFailureTime = firstFailureTime;<br /> }</p> <p>public String getIp() {<br />  return ip;<br /> }</p> <p>public void setIp(String ip) {<br />  this.ip = ip;<br /> }</p> <p>public Date getLastLoginTime() {<br />  return lastLoginTime;<br /> }</p> <p>public void setLastLoginTime(Date lastLoginTime) {<br />  this.lastLoginTime = lastLoginTime;<br /> }</p> <p>public String getUserName() {<br />  return userName;<br /> }</p> <p>public void setUserName(String userName) {<br />  this.userName = userName;<br /> }</p> <p>public int getStatus() {<br />  return status;<br /> }</p> <p>public void setStatus(int status) {<br />  this.status = status;<br /> }</p> <p>}</p> <p><br /> <strong>三、监控线EUserConnectManage.javac?/strong><br /> <br /> q是入R的核心部分Q主要实现具体的入R、记录、判断用户信息、在U用Lh{功能,q提供其它应用程序用本lg的调用接口?br /> <br /> package com.easyjf.web;</p> <p>import java.util.Date;<br /> import java.util.HashMap;<br /> import java.util.HashSet;<br /> import java.util.Iterator;<br /> import java.util.Map;<br /> import java.util.Set;</p> <p>import org.apache.log4j.Logger;<br /> /**<br /> *<br /> * </p> <p>Title:用户入R信?/p> <br /> * <p>Description:用于判断用户h情况查,默认?0U钟之内q箋q接10ơؓ时</p> <br /> * <p>Copyright: Copyright (c) 2006</p> <br /> * <p>Company: <a >www.easyjf.com</a></p> ><br /> * @author 蔡世?br /> * @version 1.0<br /> */<br /> public class UserConnectManage {<br /> private static final Logger logger = (Logger) Logger.getLogger(UserConnectManage.class.getName());<br /> private static int maxFailureTimes=10;//最大登录失败次?br /> private static long maxFailureInterval=10000;//毫秒Q达到最大登录次C在这个时间范围内<br /> private static long waitInterval=60000;//p|后接受连接的{待旉Q默?分钟<br /> private static int maxOnlineUser=200;//同时在线的最大数<br /> private final static Map users=new HashMap();//使用ip+userName为key存放用户d信息UserLoginAuth<br /> private static Thread checkThread=null;<br /> private static class CheckTimeOut implements Runnable{ <br /> private Thread parentThread;<br /> public  CheckTimeOut(Thread parentThread) <br />  {<br />   this.parentThread=parentThread; <br />   synchronized(this){<br />   if(checkThread==null){    <br />    checkThread= new Thread(this);<br />    //System.out.println("创徏一个新U程Q?);<br />    checkThread.start();   <br />      }<br />   }<br />  } <br />  public void run() {  <br />   while(true)<br />   {<br />    if(parentThread.isAlive()){<br />    try{<br />    Thread.sleep(2000);<br />    int i=0;<br />    if(users.size()>maxOnlineUser)//当达到最大用h时清?br />    {<br />     synchronized(users){//执行删除操作<br />     Iterator it=users.keySet().iterator();<br />     Set set=new HashSet();<br />     Date now=new Date();<br />     while(it.hasNext())<br />     {<br />      Object key=it.next();<br />      UserConnect user=(UserConnect)users.get(key);<br />      if(now.getTime()-user.getFirstFailureTime().getTime()>maxFailureInterval)//删除时的用?br />      {      <br />       set.add(key);<br />       logger.info("删除了一个超时的q接"+i);<br />       i++;<br />      }<br />     }<br />     if(i<5)//如果删除于5个,则强行删?/2在线记录Q牺牲性能的情况下保证内存<br />     {<br />      int num=maxOnlineUser/2;<br />      it=users.keySet().iterator();<br />      while(it.hasNext() && i<br />      {      <br />       set.add(it.next());<br />       logger.info("删除了一个多余的q接"+i);<br />       i++;<br />      }<br />     }<br />     users.keySet().removeAll(set);<br />     }<br />    }<br />    <br />    }<br />    catch(Exception e)<br />    {<br />     e.printStackTrace();<br />    }<br />    <br />   }<br />    else<br />    {   <br />    break;<br />    }<br />   }<br />   logger.info("监视E序q行l束Q?); <br />  }<br /> }<br /> //通过checkLoginValidate判断是否合法的登录连接,如果合法则l,非法则执?br /> public static boolean checkLoginValidate(String ip,String userName)//只检查认证失败次?br /> {<br />  boolean ret=true;<br />  Date now=new Date(); <br />  String key=ip+":"+userName;<br />  UserConnect auth=(UserConnect)users.get(key);<br />  if(auth==null)//把用户当前的讉K信息加入到users容器?br />  {<br />   auth=new UserConnect();<br />   auth.setIp(ip);<br />   auth.setUserName(userName);<br />   auth.setFailureTimes(0);<br />   auth.setFirstFailureTime(now);<br />   users.put(key,auth);  <br />   if(checkThread==null)new CheckTimeOut(Thread.currentThread());<br />  } <br />  else<br />  {<br />   if(auth.getFailureTimes()>maxFailureTimes)<br />   {<br />            //如果在限定的旉间隔内,则返回拒l用戯接的信息<br />    if((now.getTime()-auth.getFirstFailureTime().getTime())<br />    {<br />     ret=false;<br />     auth.setStatus(-1);<br />    }<br />    else  if(auth.getStatus()==-1 && (now.getTime()-auth.getFirstFailureTime().getTime()< (maxFailureInterval+waitInterval)))//重置计数?br />    {<br />     ret=false;<br />    }<br />    else    <br />    {    <br />     auth.setFailureTimes(0);<br />     auth.setFirstFailureTime(now);<br />     auth.setStatus(0);<br />    }<br />    <br />   }<br />   //dơ数?<br />   auth.setFailureTimes(auth.getFailureTimes()+1);<br />  }<br />  //System.out.println(key+":"+auth.getFailureTimes()+":"+ret+":"+(now.getTime()-auth.getFirstFailureTime().getTime()));<br />  return ret;<br /> } <p>public static void reset(String ip,String userName)//重置用户信息<br /> { <br />  Date now=new Date(); <br />  String key=ip+":"+userName;<br />  UserConnect auth=(UserConnect)users.get(key);<br />  if(auth==null)//把用户当前的讉K信息加入到users容器?br />  {<br />   auth=new UserConnect();<br />   auth.setIp(ip);<br />   auth.setUserName(userName);<br />   auth.setFailureTimes(0);<br />   auth.setFirstFailureTime(now);<br />   users.put(key,auth);<br />  } <br />  else<br />  {<br />   auth.setFailureTimes(0);<br />   auth.setFirstFailureTime(now);<br />  }<br /> }<br /> public static void remove(String ip,String userName)//删除用户在容器中的记?br /> {<br />  String key=ip+":"+userName;<br />  users.remove(key);<br /> }<br /> public static void clear()//清空容器中内?br /> {<br />  if(!users.isEmpty())users.clear();<br /> }<br /> public static long getMaxFailureInterval() {<br />  return maxFailureInterval;<br /> }</p> <p>public static void setMaxFailureInterval(long maxFailureInterval) {<br />  UserConnectManage.maxFailureInterval = maxFailureInterval;<br /> }</p> <p>public static int getMaxFailureTimes() {<br />  return maxFailureTimes;<br /> }</p> <p>public static void setMaxFailureTimes(int maxFailureTimes) {<br />  UserConnectManage.maxFailureTimes = maxFailureTimes;<br /> }</p> <p>public static int getMaxOnlineUser() {<br />  return maxOnlineUser;<br /> }</p> <p>public static void setMaxOnlineUser(int maxOnlineUser) {<br />  UserConnectManage.maxOnlineUser = maxOnlineUser;<br /> }</p> <p>public static long getWaitInterval() {<br />  return waitInterval;<br /> }</p> <p>public static void setWaitInterval(long waitInterval) {<br />  UserConnectManage.waitInterval = waitInterval;<br /> }</p> <p><br /> <strong>四、调用接?/strong><br /> <br /> 在需要进入R判断的地方Q直接用UserConnectManagecM的checkLoginValidateҎ卛_。如EasyJWeb的核心Servlet <br /> <br /> com.easyjf.web.ActionServlet中调用UserConnectManage的代码:<br />    if(!UserConnectManage.checkLoginValidate(request.getRemoteAddr(),"guest"))<br />        {            <br />            info(request,response,new Exception("您对面的刷新太?L?+UserConnectManage.getWaitInterval()/1000+"U?br /> <br /> 后再h面Q?));<br />            return;<br />        }      <br /> <br /> <br /> <br /> <strong>五、ȝ</strong><br /> 当然Q这里提供的Ҏ只是一个简单的实现CZQ由于上面的用户信息是直接保存在内存中,若ƈ发用户很大的时候的代码的占用,可以考虑引入数据库来记录? L讉K信息Q当然相应的执行效率肯定用降低。上面介l的实现中,入R判断的{略也只有用戯问次数及旉间隔两个元素Q您q可以根据你的实现情况增 加其它的元素?br /> <br /> ׃水^有限Q该设计上有N不合理或者需要改q的地方Q恳请大家指正!</p> <br /> <br /> <p id="TBPingURL">Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=657199</p> <br /> <br /><img src ="http://www.tkk7.com/hulizhong/aggbug/266178.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/hulizhong/" target="_blank">二胡</a> 2009-04-17 16:42 <a href="http://www.tkk7.com/hulizhong/archive/2009/04/17/266178.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>?JSONP的v?/title><link>http://www.tkk7.com/hulizhong/archive/2009/03/24/261764.html</link><dc:creator>二胡</dc:creator><author>二胡</author><pubDate>Tue, 24 Mar 2009 13:34:00 GMT</pubDate><guid>http://www.tkk7.com/hulizhong/archive/2009/03/24/261764.html</guid><wfw:comment>http://www.tkk7.com/hulizhong/comments/261764.html</wfw:comment><comments>http://www.tkk7.com/hulizhong/archive/2009/03/24/261764.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.tkk7.com/hulizhong/comments/commentRss/261764.html</wfw:commentRss><trackback:ping>http://www.tkk7.com/hulizhong/services/trackbacks/261764.html</trackback:ping><description><![CDATA[?http://www.cn-cuckoo.com/2008/09/13/the-origin-of-jsonp-262.html<br /> <br /> <p>览器安全模型规定,XMLHttpRequest、框ӞframeQ等只能在一个域中通信。从安全角度考虑Q这个规定很合理Q但是,也确实给分布式(面向服务、؜搭等{本周提到的概念QWeb开发带来了ȝ?/p> <p>Z实现跨域通信Q通常的解x案有3U:</p> <p><strong>本地代理Q?/strong><br /> 需要一些硬件设施(没有服务器的客户端无法运行)Qƈ且带宽和潜伏旉也要加倍(q程服务器-代理服务器-客户端)?/p> <p><strong> FlashQ?/strong><br /> q程L中需要部|一个crossdomain.xml文gQ而且QFlash作ؓ一门专有技术,其前途尚不明朗;换句话说Q开发h员很可能要学习一U目标不定的编E语a?/p> <p><strong>Script标签Q?/strong><br /> 无法切知道内容是否有效Q没有标准的实现ҎQ又可能被认为是一U?#8220;安全风险”?/p> <p></p> <hr /> 在此Q我使用一U新技术,也是一U独立于标准的方法,即通过script标签来跨域获取数据,名ؓJSON with PaddingQ或者就?strong>JSONP</strong>。JSONP的原理很单,但需要服务器端给予相应配合。大致来_JSONP的实现思\是在客L~程时作好用JSON数据的准备,然后再通过圆括号将q些数据括v来以创徏一条有效的JavaScript语句Q可能是一ơ有效的函数调用Q? <p>也就是说Q客L可以使用一个用于命名jsonp的查询参数来军_可以获取的数据。最单的情况下,如果jsonp参数为空Q则q回的数据就是被括在圆括号中的JSON?/p> <p>下面Q我们就?a title="http://del.icio.us/help/json" target="_blank">del.icio.us的JSON API</a>ZQ来说明JSONP的原理。该API有一?#8220;script tag”变量Q即Q可以将下面的URL作ؓscript标签的src属性|用以加蝲del.icio.usq个API提供的数据。——译者注Q如下所C:</p> <p>http://del.icio.us/feeds/json/bob/mochikit+interpreter:</p> <div id="bvn99hr" class="hl-surround"> <ol class="hl-main ln-show" title="双击隐藏行号" ondblclick="linenumber(this)"> <li id="39xrjjb" class="hl-firstline">if(typeof(Delicious) == 'undefined') Delicious = {};</li> <li>Delicious.posts = [{</li> <li> "u": "http://mochikit.com/examples/interpreter/index.html",</li> <li> "d": "Interpreter - JavaScript Interactive Interpreter",</li> <li> "t": [</li> <li> "mochikit","webdev","tool","tools",</li> <li> "javascript","interactive","interpreter","repl"</li> <li> ]</li> <li>}]</li> </ol> </div> <p>如果用JSONP的方式来表示Q那么与此具有相同语义的URL应该是这LQ?/p> <p>http://del.icio.us/feeds/json/bob/mochikit+interpreter?<br /> jsonp=if(typeof(Delicious)%3D%3D%27undefined%27)<br /> Delicious%3D%7B%7D%3BDelicious.posts%3D</p> <p>单纯看这个URLg没有什么,但我们可以要求服务器在数据有效时l出通知。因此,我可以编写一个用于跟t数据的系l:</p> <div id="hvpbtl7" class="hl-surround"> <ol class="hl-main ln-show" title="双击隐藏行号" ondblclick="linenumber(this)"> <li id="vtnnhxj" class="hl-firstline">var delicious_callbacks = {};</li> <li>function getDelicious(callback, url) {</li> <li> var uid = (new Date()).getTime();</li> <li> delicious_callbacks[uid] = function () {</li> <li> delete delicious_callbacks[uid];</li> <li> callback();</li> <li> };</li> <li> url += "?jsonp=" + encodeURIComponent("delicious_callbacks[" + uid + "]");</li> <li> // 手工输入代码Q向文档中插入script标签</li> <li>};</li> <li> </li> <li>getDelicious(doSomething, "http://del.icio.us/feeds/json/bob/mochikit+interpreter");</li> </ol> </div> <p>Ҏ以上假设Q用于获取数据的URL应该如下所C:<br /> http://del.icio.us/feeds/json/bob/mochikit+interpreter?jsonp=delicious_callbacks%5B12345%5D</p> <div id="pljvpph" class="hl-surround"> <ol class="hl-main ln-show" title="双击隐藏行号" ondblclick="linenumber(this)"> <li id="pnfztd7" class="hl-firstline">delicious_callbacks[12345]([{</li> <li> "u": "http://mochikit.com/examples/interpreter/index.html",</li> <li> "d": "Interpreter - JavaScript Interactive Interpreter",</li> <li> "t": [</li> <li> "mochikit","webdev","tool","tools",</li> <li> "javascript","interactive","interpreter","repl"</li> <li> ]</li> <li>}])</li> </ol> </div> <p>可见Q由于用圆括号括住了返回的数据Q这q当于把一个JSONPh转化成了一ơ函数调用,或者得C一个纯_的JSON直接量。服务器所要配合做的,是在JSON数据的开头添加一段文本Q即回调函数的名U。——译者注QƈJSON数据攑֜括号中!</p> <p>当然Q接下来最好是使用Mochikit、Dojo{框架来抽象JSONPQ从而让自己省去动手~写DOM以插入script标签的麻烦?/p> <p>没错QJSONP只是解决了标准化的问题。假如远E主机想通过script标签向页面中注入恶意代码Q而不是返回JSON数据Q那么页面安全可能会 随时受到威胁。不q,一旦实CJSONPQ那么对开发h员来说肯定是一件省时省力的大好事,在此基础上各U一般化的抽象、教E及文档也会应运而生的?/p> <p>注:~写?JSONP ?Bob Ippolito 在一名?“<a title="http://bob.pythonmac.org/archives/2005/12/05/remote-json-jsonp/" target="_blank">Remote JSON - JSONP</a>” 的文章中提出。但许多支持?JSONP 技术实现跨域通信的厂商没有称其ؓ JSONP。例如,雅虎公司qq种技术ؓ “JSON with callbacks”。另外,原文发表?005q?2?日?/p> <div id="f7fnhdl" class="post-info"> 九月 13th, 2008 in <a title="查看 Web开?的全部文? rel="category tag">Web开?/a> </div> <h1 class="comments-title">留言(9)</h1> <div id="1rj9pjz" class="comment" id="comment-2185"> <div id="193jdll" class="comment-avatar"> <img alt="" src="http://www.gravatar.com/avatar/be62fdd065e143bb3b693e32b795b875?s=50&d=http%3A%2F%2Fwww.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D50&r=G" class="avatar avatar-50 photo" width="50" height="50" /> </div> <div id="vvlvvfz" class="comment-content"> <div id="fnldp7d" class="comment-info"><span><a rel="external nofollow" class="url">hello cuckoo</a></span>九月 16th, 2008 at 9:42 上午 </div> <p>中秋好?br /> 怎么把我的链接去掉了呢?<br /> l我做个链接可以吗?Q?<br /> 非常感谢Q!<br /> 我的EMAILQhardcometure@163.com</p> </div> </div> <div id="533zrl9" class="comment" id="comment-2186"> <div id="3333jx9" class="comment-avatar"> <img alt="" src="http://www.gravatar.com/avatar/23fd4332db6748087c7d157ec30c6257?s=50&d=http%3A%2F%2Fwww.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D50&r=G" class="avatar avatar-50 photo" width="50" height="50" /> </div> <div id="9v9rb9j" class="comment-content"> <div id="5xx9dnv" class="comment-info"><span>vampire</span>九月 16th, 2008 at 8:41 下午 </div> <p>大概是说通过script的src来实现跨域,通过l编码的json在url中传递数据?<br /> 是不是说面上的js生带有不同参数的jsonpQ通过src传递给服务端,服务端根据该jsonpq回相应数据Q有点不太明白,能否提供一个实例?谢谢</p> </div> </div> <div id="x7llxx9" class="comment" id="comment-2188"> <div id="hzj59r9" class="comment-avatar"> <img alt="" src="http://www.gravatar.com/avatar/f225dfc5180b6ae93d054de8c06b08dc?s=50&d=http%3A%2F%2Fwww.gravatar.com%2Favatar%2Fad516503a11cd5ca435acc9bb6523536%3Fs%3D50&r=G" class="avatar avatar-50 photo" width="50" height="50" /> </div> <div id="3bjvppp" class="comment-content"> <div id="999b3r9" class="comment-info"><span><a rel="external nofollow" class="url">admin</a></span>九月 17th, 2008 at 8:43 下午 </div> <p>@vampire</p> <p>你的理解是对的,看来我的译你能看明白,呵呵。用JSONP技术时Q一般是由JS在客L面中动态插入script标签Q将其src属性设|? 为带参数的URL。当面加蝲Ӟ前述URL会将参数通过GETh发送到相应服务器端应用E序Q由URL表示Q,服务器根据参数返回数据——注意,q个 数据格式是JSONQƈ且(注意Q被包含在一个函数调用中Q例如:callback({json_data})。这P在客L面中存在预定义? callback(data)函数的定义时Q服务器q回的函数调用就会立x行,由客L的函数对数据q行操作?/p> <p>实际的例子有很多Q如Yahoo Search、Google Base、Flickr Search、Amazon Search{等。要学习和用这些站Ҏ供的支持JSONP的APIQ一般要注意两点Q一是有的站点要求注册密钥(使用时必L到参CQ,二是要注? 参数的用方法。例如,有的API要求使用预定义的回调函数Q而有的API则允怋用者自己定义回调函数?/p> <p>下面是向Flickr SearchhJSONP响应的URLQ其中用了预定义的回调函数jsonFlickrApiQ参C不必l出Q:</p> <p><a rel="nofollow">http://api.flickr.com/services/rest/</a><br /> ?method=flickr.photos.search<br /> &api_key=85618ad7d326d8ef93c6bee9ed32706f<br /> &per_page=5&format=json&text=china</p> <p>下面q个URL发送到Google BaseQ它允许开发h员用自己定义的回调函数Q?/p> <p><a rel="nofollow">http://www.google.com/base/feeds/snippets</a><br /> ?q=jquery<br /> &alt=json-in-script<br /> &callback=customCallback</p> <p>把前面的URL攑ֈ览器地址栏中Q回车,卛_看到l果?/p> </div> </div> <br /> <br /><img src ="http://www.tkk7.com/hulizhong/aggbug/261764.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/hulizhong/" target="_blank">二胡</a> 2009-03-24 21:34 <a href="http://www.tkk7.com/hulizhong/archive/2009/03/24/261764.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>cachehttp://www.tkk7.com/hulizhong/archive/2009/03/22/261319.html二胡二胡Sun, 22 Mar 2009 07:55:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/03/22/261319.htmlhttp://www.tkk7.com/hulizhong/comments/261319.htmlhttp://www.tkk7.com/hulizhong/archive/2009/03/22/261319.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/261319.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/261319.html阅读全文

二胡 2009-03-22 15:55 发表评论
]]>
?REST反模?/title><link>http://www.tkk7.com/hulizhong/archive/2009/03/22/261316.html</link><dc:creator>二胡</dc:creator><author>二胡</author><pubDate>Sun, 22 Mar 2009 07:36:00 GMT</pubDate><guid>http://www.tkk7.com/hulizhong/archive/2009/03/22/261316.html</guid><wfw:comment>http://www.tkk7.com/hulizhong/comments/261316.html</wfw:comment><comments>http://www.tkk7.com/hulizhong/archive/2009/03/22/261316.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.tkk7.com/hulizhong/comments/commentRss/261316.html</wfw:commentRss><trackback:ping>http://www.tkk7.com/hulizhong/services/trackbacks/261316.html</trackback:ping><description><![CDATA[?http://www.infoq.com/cn/articles/rest-anti-patterns<br /> <br />    Z在试验RESTӞ通常会四处寻找样例——而他们往往不仅能找C大堆自称“W合REST”或标榜ؓ“REST API”的样例,q会发现许多关于某个自称W合REST的特定服务名不副实的讨论?nbsp; <br />    <br /> <p>Z么会q样QHTTP虽不是什么新事物Q但Z使用它的方式却五花八门。其中有些做法符合Web设计者的初衷Q但许多q如此。要Z的HTTP 应用Q无论是面向人类、还是计机、或同时面向q两者用的Q应用REST原则Q意味着你要恰好反过来:量“正确?#8221;使用WebQ或者说按符合REST 的方式用WebQ倘若你不喜欢用对或错来评判的话)。对许多人来_q的是一U崭新的方式Ҏ?/p> <p>我经常在文章里作同样的声明:REST、Web和HTTP是不同的事物QREST可以用多U不同技术来实现Q而HTTP只是一U恰好符合REST? 构风格的具体架构。所以,其实我应该小心区?#8220;REST”?#8220;REST式HTTP”q两个概늚。但我没有这么做Q在本文剩余部分Q我们姑且认为它们是? 同的事物?/p> <p>跟Q何新的方式方法一P发掘一些共同的模式是有益的。在本系列的<a >W一</a>?a >W二</a>? 文章中,我已l讲qC一些基——比如集合资源的概念、将计算l果转换源本w、以及用聚合QsyndicationQ来模仿事g。后l文章将q一步讲 q这些及其他模式。不q在本文中,我想主要说说反模式(anti-patternsQ——即那些力求W合REST式HTTP、但未能成功而造成问题的典? 做法?/p> <p>首先我们来看看我发掘了哪些反模式Q?/p> <ol> <li>全部采用GET</li> <li>全部采用POST</li> <li>忽视~存</li> <li>忽视响应代码</li> <li>误用cookies</li> <li>忘记媒?/li> <li>忽视MIMEcd</li> <li>破坏自描q?/li> </ol> <p>下面我们来逐个详细说明?</p> <h2>全部采用GET</h2> <p>在许多h看来QREST仅仅意味着用HTTP暴露一些应用功能。HTTP GET是最重要的基本操作(operation Q(严格地讲Q用“动词QverbQ?#8221;?#8220;ҎQmethodQ?#8221;q样的术语比较好Q。GETҎ应当用于获取由URI标识的资源的一个表C? QrepresentationQ,而许多(即便谈不上所有)现有的HTTP库和服务器编EAPI不是URI视ؓ一U资源标识符Qresource identifierQ,而是之视ؓ一U传递参数的便利手段。这D了以下这UURIs的出玎ͼ</p> <pre><code>http://example.com/some-api?method=deleteCustomer&id=1234<br /> </code></pre> <p>实际上,你无法根据构成URI的字W获知关于给定系l的“REST性(RESTfulnessQ?#8221;的Q何信息,不过对于上面那个URIQ我们可以判 断该 GET操作不是“安全的(safeQ?#8221;——也是_调用者很可能要ؓl果Q删除一个客P负责Q尽规范里说在q种情况下用GETҎ是错误的?/p> <p>q种做法唯一有利的方面在于它~程hҎQ而且在浏览器中调试也单——你只要把URI_脓到浏览器地址栏里、然后调整一?#8220;参数”p了。这U反模式主要存在以下问题Q?</p> <ol> <li>URI没有被用作资源标识符Q而是被用于传递操作及其参C?/li> <li>HTTPҎQHTTP methodQ不一定跟语义相符?/li> <li>q种链接一般不可加入书{?/li> <li>?#8220;爬虫”造成非预期副作用的风险?/li> </ol> <p>注意Q符合这一反模式的APIs没准最l?a title="Web Things, by Mark Baker » » Accidentally RESTful">yW合REST原则</a>。这里有个例子:</p> <pre><code>http://example.com/some-api?method=findCustomer&id=1234<br /> </code></pre> <p>q个URI是标识操作及其参数呢Q还是标识一个资源呢Q两U情况都有可能:它可以是一个完全合法的、可加入书签的URIQ对它做GET操作也许?#8220; 安全? ”Q它也许会根据Accept报头q回不同的格式,q支持复杂的~存机制。在很多情况下,q将是偶然的。APIl常在刚开始时采用q种方式来暴露一?#8220;? ”接口Q但当开发者要增添“?#8221;功能时就有问题了Q因Z无法通过对上qURI做PUT操作来更C个客户——开发者得构造另一个URIQ?/p> <h2>全部采用POST</h2> <p>q一反模式跟前一个颇为相|只不q这里用的是POSTҎ而已。POST除了携带一个URIQ还携带一个实体主体(entity bodyQ。一个典型的场景是:单个URI作ؓPOSTh的目标、通过发送不同的消息来表达不同的意图。实际上QSOAP 1.1 Web服务是q样做的Q它把HTTP当作一U?#8220;传输协议”来用。服务器ҎSOAP消息Q可能还包括一些WS-Addressing SOAP报头Q决定做什么?/p> <p>可能有h认ؓ“全部采用POST”?#8220;全部采用GET”存在的问题完全一P只是它更隄一些,而且不能利用~存Q甚臌偶尔的机会都没有Q,且无法支持书{。事实上Q它q不是违反了哪条REST原则Q而是Ҏ忽视了REST原则?</p> <h2>忽视~存</h2> <p>即你按各个动词的原本意图来使用它们Q你仍可以轻易禁止缓存机制。最单的做法是在你的HTTP响应里增加这样一个报_ </p> <pre><code>Cache-control: no-cache<br /> </code></pre> <p>q样可以止~存机制发挥作用。当Ӟq也许正是你惌做的Q然而通常q只是你的Web框架规定的一个缺省设|。不q,寚w效的~存与再验证 Qcaching and re-validationQ的支持Q是采用REST式HTTP的诸多关键优点之一。Sam Ruby表示Q在判断是否W合REST原则时的一个关键问题就?#8220;<a title="InfoQ: Interview and Book Excerpt: RESTful Web Services">你支持ETag?/a>”Q? QETag是HTTP 1.1里引入的一U机Ӟ它允许客L通过加密的校验和来验证一个被~存的表C是否仍然有效)。要生成正确的报_最单的做法是把这个Q务交l一?#8220; 知道”怎样做的基础设施——例如通过在Web服务器(比如Apache HTTPDQ的目录里生成一个文件?/p> <p>当然Q这也要涉及到客L一方:你在Z个REST式服务实现程序客LӞ你应充分利用现有的缓存机Ӟ以免每次都重新获取表C。例如,服务器也 许已l发Z息:初次q回的表C在600U内都可被认为是“新的”Q比方说因ؓ后端pȝ?0分钟才轮询一ơ)。这L话,短时间内重复h同一信息完 全没必要了。在客户端设|一个代理缓存(比如SquidQ也许比自行构徏相应逻辑更好?/p> <p>HTTP的缓存机制强大而复杂;<a title="Caching Tutorial for Web Authors and Webmasters">Mark Nottingham的《缓存指南(Cache TutorialQ?/a>是一个很好的指南?</p> <h2>忽视响应代码</h2> <p>HTTP提供了一l丰富的<a title="HTTP/1.1: Status Code Definitions">应用U状态代?/a>Q? 它们可用于应付不同场合,不过许多Web开发者对此ƈ不知晓。大部分人对200Q?#8220;OK”Q?04Q?#8220;Not found”Q和500Q?#8220;Internal server error”Q这些状态代码是比较熟悉的。但除此以外q有很多其他状态代码,正确使用q些状态代码意味着客户端与服务器可以在一个具备较丰富语义的层ơ上 q行沟通?/p> <p>例如Q?01Q?#8220;Created”Q响应代码表明已l创Z一个新的资源,其URI在Location响应报头里?09Q?#8220;Conflict”Q? 告诉客户端存在冲H,比如随PUTh发送的是基于老版本资源的数据。再如,412Q?#8220;Precondition Failed”Q表明服务器不能满客户端的预期?/p> <p>正确使用状态代码的另一斚w涉及客户端:应该Ҏ一U统一的MҎ对不同类别的状态代码(例如所?xxD代码、所?xxD代码)作不同处理——例如,即便客户端不具备处理特定代码的逻辑Q但臛_应把所?xxD代码视为成功信受?/p> <p>许多声称W合REST的应用仅仅返?00?00Q甚臛_q回200Qƈ在响应实体主体里l出错误文本——SOAP是q样的)。你要是愿意Q可 以称之ؓ“通过状态代?00传达错误”Q但无论你觉得采用哪个术语好Q假如你不利用HTTP状态代码丰富的应用语义Q那么你错失提高重用性、增Z? 作性和提升松耦合性的Z?/p> <h2>误用cookies</h2> <p id="scroll_to_here">利用cookies来传播某个服务端会话状态的键(keyQ是另一UREST反模式?</p> <p>Cookies表明肯定哪个地方不符合REST了。是q样吗?不;不一定。REST的关键思想之一是无状态性(statelessnessQ——不 是说一个服务器不能保存M数据Q倘若是资源状态(resource stateQ或客户端状态(client stateQ,那是可以的。服务器不能保存的是会话状态(session stateQ,因ؓ那会造成可׾~性、可靠性及耦合斚w的问题。Cookies的最典型的用法是Q保存一个跟“某个保存在服务端内存里的数据l构”相关? 的键QkeyQ。这意味着Q浏览器随各ơ请求发出去的cookie是被用于构徏会话状态的?/p> <p>如果一个cookie被用于保存一?#8220;服务器不依赖于会话状态即可验?#8221;的信息(比如认证令牌Q,那么q样的cookies是完全符合REST原则 的—? 不过有一炚w要注意:如果有其他更为标准的方式来传递一则信息(比如攑֜URI里、放在某个标准报头里、或较少见地攑֜消息M里)Q那׃应将之放? cookie里。例如,按REST式HTTP的观Ҏ使用HTTP认证比较好?/p> <h2>忘记媒?/h2> <p>最不易接受的REST思想是标准的方法集合。REST理论q没有规定标准集合由哪些Ҏl成Q它只是规定必须有一l适用于所有资源的Ҏ集合。对 ? HTTP来说Q这l集合是GET、PUT、POST和DELETEQ至v初是q样Q,你需要一定适应旉才能掌握如何所有应用语义投到q四个动? 上。但你一旦适应了,可以开始运用这个REST的子集——一U基于Web的CRUDQCreate、Read、Update? DeleteQ架构——了。暴露这U反模式的应用不是真正的“非REST?#8221;应用Q假如存在这U事物的话)Q它们只是未能利用一个REST核心概念—?#8220; 媒体即应用状态引擎(hypermedia as the engine of application stateQ?#8221;?/p> <p>媒体(hypermediaQ是一个把事物链接h的概念,正是它造就了Webq个|——一个互联的资源集合Q应用通过跟随链接从一个状态进入另一个状态。这听上M许有Ҏ奥,不过其实遵从q一原则是有正当理由的?</p> <p>“忘记媒?#8221;反模式的首要表现是Q表C(representationQ里~少链接。尽通常客户端可以根据一定的规则来构造URIQ但是因? 服务器没有发送Q何链接,所以客L无法跟随链接。一U较好的做法是:x持构造URIQ又支持跟随链接——这里的链接通常反映了下层数据模型中的关 pR但最好的情况是:客户端应该只需知道一个URIQ其他URIQ各个URI及其构造模式,如:各种查询字符Ԍ应该通过媒体(作ؓ资源表示里的链接Q? 来传达?Atom发布协议QAtom Publishing ProtocolQ就是一个好例子Q它有一个服务文档(service documentsQ的概念Q服务文档ؓ它所描述的域内的各个集合提供具名元素Qnamed elementsQ。最后,应用可能l历的状态迁Ud该是动态传播的Q客L应该可以不用掌握多少知识可以跟随它们。HTML是一个好榜样Q它包含? 够的信息Q以便浏览器可以向用h供一个完全动态的接口?/p> <p>我本惛_加一?#8220;人类可读的URI”反模式的。但我没那么做,因ؓ我跟其他Z样也喜欢可读的、好“改”的URI。但是当Z采用RESTӞ? 们经常浪费许多时间来讨论“正确?#8221;URI设计Q而忘C媒体方面。所以,我徏议你不要花太多时间来L正确的URI设计Q毕竟,它们只是字符串? ԌQ而是多花一些精力在表示里寻找提供链接的正确地方?/p> <h2>忽视MIMEcd</h2> <p>HTTP有个内容协商Qcontent negotiationQ的概念Q它允许客户端根据需要获取资源的不同表示QrepresentationsQ。例如,一个资源也许有不同格式的表C(? XML、JSON或YAML{)以便于用各种不同语言Q如Java、JavaScript及RubyQ实现的消费者所使用。再如,一个资源可能即有面向h cȝPDF或JPEG版表C,又有“机器可读?#8221;XML版表C。还有,一个资源可能同时支持v1.1版和v1.2版的自定义表C格式。不怎样Q也许可? ?#8220;只有一个表C格?#8221;扑ֈ理由Q但q常常意味着丢掉某种Z?/p> <p>昄Q若一个服务能为更多未预见到的客户端所用(或重用)那更好。因此,依靠现有、预定义、广Zh知的格式Q要好过发明U有格式——这会导致本文讲q的最后一个反模式?/p> <h2>破坏自描q?/h2> <p>q种反模式是如此普遍Q以至于几乎在每个、甚至那些由所谓的“REST狂热者们”Q包括我在内Q创建的REST应用里都可以看到Q违反自描述性约? Q这一努力目标q不像h们最初想象的那样跟h工智能科qd说有多大牵连Q。理x况下Q一个消息(HTTP h或HTTP响应Q包括报头与MQ应该包含够信息,以便M通用客户端、服务器或媒介(intermediaryQ能够处理它。例如,当你的浏览器 获取某个受保护资源的PDF表示QrepresentationQ时Q你可以看到由标准达成的协定是如何v作用的:有些HTTP认证交换发生Q可能会发生 一些缓存(cachingQ和/或再验证QrevalidationQ,服务器发送的content-type报头Q?a title="RFC 3778: The application/pdf Media Type">application/pdf</a>Q触发了你系l里注册的PDF阅读器,最后你得以在自q屏幕上阅读该PDF。所有用户都可以用他们自q基础设施来执行同Lh。若服务器开发者另外增加一U内容类型,那么服务器的客户端(或服务的消费者)只需保他们安装了正的阅读器即可?/p> <p>你要是发明自q报头、格式或协议Q那׃定程度上破坏了自描述性约束。极端地Ԍ所有没有被某个标准化组l官Ҏ准化的东襉Kq反此约束,因而可 被认为符合本反模式。在实践中,你应努力做到可能遵循标准,q懂?#8220;某些协定可能只在一个较的领域Q比方说Q你的服务和客户端是专门针对它开发的Q中 适用” 的道理?/p> <h2>ȝ</h2> <p>自从“四hl(Gang of FourQ?#8221;出版?a title="Design Patterns - Wikipedia, the free encyclopedia">书籍</a>? 掀h式运动的开端以来,许多解了它,q试囑֜可能多的场合下应用模式——这已被其他人所取笑。模式应当仅在符合上下文时才被应用。同样地Q可能有 Z不遗余力地在所有场合下虔诚地努力避免所有反模式。许多时候,你有充分理由q反某一规则Q或者按REST的术语放松某一U束。这么做是没问题的——但 了解实际情况、作出知情决{是有益的?/p> <p>但愿本文能有助于你在开始首个REST目旉免落入这些常见的陷阱?</p> <p>非常感谢Javier Botana和Burkhard NeppertҎ文初E的反馈?</p> <p><a >Stefan Tilkov</a>是InfoQ SOAC֌的首席编辑,以及位于德国/瑞士?a >innoQ</a>公司的合伙h、首席顾问和主要的REST狂热M者?/p> <br /><img src ="http://www.tkk7.com/hulizhong/aggbug/261316.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/hulizhong/" target="_blank">二胡</a> 2009-03-22 15:36 <a href="http://www.tkk7.com/hulizhong/archive/2009/03/22/261316.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>?深入出RESThttp://www.tkk7.com/hulizhong/archive/2009/03/22/261311.html二胡二胡Sun, 22 Mar 2009 06:30:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/03/22/261311.htmlhttp://www.tkk7.com/hulizhong/comments/261311.htmlhttp://www.tkk7.com/hulizhong/archive/2009/03/22/261311.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/261311.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/261311.html
    不知你是否意识到Q围l着什么才是实现异构的应用到应用通信?#8220;正确”方式Q一Z论正q行的如火如|虽然当前L的方式明昑֜集中在基于SOAP? WSDL和WS-*规范的Web Services领域Q但也有数人用l小但洪亮的声音d说更好的方式是RESTQ表q性状态{U(REpresentational State TransferQ的U。在本文中,我不会涉及争论的话题Q而是试对REST和RESTful HTTP应用集成做实用性的介绍。以我的l验Q有些话题一旦触及就会引来众多的讨论Q当涉及到这斚w话题的时候,我会深入详细地阐q?br />    

REST关键原则

大部分对REST的介l是以其正式的定义和背景作ؓ开场的。但q儿且先按下不表Q我先提Z个简单扼要的定义QREST定义了应该如何正地使用 Q这和大多数人的实际使用方式有很大不同)Web标准Q例如HTTP和URI。如果你在设计应用程序时能坚持REST原则Q那预C着你将会得C个? 了优质Web架构Q这让你受益)的系l。MQ五条关键原则列丑֦下:

  • 为所?#8220;事物”定义ID
  • 所有事物链接在一?
  • 使用标准Ҏ
  • 资源多重表述
  • 无状态通信

下面让我们进一步审视这些原则?/p>

为所?#8220;事物”定义ID

在这里我使用?#8220;事物”来代替更正式准确的术?#8220;资源”Q因Z条如此简单的原则Q不应该被没在术语当中。思考一下h们构建的pȝQ通常会找C pd值得被标识的关键抽象。每个事物都应该是可标识的,都应该拥有一个明昄ID——在Web中,代表ID的统一概念是:URI。URI构成了一个全局? 名空_使用URI标识你的关键资源意味着它们获得了一个唯一、全局的ID?

对事物用一致的命名规则Qnaming schemeQ最主要的好处就是你不需要提q规则——而是依靠某个已被定义Q在全球范围中几乎完运行,q且能被l大多数人所理解的规则。想一下你 构徏的上一个应用(假设它不是采用RESTful方式构徏的)中的L一个高U对象(high-level objectQ,那就很有可能看到许多从用唯一标识中受益的用例。比如,如果你的应用中包含一个对֮的抽象,那么我可以相当肯定,用户会希望将一个指 向某个顾客的链接Q能通过电子邮g发送到同事那里Q或者加入到览器的书签中,甚至写到U怸。更透彻地讲Q如果在一个类gAmazon.com的在U商 城中Q没有用唯一的IDQ一个URIQ标识它的每一件商品,可想而知q将是多么可怕的业务决策?/p>

当面对这个原则时Q许多h惊讶于这是否意味着需要直接向外界暴露数据库记录(或者数据库记录IDQ——自从多q以来面向对象的实践告诫我们Q要持 久化的信息作为实现细节隐藏v来之后,哪怕是刚有Ҏ法都怼使h惊恐。但是这条原则与隐藏实现l节两者之间ƈ没有M冲突Q通常Q值得被URI标识的事 物——资源——要比数据库记录抽象的多。例如,一个定单资源可以由定单V地址以及许多其它斚wQ可能不希望作ؓ单独标识的资源暴露出来)l成。标识所? 值得标识的事物,领会q个观念可以q一步引g创造出在传l的应用E序设计中不常见的资源:一个流E或者流E步骤、一ơ销售、一ơ谈判、一份报仯求—? q都是应该被标识的事物的CZ。同Pq也会导致创建比非RESTful设计更多的持久化实体?

下面是一些你可能惛_的URI的例子:

http://example.com/customers/1234
http://example.com/orders/2007/10/776654
http://example.com/products/4554
http://example.com/processes/salary-increase-234

正如我选择了创Z于阅ȝURI——这是个有用的观点,管不是RESTful设计所必须的——应该能十分Ҏ地推出URI的含义:它们明显地标识着单一“数据?#8221;。但是再往下看Q?/p>

http://example.com/orders/2007/11
http://example.com/products?color=green

首先Q这两个URI看v来与之前的稍有不同——毕竟,它们不是对一件事物的标识Q而是对一cM物集合的标识Q假定第一个URI标识了所有在2007q?1月䆾提交的定单,W二个则是绿颜色产品的集合)。但是这些集合自w也是事物(资源Q,也应该被标识?/p>

注意Q用唯一、全局l一的命名规则的好处Q既适用于浏览器中的Web应用Q也适用于机ҎQmachine-to-machineQm2mQ通信?/p>

来对W一个原则做下ȝQ用URI标识所有值得标识的事物,特别是应用中提供的所?#8220;高”资源Q无些资源代表单一数据V数据项集合、虚拟亦或实际的对象q是计算l果{?/p>

所有事物链接在一?

接下来要讨论的原则有一个有点o人害怕的正式描述Q?#8220;媒体被当作应用状态引擎(Hypermedia as the engine of application stateQ?#8221;Q有时简写ؓHATEOAS。(严格地说Q这不是我说的。)q个描述的核心是媒?/strong>概念Q换句话_?strong>链接的思想。链接是我们在HTML中常见的概念Q但是它的用处绝不局限于此(用于Z|络览Q。考虑一下下面这个虚构的XML片段Q?

 
23


如果你观察文档中product和customer的链接,可以很Ҏ地想象到Q应用程序(已经索过文档Q如?#8220;跟随”链接索更多的信息。当Ӟ如果使用一个遵守专用命名规范的?#8220;id”属性作为链接,也是可行的—?strong>但是仅限于应用环境之?/strong>。用URI表示链接的优雅之处在于,链接可以指向׃同应用、不同服务器甚至位于另一个大陆上的不同公司提供的资源——因为URI命名规范是全球标准,构成Web的所有资源都可以互联互通?/p>

媒体原则还有一个更重要的方面——应?#8220;状?#8221;。简而言之,实际上服务器端(如果你愿意,也可以叫服务提供者)为客LQ服务消费者)提供一l链 接,使客L能通过链接应用从一个状态改变ؓ另一个状态。稍后我们会在另一文章中探究q个斚w的媄响;目前Q只需要记住:链接是构成动态应用的非常? 效的方式?/p>

Ҏ原则ȝ如下QQ何可能的情况下,使用链接指引可以被标识的事物Q资源)。也正是链接造就了现在的Web?/p>

使用标准Ҏ

在前两个原则的讨Z暗含着一个假设:接收URI的应用程序可以通过URI明确?strong>?/strong>一些有意义的事情。如果你在公共汽车上看到一个URIQ你可以它输入览器的地址栏中q回车——但是你的浏览器如何知道需要对q个URI做些什么呢Q?/p>

它知道如何去处理URI的原因在于所有的资源都支持同L接口Q一套同LҎQ只要你乐意Q也可以UCؓ操作Q集合。在HTTP中这被叫做动? QverbQ,除了两个大家熟知的(GET和POSTQ之外,标准Ҏ集合中还包含PUT、DELETE、HEAD和OPTIONS。这些方法的含义q同 行ؓ许诺都一起定义在HTTP规范之中。如果你是一名OO开发h员,可以想象到RESTful HTTPҎ中的所有资源都l承自类gq样的一个类Q采用类Java、C#的伪语法描述Q请注意关键的方法)Q?/p>

class Resource {
Resource(URI u);
Response get();
Response post(Request r);
Response put(Request r);
Response delete();
}

׃所有资源用了同样的接口,你可以依此用GETҎ索一?strong>表述QrepresentationQ——也 是对资源的描述。因范中定义了GET的语义,所以可以肯定当你调用它的时候不需要对后果负责——这是Z么可?#8220;安全”地调用此Ҏ。GETҎ 支持非常高效、成熟的~存Q所以在很多情况下,你甚至不需要向服务器发送请求。还可以肯定的是QGETҎhq等?/strong>[? 注:指多个相同请求返回相同的l果]——如果你发送了一个GETh没有得到l果Q你可能不知道原因是h未能到达目的圎ͼq是响应在反馈的途中丢失了? q等性保证了你可以简单地再发送一ơ请求解决问题。幂{性同样适用于PUTQ基本的含义?#8220;更新资源数据Q如果资源不存在的话Q则Ҏ此URI创徏一个新 的资?#8221;Q和DELETEQ你完全可以一遍又一遍地操作它,直到得出l论——删除不存在的东西没有Q何问题)Ҏ。POSTҎQ通常表示“创徏一个新? ?#8221;Q也能被用于调用?font color="#000000">?/font>q程Q因而它既不安全也不hq等性?/p>

如果你采用RESTful的方式暴露应用功能(如果你乐意,也可以称为服务功能)Q?strong>那这条原则和它的U束同样也适用于你。如果你已经习惯于另外的设计方式Q则很难L受这条原则——毕竟,你很可能认ؓ你的应用包含了超些操作表达范围的逻辑。请允许我花费一些时间来让你怿不存在这L情况?

来看下面q个单的采购Ҏ例子Q?

Sample Scenario

可以看到Q例子中定义了两个服务程序(没有包含M实现l节Q。这些服务程序的接口都是Z完成dQ正是我们讨论的 OrderManagement和CustomerManagement服务Q而定制的。如果客LE序试图使用q些服务Q那它必针对这些特定接口进? ~码——不可能在这些接口定义之前,使用客户E序L目的地和接口协作。这些接口定义了服务E序的应用协议(application protocolQ?/p>

在RESTful HTTP方式中,你将通过l成HTTP应用协议的通用接口讉K服务E序。你可能会想出像q样的方式:

Sample Scenario, done RESTfully

可以看到Q服务程序中的特定操作被映射成ؓ标准的HTTPҎ——ؓ了消除歧义,我创Z一l全新的资源?#8220;q是骗h的把?#8221;Q我听见你叫L? 不,q不是欺骗。标识一个顾客的URI上的GETҎ正好相当于getCustomerDetails操作。有人用三角形Ş象化地说明了q一点:

Knobs one can turn

把三个顶Ҏ象ؓ你可以调节的按钮。可以看到在W一U方法中Q你拥有许多操作Q许多种cȝ数据以及固定数量?#8220;实例”Q本质上和你拥有的服务程序数 量一_。在W二U方法中Q你拥有固定数量的操作,许多U类的数据和许多调用固定Ҏ的对象。它的意义在于,证明了通过q两U方式,你基本上可以表示M 你喜Ƣ的事情?/p>

Z么用标准方法如此重要?从根本上_它你的应用成ؓWeb的一部分——应用程序ؓWeb变成Internet上最成功的应用所做的贡献Q与 它添加到Web中的资源数量成比例。采用RESTful方式Q一个应用可能会向Web中添加数以百万计的客户URIQ如果采用CORBA技术ƈl持应用? 原有设计方式Q那它的贡献大抵只是一?#8220;端点QendpointQ?#8221;——就好比一个非常小的门Q仅仅允许有钥匙的hq入其中的资源域?/p>

l一接口也得所有理解HTTP应用协议的组件能与你的应用交互。通用客户E序Qgeneric clientQ就是从中受益的lg的例子,例如curl、wget、代理、缓存、HTTP服务器、网兌有Google、Yahoo!、MSN{等?

ȝ如下Qؓ使客LE序能与你的资源怺协作Q资源应该正地实现默认的应用协议(HTTPQ,也就是用标准的GET、PUT、POST和DELETEҎ?/p>

资源多重表述

到目前ؓ止我们一直忽略了一个稍微复杂的问题Q客L序如何知道该怎样处理索到的数据,比如作ؓGET或者POSTh的结果?原因是,HTTP 采取的方式是允许数据处理和操作调用之间关pdȝ。换句话_如果客户E序知道如何处理一U特定的数据格式Q那可以与所有提供这U表q格式的资源? 互。让我们再用一个例子来阐明q个观点。利用HTTP内容协商Qcontent negotiationQ,客户E序可以h一U特定格式的表述Q?/p>

GET /customers/1234 HTTP/1.1
Host: example.com
Accept: application/vnd.mycompany.customer+xml

h的结果可能是一些由公司专有的XML格式表述的客户信息。假讑֮L序发送另外一个不同的hQ就如下面这P

GET /customers/1234 HTTP/1.1
Host: example.com
Accept: text/x-vcard

l果则可能是VCard格式的客户地址。(在这里我没有展示响应的内容,在其HTTP Content-type头中应该包含着关于数据cd的元数据。)q说明ؓ什么理想的情况下,资源表述应该采用标准格式——如果客L序对HTTP应用? 议和一l数据格式都有所“了解”Q那么它可以用一U有意义的方?strong>与世界上L一个RESTful HTTP应用交互? 不幸的是Q我们不可能拿到所有东西的标准格式Q但是,或许我们可以惛_在公司或者一些合作伙伴中使用标准格式来营造一个小环境。当然以上情况不仅适用于从 服务器端到客L的数据,反之既然——倘若从客L传来的数据符合应用协议,那么服务器端可以用特定的格式处理数据Q而不d心客L的类型?

在实践中Q资源多重表q还有着其它重要的好处:如果你ؓ你的资源提供HTML和XML两种表述方式Q那q些资源不仅可以被你的应用所用,q可以被L标准Web览器所用——也是_你的应用信息可以被所有会使用Web的h获取到?/p>

资源多重表述q有另外一U用方式:你可以将应用的Web UIU_到Web API中——毕竟,API的设计通常是由UI可以提供的功能驱动的Q而UI也是通过API执行动作的。将q两个Q务合二ؓ一带来了o人惊讶的好处Q这使得 使用者和调用E序都能得到更好的Web接口?/p>

ȝQ针对不同的需求提供资源多重表q?/p>

无状态通信

无状态通信是我要讲到的最后一个原则。首先,需要着重强调的是,虽然REST包含无状态性(statelessnessQ的观念Q但qƈ不是说暴露功能的应用不能有状态—?br /> ? 实上Q在大部分情况下q会D整个做法没有M用处。REST要求状态要么被攑օ资源状态中Q要么保存在客户端上。或者换句话_服务器端不能保持除了? ơ请求之外的QQ何与光信的客L的通信状态。这样做的最直接的理由就是可伸羃性—? 如果服务器需要保持客L状态,那么大量的客L交互会严重媄响服务器的内存可用空_footprintQ。(注意Q要做到无状态通信往往需要需要一? 重新设计——不能简单地一些session状态绑~在URI上,然后宣U这个应用是RESTful。)

但除此以外,其它斚w可能昑־更ؓ重要Q无状态约束服务器的变化对客L是不可见的,因ؓ在两ơ连l的h中,客户端ƈ不依赖于同一台服务器。一 个客L从某台服务器上收C份包含链接的文档Q当它要做一些处理时Q这台服务器宕掉了,可能是硬盘坏掉而被拿去修理Q可能是软g需要升U重启——如果这 个客L讉K了从q台服务器接收的链接Q它不会察觉到后台的服务器已l改变了?/p>

理论上的REST

我承认:以上我所说的REST不是真正的RESTQ而且我可能有点过多地热衷于简单化。但因ؓ我想有一个与众不同的开场,所以没有在一开始就介绍其正式的定义和背景。现在就让我们稍微简要地介绍一下这斚w的内宏V?/p>

首先Q先前我q没有明地区分HTTP、RESTful HTTP和REST。要理解q些不同斚w之间的关p,我们要先来看看REST的历双Ӏ?/p>

Roy T. Fielding在他?a title="博士学位论文" id="ewd-">博士学位论文Q实际上你应该访问这个链接——至对于一学术论文来_它是相当易读的。此论文已被译?a title="中文" id="nev8">中文Q? 中定义了术语REST。Roy曾是许多基本Web协议的主要设计者,其中包括HTTP和URIsQƈ且他在论文中对这些协议提Z很多x。(q篇论文? 誉ؓ“REST圣经”Q这是恰当的——毕竟,是作者发明了q个术语Q所以在定义上,他写的Q何内定w被认为是权威的。)在论文中QRoy首先定义一U方? 论来谈论架构风格——高U、抽象的模式Q来表达架构Ҏ背后的核心理c每一个架构风格由一pd?strong>U束QconstraintsQ定义Ş成。架构风格的例子包括“没有风格”Q根本没有Q何约束)、管道和qo器(pipe and filterQ、客L/服务器、分布式对象以及——你猜到它了——REST?/p>

如果对你来说q些听v来都太抽象了Q那对了——REST在本质上是一个可以被许多不同技术实现的高层ơ的风格Q而且可以被实例化——通过为它的抽 象特性赋上不同的倹{比如,REST中包含资源和l一接口的概念——也是_所有资源都应该对这些相同的Ҏ作出反应。但是RESTq没有说明是哪些? 法,或者有多少Ҏ?/p>

REST风格的一?#8220;化n”便是HTTPQ以及一套相关的一套标准,比如URIQ,或者稍微抽象一些:Web架构自n。接着上面的例子,HTTP? 用HTTP动词作ؓRESTl一接口?#8220;实例”。由于Fielding是在Web已经Q或者至是大部分)“完善”了之后才定义的REST风格Q有人可? 会争Z者是不是100%的匹配。但是无论如何,整体上来说Web、HTTP和URI仅仅是REST风格的一个主要实现。不q,׃Roy FieldingxREST论文的作者,又对Web架构设计有过p的媄响,两者相g在情理之中?/p>

最后,我在前面一ơ又一ơ地使用着术语“RESTful HTTP”Q原因很单:许多使用HTTP的应用因Z些理由ƈ没有遵@REST原则Q有Z说用HTTP而不遵@REST原则q同于滥用HTTP? 当然q听h有点狂热——事实上q反RESTU束的原因通常是,仅仅因ؓ每个U束带来的设计权衡可能不适合于一些特D情c但通常Q违背RESTU束的原 因可归咎于对其好处认知的~Z。来看一个明昄反面案例Q用HTTP GET调用cM于删除对象的操作Q这q反了REST的安全约束和一般性常识(客户E序不应为此负责Q服务器端开发h员大概不是有意而ؓ之)。但在随后的? 章中Q我会提及更多这h那样的对HTTP的滥用?/p>

ȝ

本文试图对RESTQWeb架构Q背后的概念提供快速的介绍。RESTful HTTP暴露功能的方式与RPC、分布式对象以及Web Services是不相同的;要真正理解这些不同是需要一些心态的转变。不你构徏的应用是仅仅x露Web UIq是xAPI变成Web的一份子Q了解下REST的原则还是有好处的?/p>

Stefan Tilkov是InfoQ SOAC֌的首席编辑,q且是位于d国和瑞士?/strong>innoQ公司的共同创始h、首席顾问和REST狂热分子首领?/strong>

   译者简介:苑永凯,软g设计师,毕业于山东大学。主要关注领域ؓJava EE企业应用、Java EE中间件技术以及敏捷开发方法实践,微有心得。他的Blog?a >http://blog.csdn.net/ai92Q您也可以通过yuanyk[AT]gmail.com与他联系。参与InfoQ中文站内容徏设,请邮件至china-editorial[at]infoq.com?




二胡 2009-03-22 14:30 发表评论
]]>?理解REST软g架构http://www.tkk7.com/hulizhong/archive/2009/03/22/261310.html二胡二胡Sun, 22 Mar 2009 06:20:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/03/22/261310.htmlhttp://www.tkk7.com/hulizhong/comments/261310.htmlhttp://www.tkk7.com/hulizhong/archive/2009/03/22/261310.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/261310.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/261310.html

作? 骆古?/strong> 发布? 2007q??7?下午8?8?

Ruby,
Java
主题
Web框架,
REST,
架构,
Ruby on Rails
标签
AJAX,
Ruby on Rails

一 U思维方式影响了Y件行业的发展。REST软g架构是当今世界上最成功的互联网的超媒体分布式系l。它让h们真正理解我们的|络协议HTTP本来面貌。它 正在成ؓ|络服务的主技术,同时也正在改变互联网的网lY件开发的全新思维方式。AJAX技术和Rails框架把REST软g架构思想真正地在实际中很 好表现出来。今天微软也已经应用RESTq且提出把我们现有的|络变成Z?a >语义|?/a>Q这U网l将会得搜索更加智能化?

REST与HTTP协议

REST软g架构是由Roy Thomas Fielding博士?000q首ơ提出的。他为我们描l了开发基于互联网的网lY件的蓝图。REST软g架构是一个抽象的概念Q是一Uؓ了实现这一互联|的媒体分布式pȝ?strong>行动指南。利用Q何的技术都可以实现q种理念。而实现这一软g架构最著名的就是HTTP协议。通常我们把REST也写作ؓREST/HTTPQ在实际中往往把REST理解为基于HTTP的REST软g架构Q或者更q一步把REST和HTTP看作为等同的概念?

今天QHTTP是互联网上应用最q泛的计机协议。HTTP不是一个简单的q蝲数据的协议,而是一个具有丰富内늚|络软g? 协议。它不仅仅能够对于互联网资源q行唯一定位Q而且q能告诉我们对于该资源进行怎样q作。这也是REST软g架构当中最重要的两个理c而REST软g 架构理念是真正理解HTTP协议而Ş成的。有了REST软g架构理念出现Q才使得软g业避免了对HTTP协议的片面理解。只有正的理论指导Q才能避免在 软g开发的实际工作q程中少走弯路?

REST与URIQ资源定位)

REST软g架构之所以是一个超媒体pȝQ是因ؓ它可以把|络上所有资源进行唯一的定位,不管你的文g是图片、文件Wordq是视频文gQ也不管? 的文件是txt文g格式、xml文g格式q是其它文本文g格式。它利用支持HTTP的TCP/IP协议来确定互联网上的资源?

REST与CRUD原则

REST软g架构遵@了CRUD原则Q该原则告诉我们对于资源Q包括网l资源)只需要四U行为:创徏QCreateQ、获取(ReadQ、更? QUpdateQ和销毁(DELETEQ就可以完成对其操作和处理了。其实世界万物都是遵循这一规律Q生、变、见、灭。所以计机世界也不例外。这个原? 是源自于我们对于数据库表的数据操作:insertQ生Q、selectQ见Q、updateQ变Q和deleteQ灭Q,所以有时候CRUD也写作ؓ RUDIQ其中的I是insert。这四个操作是一U原子操作,即一U无法再分的操作Q通过它们可以构造复杂的操作q程Q正如数学上四则q算是数字的最 基本的运一栗?

REST与网l服?

管在Java语言世界中网l服务目前是以SOAP技术ؓ主,但是REST是是网l服务的另一选择Qƈ且是真正意义上的|络服务。基于REST? 想的|络服务不久的将来也会成为是|络服务的主技术。REST不仅仅把HTTP作ؓ自己的数据运输协议,而且也作为直接进行数据处理的工具。而当前的|? l服务技术都需要用其它手D|完成数据处理工作Q它们完全独立于HTTP协议来进行的Q这样增加了大量的复杂Y件架构设计工作。REST的思想充分利用 了现有的HTTP技术的|络能力。在德国电视C曄出现q一个这L五十万欧元智力题Q如何实现网l服务才能充分利用现有的HTTP协议Q该问题l出? 四个{案Q去问微软;WSDL2.0/SOAP1.2QWS-TransferQ根本没有。这个问题告诉我们HTTPq不是一个简单的数据传来传去的协 议,而是一个聪明的会表现自q协议Q这也许是REST = Representational State Transfer的真正含义?

实际上目前很多大公司已经采用了REST技术作为网l服务,如Google、Amazon{。在Java语言中重要的两个以SOAP技术开始的|络服务框架XFire和Axis也把REST作ؓ自己的另一U选择。它们的新的目分别?a >Apache CXF ?a >Axis2 。Java语言也制定关于REST|络服务规范QJAX-RS: Java API for RESTful Web Services (JSR 311)。相信还会出现更多与REST相关的激动h心的信息?

REST与AJAX技?

管AJAX技术的出现才不Cq时_但是AJAX技术遵循了REST的一些重要原则。AJAX技术充分利用了HTTP来获取网l资源ƈ且实C HTTP没有的对于异步数据进行传输的功能。AJAX技术还使得软g更好地实现分布性功能,在一个企业内只要一个h下蝲了AJAX引擎Q其它企业内部的? 员,可以共享该资源了。AJAX技术遵守REST准则的应用程序中单和可׾~的架构Q凡是采用AJAX技术的面z而又丰富Q一个页面表C丰富? 彩的形态?

AJAX技术还使用了一U不同于XML格式的JSON文g格式Q这个意义在哪里呢?在REST软g架构下我们不能对于XML文gq行序列化处理,q? L序员必须要用自qXMLl定框架。而以序列化的JavaScript对象为基的JSON已经获得了广泛认可,它被认ؓ能以q比XML更好的方? 来序列化和传输简单数据结构,而且它更z。这对REST是一个极大A献和补充?

当前的网l应用Y件还q背了REST?#8220;无状态服务器”U束。REST服务器只知道自己的状态。REST不关心客L的状态,客户端的状态自己来? 理,q是AJAX技术的应用之地。通过AJAX技术,可以发挥有状态网l客h的优ѝ而REST的服务器兛_的是从所有网l客L发送到服务器操作的? 序。这样得互联网q样一个巨大的|络得到有序的管理?

REST与Rails框架

Ruby on Rails框架Q简URails或者Rails框架Q是一个基于Ruby语言的越来越行的网l应用Y件开发框架。它提供了关于REST最好的支持Q也? 当今应用REST最成功的一个Y件开发框架。Rails框架Q从版本1.2.xP成ؓ了第一个引入REST作ؓ核心思想的主网lY件开发框架。在 Rails框架的充分利用了REST软g架构之后Qh们更加坚信REST的重要性和必要性。Rails利用REST软g架构思想对网l服务也提供了一的 支持。从最直观的角度看待RESTQ它是网l服务最理想的手D,但是Rails框架把REST带到了网l应用Y件开发框架。这是一ơ飞跃,让REST的? 想从|络服务的应用提升到了网l应用Y件开发。利用REST思想的simply_restful插g已经成ؓ了Rails框架的核心内宏V?

REST安全?

我们把现有基于SOAP的网l服务和ZREST/HTTP|络服务作个比喻Q前者是一U传l的寄信方式Q而后者是C|络的电子邮件方式。要是是 寄信和电子邮仉有病毒存在的话,传统的寄信被送到Ҏ很危险Q而电子邮件是开发的Q电子邮件供应商比如Google为我们检查了电子邮g是否有病毒? q里q不是说明SOAP|络服务消息包含义病毒,而是说明HTTP是无法处理SOAP信息包究竟好不好Q需要额外的软g工具解决q一问题Q包括防火墙也用 不上和管不了?

REST/HTTP|络服务的信息包可以被防火墙理解和控制。你可以按照操作和链接进行过滤信息包Q如你可以规定从外部来的只能dQGET操作Q? 自己服务器的资源。这样对于系l管理员而言使得软g理更ؓ单。REST的安全性还可以利用传输安全协议SSL/TLS、基本和摘要式认证(Basic und Digest AuthenticationQ。除了这些REST自n的安全性功能外Q还可以利用像基于信息的Web Services SecurityQJSR 155Q作为REST不错的补充?

参考文?/h4> 中文参考文?/strong> Roy Thomas Fielding博士论文中文版本 Roy Thomas Fielding博士论文英文版本
作者简介:骆古道,|名CnrubyQ八十年代初毕业于西北工业大学数理力学系Q?988q公z学d国,从事l合最优化理论研究Q从九十q代初期起一直致力于计算机领域Y件开发、设计和理{方面工作,个h博客?#8220;道喜技术日?/a>”?



二胡 2009-03-22 14:20 发表评论
]]>Cookie, iframe ?P3P 的那点事?/title><link>http://www.tkk7.com/hulizhong/archive/2009/03/21/261194.html</link><dc:creator>二胡</dc:creator><author>二胡</author><pubDate>Sat, 21 Mar 2009 05:34:00 GMT</pubDate><guid>http://www.tkk7.com/hulizhong/archive/2009/03/21/261194.html</guid><wfw:comment>http://www.tkk7.com/hulizhong/comments/261194.html</wfw:comment><comments>http://www.tkk7.com/hulizhong/archive/2009/03/21/261194.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.tkk7.com/hulizhong/comments/commentRss/261194.html</wfw:commentRss><trackback:ping>http://www.tkk7.com/hulizhong/services/trackbacks/261194.html</trackback:ping><description><![CDATA[    看了<a title="http://www.dbanotes.net/" >Fenng</a>的一blog关于Cookie, iframe ?P3PQ自׃是很清楚Q查询了相关资料Q把相关心得贴出来?br />      应用场景Q?br />      比如一?www.a.com中有个服务应用是引用b.com域的Q? <br /> 通常情况下用L隐私{略讄Z或者更?在b.com域的cookie是不能够正常d的,但是服务端能够改变用户对b.com域的隐私{略使b.com域下的cookie正常存取.<br />      也可以说是利用p3p解决W三方cookie存取的问题?br />      W一方Cookie:是来自当前正在查看的|站Q或者发送到当前正在查看的网站?<br />      W三方Cookie:是来自当前正在查看的|站以外的网站,或者发送到当前正在查看的网站以外的|站。第三方|站通常提供正在查看的网站上的内宏V例如,许多站点使用来自W三方网站的q告Q或者iframe的别的网站的urlQ这些第三方的网站可能用的Cookie?<br />     <br />      p3p相关东东<br />      P3P是万l网联盟QW3CQ公布的一wU保护推荐标准,旨在为网上冲的Internet用户提供隐私保护。现在有来多的网站在消费者访问时Q都 会收集一些用户信息。制定P3P标准的出发点是Z减轻消费者因|站攉个h信息所引发的对于隐U权可能受到늊的忧虑。P3P标准的构xQWeb 站点的隐U策略应该告之访问者该站点所攉的信息类型、信息将提供l哪些h、信息将被保留多时间及其用信息的方式Q如站点应做诸如 “本网站将监测您所讉K的页面以提高站点的用率”?#8220;本网站将可能ؓ您提供更合适的q告”{申明。访问支持P3P|站的用h权查看站炚wU报告,然后军_是否接受cookie或是否用该|站?br />      通过指定用户隐私{略Q就可以辑ֈ存取W三方cookie的目的,看到q也怼觉得跟web应用毫无关系Q真正的问题是如何让服务器来指定用户览器的隐私{略Q?br />       其实很简单,只要在响应用戯求的时候在http的头信息中增加关于p3p的配|信息就可以了?br />      <span><span>response.setHeader(</span><span id="rdffx7v" class="string">"P3P"</span><span>,</span><span id="t9npp9n" class="string">"CP='IDC DSP COR ADM DEVi TAIi PSA PSD IVAi IVDi CONi HIS OUR IND CNT'"</span><span>)Q?br />       不对之处Q请大家指正Q?br /> 参考:<br /> http://viralpatel.net/blogs/2008/12/how-to-set-third-party-cookies-with-iframe.html<br /> http://www.dbanotes.net/web/cookie_p3p.html<br /> http://ding--lin.javaeye.com/blog/94336<br /> http://www.cnblogs.com/gudai/archive/2008/09/21/1295432.html<br /> http://cailin.javaeye.com/blog/175422  <br /> </span></span>     <img src ="http://www.tkk7.com/hulizhong/aggbug/261194.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/hulizhong/" target="_blank">二胡</a> 2009-03-21 13:34 <a href="http://www.tkk7.com/hulizhong/archive/2009/03/21/261194.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>?不唐H的JavaScript的七条准?/title><link>http://www.tkk7.com/hulizhong/archive/2009/03/19/260809.html</link><dc:creator>二胡</dc:creator><author>二胡</author><pubDate>Thu, 19 Mar 2009 08:13:00 GMT</pubDate><guid>http://www.tkk7.com/hulizhong/archive/2009/03/19/260809.html</guid><wfw:comment>http://www.tkk7.com/hulizhong/comments/260809.html</wfw:comment><comments>http://www.tkk7.com/hulizhong/archive/2009/03/19/260809.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.tkk7.com/hulizhong/comments/commentRss/260809.html</wfw:commentRss><trackback:ping>http://www.tkk7.com/hulizhong/services/trackbacks/260809.html</trackback:ping><description><![CDATA[?http://www.cnblogs.com/JustinYoung/articles/1350887.html<br /> <h1 class="postTitle"> <a id="AjaxHolder_ctl01_TitleUrl" class="postTitle2" >不唐H的JavaScript的七条准?/a> </h1> <div id="tzz7hj3" class="postBody"> <p>l过多年的开发、教学和~写不唐H的JavaScriptQ?我发C下面的一些准则。我希望它们可以帮助你对“Z么这栯计和执行JavaScript比较?#8221;有一点理解。这些规则曾l帮助我更快C付品,q且产品的质量更高,也更Ҏl护?/p> <p><strong>1.不要做Q何假?/strong> <br /> QJavaScript是一个不可靠的助手)</p> <p>可能不唐H的JavaScript 的最重要的一个特性就是——你要停止Q何假设:</p> <ul> <li>不要假设JavaScript是可用的Q你最好认为它很有可能是不可用的,而不是直接依赖于它?</li> <li>在你l过试认一些方法和属性可以用之前,不要假设览器支持它们?</li> <li>不要假设HTML代码如你惌的那h,每次都要q行查,q且当其不可用的时候就什么也不要做?</li> <li>让JavaScript的功能独立于输入讑֤ </li> <li>要记住其他的脚本可能会媄响你的JavaScript的功能,所以要保证你的脚本的作用域可能地安全?</li> </ul> <p>在开始设计你的脚本之前,要考虑的第一件事情就是检查一下你要ؓ其编写脚本的HTML代码Q看看有什么东西可以帮助你辑ֈ目的?/p> <p><strong>2.扑և钩子和节点关p?/strong>QHTML是脚本的基石Q?/p> <p>? 开始编写脚本之前,要先看一下你要ؓ之编写JavaScript的HTML。如果HTML是未l组l的或者未知的Q那么你几乎不可能有一个好的脚本编写方 案——很可能׃出现下面的情况:要么是会用JavaScript创徏太多标记Q要么就是应用太依赖于JavaScript?/p> <p>在HTML中有一些东襉K要考虑Q那是钩子和节点关pR?/p> <p><strong><1>.HTML 钩子</strong></p> <p>HTML 最初的和最重要的钩子就是IDQ而且ID可以通过最快的DOMҎ——getElementById 讉K到。如果在一个有效的HTML文档中所有的ID都是独一无二的话Q在IE中关于name ?ID 有一个bugQ不q有些好的类库解决了q个问题Q,使用ID是安全可靠的,q且易于试?/p> <p>其他一些钩子就是是HTML元素和CSS c,HTML元素可以通过getElementsByTagNameҎ讉KQ而在多数览器中都还不能通过原生的DOMҎ来访问CSScR不q,有很 多外部类库提供了可以讉KCSScdQ类g getElementsByClassNameQ?的方法?/p> <p><strong><2>.HTML 节点关系</strong></p> <p>关于HTML的另外比较有意思的一点就是标C间的关系Q思考下面的问题Q?/p> <ul> <li>要怎样才可以最Ҏ地、通过最的DOM遍历来到辄标节点? </li> <li>通过修改什么标讎ͼ可以可能多地访问到需要修改的子节点? </li> <li>一个给定的元素有什么属性或信息可以用来到达另外一个元素? </li> </ul> <p>遍历DOM很耗资源而且速度很慢Q这是Z么要量使用览器中已经在用的技术来做这件事情?/p> <p><strong>3.把遍历交l专家来?/strong>QCSSQ更快地遍历DOMQ?/p> <p>? 关DOM的脚本和使用Ҏ或属性(getElementsByTagName, nextSibling, previousSibling, parentNode以及其它Q来遍历DOMgqh了很多hQ这点很有意思。而有的是,我们其实早已l通过另外一U技术—?CSS ——做了这些事情?/p> <p>CSS 是这样一U技术,它用CSS选择器,通过遍历DOM来访问目标元素ƈ改变它们的视觉属性。一D复杂的使用DOM的JavaScript可以用一个CSS选择器取代:</p> <p>var n = document.getElementById('nav'); <br /> if(n){ <br /> var as = n.getElementsByTagName('a'); <br /> if(as.length > 0){ <br /> for(var i=0;as[i];i++){ <br /> as[i].style.color = ‘#369′; <br /> as[i].style.textDecoration = ‘none’; <br /> } <br /> } <br /> }</p> <p>/* 下面的代码与上面功能一?*/</p> <p>#nav a{ <br /> color:#369; <br /> text-decoration:none; <br /> }</p> <p>q是一个可以好好利用的很强大的技巧。你可以通过动态ؓDOM中高层的元素dclass 或者更改元素ID来实现这一炏V如果你使用DOM为文档的bodyd了一个CSSc,那么设计师就很可以容易地定义文档的静态版本和动态版本?/p> <p>JavaScript:</p> <p>var dynamicClass = 'js'; <br /> var b = document.body; <br /> b.className = b.className ? b.className + ' js' : 'js';</p> <p>CSS:</p> <p>/* 静态版?*/</p> <p>#nav { <br /> .... <br /> }</p> <p>/* 动态版?*/</p> <p>body.js #nav { <br /> .... <br /> }</p> <p><strong>4.理解览器和用户</strong>Q在既有的用模式上创徏你所需要的东西Q?/p> <p>? 唐突的JavaScript 中很重要的一部分是理解览器是如何工作的(其是浏览器是如何崩溃的Q以及用h望的是什么。不考虑览器你也可以很ҎC用JavaScript 创徏一个完全不同的界面。拖拽界面,折叠区域Q滚动条和滑动块都可以用JavaScript创徏Q但是这个问题ƈ不是个简单的技术问题,你需要思考下? 的问题:</p> <ul> <li>q个新界面可以独立于输入讑֤么?如果不能Q那么可以依赖哪些东西? </li> <li>我创建的q个新界面是否遵循了览器或者其它富界面的准则(你可以通过鼠标在多U菜单中直接切换吗?q是需要用tab键?Q?</li> <li>我需要提供什么功能但是这个功能是依赖于JavaScript? </li> </ul> <p>最 后一个问题其实不是问题,因ؓ如果需要你可以用DOM来凭I创建HTML。关于这点的一个例子就?#8220;打印”链接Q由于浏览器没有提供一个非 JavaScript的打印文档功能,所以你需要用DOM来创c链接。同样地Q一个实C展开和收~内Ҏ块的、可以点ȝ标题栏也属于q种情况? 标题栏不能被键盘Ȁz,但是链接可以。所以ؓ了创Z个可以点ȝ标题栏你需要用JavaScript链接加入进去,然后所有用键盘的用户可以收 ~和展开内容模块了?/p> <p>解决q类问题的极好的资源是设计模式库。至于要知道览器中的哪些东西是独立于输入设备的Q那p靠经验的U篏了。首先你要理解的是事g处理机制?/p> <p><strong>5.理解事g</strong>Q事件处理会引v改变Q?/p> <p>? 件处理是走向不唐H的JavaScript的第二步。重点不是让所有的东西都变得可以拖拽、可以点L者ؓ它们d内联处理Q而是理解事g处理是一个可? 完全分离出来的东ѝ我们已l将HTML,CSS和JavaScript分离开来,但是在事件处理的分离斚w却没有走得很q?/p> <p>事g处理器会监听发生在文档中元素上的变化Q如果有事g发生Q处理器׃扑ֈ一个很奇妙的对象(一般会是一个名为e的参敎ͼQ这个对象会告诉元素发生了什么以及可以用它做什么?/p> <p>? 于大多数事g处理来说Q真正有的是它不止发生在你惌讉K的元素上Q还会在DOM中较高层U的所有元素上发生Q但是ƈ不是所有的事g都是q? Pfocus和blur事g是例外)。D例来_利用q个Ҏ你可以Z个导航列表只d一个事件处理器Qƈ且用事件处理器的方法来获取真正触发事g 的元素。这U技术叫做事件委托,它有几点好处Q?/p> <ul> <li>你只需要检查一个元素是否存在,而不需要检查每个元?</li> <li>你可以动态地d或者删除子节点而ƈ不需要删除相应的事g处理?</li> <li>你可以在不同的元素上对相同的事g做出响应 </li> </ul> <p>需 要记住的另一件事是,在事件向父元素传播的时候你可以停止它而且你可以覆写掉HTML元素Q比如链接)的缺省行为。不q,有时候这q不是个好主意,因ؓ? 览器赋予HTML元素那些行ؓ是有原因的。D个例子,链接可能会指向页面内的某个目标,不去修改它们能确保用户可以将面当前的脚本状态也加入书签?/p> <p><strong>6.Z人着?/strong>Q命名空_作用域和模式Q?/p> <p>你的代码几乎从来不会是文档中的唯一的脚本代码。所以保证你的代码里没有其它脚本可以覆盖的全局函数或者全局变量显得尤为重要。有一些可用的模式可以来避免这个问题,最基础的一点就是要使用 var 关键字来初始化所有的变量。假设我们编写了下面的脚本:</p> <p>var nav = document.getElementById('nav'); <br /> function init(){ <br /> // do stuff <br /> } <br /> function show(){ <br /> // do stuff <br /> } <br /> function reset(){ <br /> // do stuff <br /> }</p> <p>上面的代码中包含了一个叫做nav的全局变量和名字分别ؓ init,show ?reset 的三个函数。这些函数都可以讉K到navq个变量q且可以通过函数名互相访问:</p> <p>var nav = document.getElementById('nav'); <br /> function init(){ <br /> show(); <br /> if(nav.className === 'show'){ <br /> reset(); <br /> } <br /> // do stuff <br /> } <br /> function show(){ <br /> var c = nav.className; <br /> // do stuff <br /> } <br /> function reset(){ <br /> // do stuff <br /> }</p> <p>你可以将代码装C个对象中来避免上面的那种全局式编码,q样可以将函数变成对象中的ҎQ将全局变量变成对象中的属性?你需要?#8220;名字+冒号”的方式来定义Ҏ和属性,q且需要在每个属性或Ҏ后面加上逗号作ؓ分割W?/p> <p>var myScript = { <br /> nav:document.getElementById('nav'), <br /> init:function(){ <br /> // do stuff <br /> }, <br /> show:function(){ <br /> // do stuff <br /> }, <br /> reset:function(){ <br /> // do stuff <br /> } <br /> }</p> <p>所有的Ҏ和属性都可以通过使用“cd+Ҏ作符”的方式从外部和内部访问到?/p> <p>var myScript = { <br /> nav:document.getElementById('nav'), <br /> init:function(){ <br /> myScript.show(); <br /> if(myScript.nav.className === 'show'){ <br /> myScript.reset(); <br /> } <br /> // do stuff <br /> }, <br /> show:function(){ <br /> var c = myScript.nav.className; <br /> // do stuff <br /> }, <br /> reset:function(){ <br /> // do stuff <br /> } <br /> }</p> <p>q种模式的缺点就是,你每ơ从一个方法中讉K其它Ҏ或属性都必须在前面加上对象的名字Q而且对象中的所有东襉K是可以从外部讉K的。如果你只是惌部分代码可以被文档中的其他脚本访问,可以考虑下面的模块(moduleQ模式:</p> <p>var myScript = function(){ <br /> //q些都是U有Ҏ和属? <br /> var nav = document.getElementById('nav'); <br /> function init(){ <br /> // do stuff <br /> } <br /> function show(){ <br /> // do stuff <br /> } <br /> function reset(){ <br /> // do stuff <br /> } <br /> //公有的方法和属性被使用对象语法包装在return 语句里面 <br /> return { <br /> public:function(){</p> <p>}, <br /> foo:'bar' <br /> } <br /> }();</p> <p>? 可以使用和前面的代码同样的方式访问返回的公有的属性和ҎQ在本示例中可以q么讉KQmyScript.public() ? myScript.foo 。但是这里还有一点让得不舒服Q当你想要从外部或者从内部的一个私有方法中讉K公有Ҏ的时候,q是要写一个冗长的名字Q对象的名字可以非常长)。ؓ 了避免这一点,你需要将它们定义为私有的q且在return语句中只q回一个别名:</p> <p>var myScript = function(){ <br /> // q些都是U有Ҏ和属? <br /> var nav = document.getElementById('nav'); <br /> function init(){ <br /> // do stuff <br /> } <br /> function show(){ <br /> // do stuff <br /> // do stuff <br /> } <br /> function reset(){ <br /> // do stuff <br /> } <br /> var foo = 'bar'; <br /> function public(){</p> <p>}</p> <p>//只返回指向那些你惌讉K的私有方法和属性的指针 <br /> return { <br /> public:public, <br /> foo:foo <br /> } <br /> }();</p> <p>q就保证了代码风g致性,q且你可以用短一点的别名来访问其中的Ҏ或属性?/p> <p>如果你不惛_外部暴露M的方法或属性,你可以将所有的代码装C个匿名方法中Qƈ在它的定义结束后立刻执行它:</p> <p>(function(){ <br /> // these are all private methods and properties <br /> var nav = document.getElementById('nav'); <br /> function init(){ <br /> // do stuff <br /> show(); // q里不需要类名前~ <br /> } <br /> function show(){ <br /> // do stuff <br /> } <br /> function reset(){ <br /> // do stuff <br /> } <br /> })();</p> <p>对于那些只执行一ơƈ且对其它函数没有依赖的代码模块来_q种模式非常好?/p> <p>通过遵@上面的那些规则,你的代码更好Cؓ用户工作Q也可以使你的代码在机器上更好地q行q与其他开发者的代码和睦相处。不q,q有一个群体需要考虑到?/p> <p><strong>7.为接手的开发者考虑</strong>Qɾl护更加ҎQ?/p> <p>使你的脚本真正地unobtrusive的最后一步是在编写完代码之后仔细查一遍,q且要照ֈ一旦脚本上U之后要接手你的代码的开发者。考虑下面的问题:</p> <ul> <li>所有的变量和函数名字是否合理ƈ且易于理解? </li> <li>代码是否l过了合理的l织Q从头到N很流畅吗Q?</li> <li>所有的依赖都显而易见吗Q?</li> <li>在那些可能引h؜淆的地方都添加了注释吗? </li> </ul> <p>最重要的一ҎQ要认识到文档中的HTML和CSS代码相对于JavaScript来说更有可能被改变(因ؓ它们负责视觉效果Q。所以不要在脚本代码中包含Q何可以让l端用户看到的class和IDQ而是要将它们分离出来攑ֈ一个保存配|信息的对象中?/p> <p>myscript = function(){ <br /> var config = { <br /> navigationID:'nav', <br /> visibleClass:'show' <br /> }; <br /> var nav = document.getElementById(config.navigationID); <br /> function init(){ <br /> show(); <br /> if(nav.className === config.visibleClass){ <br /> reset(); <br /> }; <br /> // do stuff <br /> }; <br /> function show(){ <br /> var c = nav.className; <br /> // do stuff <br /> }; <br /> function reset(){ <br /> // do stuff <br /> }; <br /> }();</p> <p>q样l护者就知道d里修改这些属性,而不需要改动其他代码?/p> <p><strong>更多信息</strong></p> <p>以上是我发现的七条准则。如果你惌了解更多与上面所探讨的主题相关的东西Q可以看看下面的链接Q?/p> <ul> <li><a >Yahoo! Design Pattern Library </a></li> <li><a >Event Delegation </a></li> <li><a >Event Driven JavaScript Application Design</a></li> <li><a >JavaScript Programming Patterns</a></li> <li><a >Show love to the Object Literal </a></li> <li><a >A JavaScript Module Pattern</a></li> </ul> </div> <br /> <br /><img src ="http://www.tkk7.com/hulizhong/aggbug/260809.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/hulizhong/" target="_blank">二胡</a> 2009-03-19 16:13 <a href="http://www.tkk7.com/hulizhong/archive/2009/03/19/260809.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>?RichClient/RIA原则与实践(上)http://www.tkk7.com/hulizhong/archive/2009/03/18/260387.html二胡二胡Wed, 18 Mar 2009 01:52:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/03/18/260387.htmlhttp://www.tkk7.com/hulizhong/comments/260387.htmlhttp://www.tkk7.com/hulizhong/archive/2009/03/18/260387.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/260387.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/260387.html

RichClient/RIA原则与实践(上)

作?陈金z?/strong> 发布?2009q??0?上午4??

.NET,
Agile,
Java
主题
RIA,
富客L/桌面
标签
原则

Web领域的经验在q去十多q的不断的用和锤炼中,整个 开发领域的技术、理c缺陷已l趋于成熟。JavaEE Stack, .NET Stack, Ruby On Rails{框架代表了目前q个技术领域的所有经验积累。这h们在开始一个新的项目的时候,只需要选择对应语言的最佛_践,基本上不会犯大的错误。例 如,如果使用Java开发一个新的Web应用Q那么基本上Spring/Guice+Hibernate/iBatis/+Struts /SpringMVCq种架构是不会生重大的架构问题的;如果使用RoR那么你已l在使用最佛_践了Q系l的分层Q领域层Q数据库层,服务层,表现层等 {;Z保证pȝ的可扩展性,服务器端应当是无状态架构,{等。总而言之,web开发领域,它丰富的U篏使得开发者逐渐更多的_֊投入到应用本w?/p>

来看富客LQ或者富互联|应用。在我看来,今天的RichClient与RIA已经没有分别Q只要代表着丰富界面元素和丰富用户体验,需要与服务器进?交互的应用都可以UCؓRichClient或者RIAQ虽然感觉上RichClient?#8220;企业?#8221;一些(服务器往往在企业内部)QRIA?#8220;个h?#8221;一 些(服务器往往处于公网Q。从最的层面来说Q我现在正在使用的离U模式的GoogleDoc是一个RichClient应用──虽然它没有那?RichQ采用和microsoft office一样土的界? 我现在正在听音乐的Last.fm客户端显然是一个非常典型的RIA──它所有的个h喜好信息、音乐全都来自远在美国的服务器。本地的q个界面Q只是提?攉个h和音乐信息,以及控制音乐的播攑֒停止Q目前拥?150万玩家的兽世界Q则是一个挣钱最多的Q最“?#8221;的客LQ?0多G的客L包含了电?品质的广阔场景,华丽的魔法效果和极其复杂的hZ互?/p>

如今的用户需求已l达C一个新的高度,那些灰色的,Ҏ正正的界面已l逐渐不能够满_L需求。从我们工作的客L来,他们除了?#8220;完成功能”有着?本的期待外,对于应用做?#8220;?#8221;Q也抱有极大的热情。我工作的上一个项目是一个CRMpȝQ它是基?NET Framework 3.5的一个RichClient应用。它的主H口是一个带着U色渐变背景的无ҎH口Q还有请专业工制作的图标,点击某一个菜单还有华丽的二菜单?动效果。我们在q个目中获得了很多Q有些值得借鉴Q有些仍然值得反思。我仍然记得我们在项目的不同阶段Q做一个技术决定是如此的彷徨和忐忑Q因为在当时 的RichClient企业开发领域,几乎没有M丰富的经验可以借鉴Q我们重新发明了一些轮子,然后又推dQ我们偏MUI框架l我们提供的各种便利 而自己实现种U基Ҏ,只是因ؓ他们偏离了我们所倡导的测试性的原则。在写下本文的时候,我尝试搜索了一下,仍然没有比较深入的实跉|文章来介绍企业?境下RichClient开发。大多数的书Q如Swing、JavaFX?NET WPF开发等{,偏向于小规模Ҏ介l,而在大规模的企业应用中,q些的技巧对于架构决{往往帮助很小?/p>

我的工作l历应当是和大多数开始进行RichClient开发的开发者类|有着丰富的Web开发的l验之后开始进行RichClient开发。加?ThoughtWorks之后参加了多个不同的RichClient目的开发工作,使用/试q的语言包括Java Swing, Flex/Adobe Air, .NET WinForm/.NET WPF. 对于不同q_之间的种U有些体会。在q里我将q些实践和原则ȝ如下。例子很可能q时Q毕竟华丽的界面框架层出不穷Q但原则应当通用的。用和遵@q些?则将会帮助你犯错误──臛_比我们过ȝ的错误要。如果你拥有一定的web开发经验,那么q篇文章你读h会很亲切?/p>

q些原则/实践往往不是孤立的,我尝试将他们之间用图的方式关联v来,帮助你在使用的过E中q行选择。例如,你遵循了“一切皆异步”的原则,那么很可能你 需要进?#8220;U程理”?#8220;事g理”Q如果你需要引?#8220;~存与本地存?#8221;Q那?#8220;数据交互模式”你也需要进行考虑。希望这张图能够帮助读者理解不同原则之间的联系?/p>

下面列出的这些原则或者实跉|有严格意义上的区分。按照上面的图,我推荐是Q一旦你考虑C某一个实践,那么与它直接兌的实践你最好也要实现。它会得你的架构更全面Q经得v用户功能的需求和交互的需求?/p>

Z让这些实跉|加通用Q我采用伪代码书写。相信读者能够{化成相应的语a──Java, C#, ActionScript或者其他。这些实践ƈ非与某一U语a相关。在某些特定的例子中Q我会采用特定语aQ但大多数都是伪代码描述的?/p>

1 一切皆异步

所有耗时的操作都应当异步q行。这是第一条、也是最重要的原则,q背了这条原则将会导致你的应用完全不可用?/p>

考虑q样的一个功能:点击一?更新股票信息"按钮Q系l会从股市场(W三方应用)获得最新的股票信息Qƈ信息更新到ȝ面。丝毫不考虑用户体验的写法:

void updateStockDataButton_clicked() {
    stockData = stockDataService.getLatest(); // 从远E获取股信?
    updateUI(stockData); // q个Ҏ会更新界?
}

那么Q当用户点击updateStockDataButton的时候,会有什么反应?难说。如果是一个无限带宽、无限计资源的世界Q这D代码直观又?懂,而且工作的非常好Q它会从W三方股系l读到股数据,q且更新到界面上。可惜不是。这D代码在现实世界工作的时候,当用L击这个按钮,整个界面?ȝ──知道那种感觉吗?是点完q个按钮Q界面不动了Q如果你在用Windows, 然后试拽住H口到处UdQ你会发现这个窗口经q的地方都是白的。你的客户不会理解你的程序实际上在很努力的从股票市场获得数据Q他们只会很愤怒的_q?个东西把我的机器弄死了!他们的思\被打断了。于是他们不再用你的程序,你们的合作没了。你没钱了。你的狗也跑了?/p>

出现界面ȝ的原因是Q耗时操作d了UIU程。UIU程一般负责着渲染界面Q响应用户交互,如果q个U程被阻塞,它将无法响应所有的用户交互hQ甚?包括拖拽H口q样单的操作。所有的界面框架Q无论是Java/.NET/ActionScript/JavaScript, 都只有一个UIU程Q这个估计永q都不会变?/p>

用户看到的应用通常与程序员大相径庭。用户对应用的期待别分别是Q能用、可用、好用、好看。而我观察到的大多数程序员停留在第一阶段Q能用?#8220;一切皆异步”q个原则说来单,做v来也不会很难。把上面的代码稍作改动,如下Q?/p>

void updateStockDataButton_clicked() {
    runInAnotherThread( function () {
        stockData = stockDataService.getLatest(); // 从远E获取股信?
        updateUI(stockData); // q个Ҏ在UIU程更新界面
    }
}

注意加粗部分?code>runInAnotherThread是跟语言q_特定的。对?net C#Q可以是一?code>Dispatcher+delegate或?code>ThreadPool.QueueUserWorkItemQ对于JavaQ可以干脆是一?code>Runable。对于AJAX, 可以?code>XMLHttpRequest或者把q个计算扔到一?code>IFrame中;对于ActionScript, g没有什么好的方法,把获取数据的部分交给XML.load然后通过事g回调的方式来q行界面h吧?/p>

耗时操作一般两U来源生:|络带来的gq以及大规模q算。两者对应的异步实现方式有所不同。前者往往可以通过特定语言、^台的获取数据的方式来q行异步Q特别是~Z多线E特性的动态语a。例如典型的AJAX方式Q?/p>

xhr = new XmlHttpRequest()
xhr.send("POST", '/stockData/MSFT', function() {
    doSomethingWith(xhr.responseText);  // 只有当数据返回的时候,才会调用
})

大规模运带来的耗时在Java/C#{支持多U程的语a环境中很Ҏ实现Q而对于JavaScript/ActionScript{很难,折衷的方式是 复杂运gq到服务器端q行Q或者将复杂q算拆解成若q个耗时较少的小q算Q例如ActionScript的伪多线E实现方式?/p>

“一切皆异步”q个原则说来ҎQ但要在企业应用中以一U一致的方式q行实现很难。上例中runInAnotherThread的方式貌似简单,也可能出 现在各种GUI框架的介l中Q但l不是一个稍兯模的RichClient应当采用的方式。它很难作ؓ一U编E范式被遵@Q你l不会希望看到在你的代码?所有用到异步的地方?code>new Runnable(){...}。这样带来的问题不仅仅是异步被不被管理的到处乱扔Q还带来了测试的复杂性。ؓ了解册些只有在臛_有点规模?RichClient中才出现的问题,你最好也实现?#8220;4 U程理”Q见下篇Q,能够实现“3 事g理”Q见下篇Q更好。终极方式是这些抽象到应用的基框架中,使得所有的开发h员以一U一致的方式q行~程?/p>

2 视图理

2.1 视图生命周期理

视图q个概念在WEB开发中几乎被忽略。这里所说的视图是指面、页面块{界面元素。在WEB开发中Q视囄生命周期很短Q在q入面的时候创建,在离开面的时候销毁。一不小心页面被弄糟了,或者不能按照预期的渲染了,点下h按钮Q整个世界一片清净?/p>

WEB下的视图D也是如此自然。基于超链接的方式,每点Mơ,p够打开一个新的页面,旧的面被浏览器销毁,新的面诞生。(q里不考虑AJAX或者其他JavaScriptҎQ?/p>

如果把这U想法带入到RichClient开发,后果会很p糕。每当点L钮或者进行其他操作需要导航到新的H口Q你不加M限制的创建新H口或者新的视 图。然而CPU不是无限的。创Z个新的视N常是很耗CPU和内存的。系l响应会变慢。用户会抱怨,拒绝付钱Q于是因为饥饿,你的狗再ơ离开了你?/p>

每次新创图生的严重后果q不仅仅是非功能性的Q还包括功能性的~失。如果你用过SkypeQ当你在l张三通话的时候,再次点击张三q且q行通话Q你 会发现刚刚的通话界面会弹出来Q而不是开启新H口。在我们的一个项目中Q有一个功能:点击软g界面上的电话Lp开启一个新H口Qƈ直接q到桌上的电?拨号通话。可以想象,如果每次都会弹出新的H口QY件的逻辑是根本错误的?/p>

如何解决q个问题Q最单的方式是将所有已知的视图全都保存到本地的一个缓存中Q我们命名ؓViewFactoryQ当需要进行获取某个视囄时候,直接?code>ViewFactory拿到Q如果没有创建,那么创徏Qƈ攑ֈCache中:

class ViewFactory {
     cache = {}
     View getView(Object key) {
        if cache.contains(key) {
            return cache[key]
        }
        cache[key] = createView(key)
        return cache[key]  
    }
}

需要注意的是,ViewFactory?code>key的选择。对于简单的应用Q?code>key可以q脆是某个单独H口的类名。例如整个系l中往往只有一个配|窗口,?么key是q个cdQ对于需要复用的H口Q往往需要根据其业务主键来创建相应的视图。例如代码中只有一?code>UserDetailWindow, 需要用来展CZ同用L信息。当需要同时显CZ个以上的用户信息的时候,用同一个窗口实例显然不寏V这时?code>key的选择可以是类?用户ID?/p>

2.2 视图D

上面的方案ƈ没有解决D的问题。导航需要解决的问题有两个,如何D以及如何在导航时传递数据。这时候不得不慕WEB的解x式。我要访?code>ID?code>1的用户信息,只需要访问类gusers/1的页面就好;需要访问搜索结果第5,只需要访?code>/search?q=someword&page=5好。这?code>/search是视图,q=someword?code>page=5是传递的数据。目前我q没有发CQ何一本书来讲q如何进行视囑֯航。我们的方式是实C?code>Navigatorcȝ来导航,Navigator依赖于前面提到的ViewFactoryQ?/p>

class Navigator {
    Navigator(ViewFactory viewFactory) {
        this.viewFactory = viewFactory;
    }
    void goTo(Object viewKey) {
        this.viewFactory.getView(viewKey).show()
    }
}

Q这个类看v来跟ViewFactory没什么大的差别,但他们逻辑上是完全不同Qƈ且下面的扩展中会增强Q?/p>

q样是可以解决问题的。如果要在不同的视图之间传递数据,只需要对Navigator.goToҎE加扩展Q多d一个参数就能够传递参C。例如,在用户列表窗口点ȝ户名Q发送一条消息ƈ打开聊天H口Q可以写为:

void messageButton_clicked() {
    Navigator.goTo("ChatWindow#userId", "聊天消息")
}

然而这U方式ƈ不完。当你发现大量的数据在窗口之间交互的时候,q种主动权交给调用Ҏ制的方式Q会l状态同步带来不麻烦;如果你用了本地存储Q它过存储层直接与服务器交互的方式也会带来不少的不便之处。更好的方式是?#8220;3 事g理”Q见下篇Q。当Ӟ如果H口之间D不存在数据传递,ZNavigator的方式仍然简单ƈ且可用?/p>

相关阅读Q?/strong>

[ ThoughtWorks实践集锦Q?Q] 我和敏捷团队的五个约?/a>?/p>

[ ThoughtWorks实践集锦Q?Q] 如何在敏捷开发中做好数据q移?/p>


作者介l?/strong>Q陈金洲QBuffalo AJAX中文问题 Framework作者,ThoughtWorks咨询师,现居北京。目前的工作主要集中在RichClient开发,同时一直对Web可用性进行观察,q对其实C持兴?/p>

lInfoQ中文站投E或者参与内容翻译工作,请邮件至editors@cn.infoq.com。也Ƣ迎大家加入?a target="_blank">InfoQ中文站用戯论组中与我们的编辑和其他读者朋友交?/p>

二胡 2009-03-18 09:52 发表评论
]]>
?RichClient/RIA原则与实践(下)http://www.tkk7.com/hulizhong/archive/2009/03/18/260385.html二胡二胡Wed, 18 Mar 2009 01:46:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/03/18/260385.htmlhttp://www.tkk7.com/hulizhong/comments/260385.htmlhttp://www.tkk7.com/hulizhong/archive/2009/03/18/260385.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/260385.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/260385.html

RichClient/RIA原则与实践(下)

作?陈金z?/strong> 发布?2009q??1?下午10??

.NET,
Agile,
Java
主题
RIA,
富客L/桌面
标签
原则

3 事g理

事g理应当是整个RichClient/RIA开发中的最难以把握的部分。这部分控制的好Q你的程序用h如行云水Q用L思维不会被打断。Q何一 个做RichClient开发的E序员,可以对其他方面毫无所知,但这部分应当非常熟悉。事件是RichClient的核心,?#8220;一切皆异步”的终极实现。前面所说的例子Q实际上可以被抽象ؓ事gQ例如第一个,获取股票数据Q从事g的观点看Q应该是Q?/p>

  • 开始获取股数?
  • 正在获取股票数据
  • 获取数据完成
  • 获取数据p|

看v来相当复杂。然而这样去考虑的时候,你可以将执行计算与界面展现清晰的分开。界面只需要响应事Ӟq算可以在另外的地方 悄悄的进行,q当d完成或者失败的是时候报告相应的事g。从l验看来Q往往同样的数据会在不同的地方q行不同的展C,例如skype在通话的时候这个h 的头像会昄为占U,而具体的通话H口中又是另外不同的展现QMSN的个人签名在好友列表H口中显CZؓ一个点d以编辑控Ӟ而同时在聊天H口昄Z?不能点击只能看的标签。这是RichClient的特性,你永q不知道同一份数据会以什么Ş式来展现Q更要命的是Q当数据在一个地Ҏ新的时候,其他所?能展现的地方都需要同时做相应的更新。如果我们仍然以W一部分的例子,单采?code>runInAnoterThread是完全不能解册个问题的?/p>

我们曄犯过一些很严重的错误,D最l即侉K构都U重难返。无视事件的抽象带来的媄响是架构U别的,修补无于事?/p>

事g的实现方式可以有很多U。对于没有事件支持的语言Q接口或者干脆某一个约束的Ҏ可以。有事g支持的语a能够享受到好处,但仍然是语法U别的,Ҏ 是一L。观察者模式在q里很好用。仍然以股票ZQ被观察的对象就是获取股数据对?code>StockDataRetriverQ观察的是StockWindowQ?/p>

StockDataRetriver {
observers: []
retrieve() {
try {
theData = ...// 从远E获取数?
observers.each {|o| o.stockDataReady(theData)}  // 触发数据获取成功事g
} catch {
observers.each { |o| o.stockDataFailed() }  // 触发事g获取p|事g
}
}
}
StockDataRetriver.observers.add(StockWindow)  // StockWindow加入到观察者队?
StockWindow {
stockDataReady(theData) {
showDataInUIThread(); // 在UIU程昄数据
}
stockDataFailed() {
showErrorInUIThread(); // 在UIU程昄错误
}
}

你会发现代码变得单。UI与计之间的耦合被事件解开Qƈ且区分UIU程与运线E之间也变得Ҏ。当试以事件的视角去观察整个应用程序的时候,你会更关注于用户与界面之间的交互?/p>

让我们l抽象。如果把“获取股票数据”q个按钮点击Q让StockDataRetriver去获取数据当作事件来处理Q应该怎么写呢Q将按钮作ؓ被观?者,StockDataRetriver作ؓ观察者显然不好,好不Ҏ分开的耦合又黏在一赗引入一个中间的Events看v来不错:

Events {
listeners: {}
register(eventId, listener) {
listeners[eventId].add(listener)
}
broadcast(eventId) {
listeners[eventId].observers.each{|o| o.doSomething(); }
}
}

Events中维护了一?code>listeners的列表,它是一个简单的Hashl构Qkey?code>eventIdQvalue?code>observer的列表;它提供了两个ҎQ用来注册事件监听以及通知事g产生。对于上面的案例Q可以先注册StockDataRetriverZ个观察者,观察start_retrive_stock_data事gQ?/p>

Events.register('start_retrive_stock_data', StockDataRetriever)

当点?#8220;获取股票数据”按钮的时候,可以是这P

Events.broadcast('start_retrive_stock_data')

你会发现StockDataRetriver能够老老实实的开始获取数据了?/p>

需要注意的是,q所有事件定义ؓ全局事g是一个好的实c在更大规模的系l中Q将事gq行有效整理和分U是有好处的。在强类型的语言Q如 Java/C#Q中Q抽象出强类型的EventIdQ能够帮助理解系l和q行~程Q避免到处进行强制类型{换。例如,StockEventQ?/p>

StockDataLoadedEvent {
StockData theData;
StockDataLoadedEvent(StockData theData);
}
Event.broadcast(new StockDataLoadedEvent(loadedData))

q个事g的监听者能够不加类型{换的获得StockData数据。上面的例子是不支持事g的语aQC#语言支持自定义强cd的事Ӟ用v来要自然一些:

delegate void StockDataLoaded(StockData theData)

事g理原则我相信ƈ不难理解。然而困隄是具体实现。对一个新的UI框架不熟悉的时候,我们l常?#8220;代码的优?#8221;?#8220;界面提供的特?#8221;之间徘徊。实现这 L一个事件架构需要在目一开始就E具雏ŞQƈ且所有的事g都有良好的命名和理。避免在命名、用事件的时候的随意性,对于让代码可诅R应用稳定有?常大的意义。一个好的事件管理、通知机制是一个良好RichClient应用的根本基。一般说来,你正在用的~程q_如Swing/WinForm /WPF/Flex{能够提供良好的事g响应机制Q即监听事g、onXXX{,但一般没有统一的事件的监听和管理机制。对于架构师Q对于要使用的编E^?对于q些的原生支持要了熟于心Q在~写q样的事件架构的时候也能兼这些语a、^台提供给你的支持?/p>

采用了事件的事g后,你不得不同时实践“U程理”Q因Z件一般来说意味着耗时的操作放到别的地方完成,当完成的时候进行事仉知。简单的模式下,你可以在所有需要进行异步运的地方Q将q算攑ֈ另外一个线E,?code>ThreadPool.QueueUserWorkItem, 在运完成的时候通知事g。但从资源的角度考虑Q将q些U程资源有效的管理也是很重要的,?#8220;U程理”部分有详l的阐述。另外,如果能将你的应用转变?数据驱动的,你需要关?#8220;~存以及本地存储”?/p>

4 U程理

在WEB开发几乎无需考虑U程Q所有的面渲染由浏览器完成Q浏览器会异步的q行文字和图片的渲染。我们只需要写界面和JavaScript好。如果你认同“一切皆异步”Q你一定得考虑U程理?/p>

毫无理的线E处理是q样的:凡是需要进行异步调用的地方Q都新v一个线E来q行q算Q例如前面提到的runInThread的实现。这U方式如果托在 ?#8220;事g理”之下Q问题不大,只会l测试带来一些麻烦:你不得不wait一D|间来定是否耗时操作完成。这U方式很山寨Q也无法实现更高U功能。更?的的方式是将q些U程资源q行l筹理?/p>

U程的管理的核心功能是用来统一化所有的耗时操作Q最单的TaskExecutor如下Q?/p>

TaskExecutor {
void pendTask(task) { //task: 耗时操作d
runInThread {
task.run(); // q行d
}
}
}
RetrieveStockDataTask extends Task {
void run() {
theData = ... // 直接获取q程数据Q不用在另外U程中执?
Events.broadcast(new StockDataLoadedEvent(theData)) // q播事g
}
}

需要进行这个操作的时候,只需要执行类g下面的代码:

TaskExecutor.pendTask(new RetrieveStockDataTask())

好处很明显。通过引入TaskExecutorQ所有线E管理放在同一个地方,耗时操作不需要自行维护线E的生命周期。你可以?code>TaskExecutor中灵zd义线E策略实C些有的效果Q如暂停执行Q监控Q务状늭Q如果你愿意Qؓ了更好的q行调试跟踪Q你甚至可以所有的d以同步的方式执行?/p>

耗时d的定义与执行被分开Q得在d内部能够按照正常的方式进行编码。测试也很容易写了?/p>

不同的语aq_会提供不同的U程理能力?NET2.0提供?code>BackgroundWorker, 提供了一序列对多U程调用的封装,事g如开始调用,调用Q跨U程q回|报告q算q度{等。它内部也实C对线E的调度处理。在你要开始实现类似的TaskExecutorӞ参考一下它的API设计会有参考h倹{Java 6提供的Executor也不错?/p>

一个完善的TaskExecutor可以包含如下功能Q?/p>

  • Task的定义:一个通用的Q务定义。最单的是run()Q复杂的可以加上生命周期的管理:start()?code>end()?code>success()?code>fail()..取决于要控制到多么细致的_度?
  • pendTaskQ将d攑օq算U程?
  • reportStatusQ报告运状?
  • 事gQQ务完?
  • 事gQQ务失?

写这L一个线E管理的不难。最单的实现是每当pendTask的时候新开U程Q当q算l束的时候报告状态。或者用像BackgroundWorker或?code>Executorq样的高UAPI。对于像ActionScript/JavaScriptq样的,只能用伪U程Q?或者干脆将无法拆解的Q务扔到服务器端完成?/p>

5 ~存与本地存?/h2>

Ua的B/Sl构Q浏览器不持有Q何数据,包括基本不变的界面和实际展现的数据。RichClient的一大进步是界面部分本地持有,与服务器只作数据通讯Q从而降低数据流量。像《魔兽世界?0多G的超大型客户端,在普通的拨号|络都可以顺畅的游戏?/p>

~存与本地存储之间的差别在于Q前者是在线模式下,一D|间不变的数据~存Q最的与服务器q行交互Q更快的响应客户Q后者是在离U模式下Q应用仍然能 够完成某些功能。一般来_凡是需要类g“查看XXX历史”功能的,需?#8220;点击列表查看详细信息”的,都会存在本地存储的必要,无论q个功能是否需要向 用户开放?/p>

无论是缓存还是本地存储,最需要处理的问题如何处理本地数据与服务器数据之间的更新机制。当新数据来的时候,当旧数据更新的时候,当数据被删除的时候,{?{。一般来_引入q个实践Q最好也实现Z数据变化?#8220;事g理”。如果能够实?#8220;客户?服务器数据交互模?#8221;那就更完了?/p>

我们犯过q样一个错误。系l启动的时候,当前用L联系人列表读取出来,攑ֈ内存中。当用户双击q个联系人的时候,弹出q个联系人的详细信息H口。由?没有本地存储Q由于采用了Navigator方式的导航,于是很自然的采用?code>Navigator.goTo('ContactDetailWindow', theContactInfo)。由于列表页面一般是不变的,因此昄出来的永q是那䆾旧的数据。后来有了编辑联pMh信息的功能,ZL昄更新的数 据,我们调用更改ؓNavigator.goTo('ContactDetailWindow', 'contactId')Q然后在ContactDetailWindow中按?code>contactId把联pMh信息重新d一ơ。远在南非的用户抱怨慢。还 好我没养狗,没有狗离开我。后来我们慢慢的实现了本地存储,所有的数据d都从q个地方获得。当数据需要更新的时候,直接更新q个本地存储?/p>

本地存储会在Ҏ上媄响RichClientE序的架构。除非本C保存M信息Q否则本地存储一定需要优先考虑。某些编E^台需要你在本地存储界面和?据,如Google Gears的本地存储,|于Adobe Air的AJAX应用{,某些~程q_只需要存储数据,因ؓ界面完全是本地绘制的Q如Java/JavaFX/WinForm/WPF{。缓存界面与~存 数据在实C差别很大?/p>

本地存储的存储机制最好是采用某一U基于文件的关系数据库,如SQLite、H2QHypersonicSQLQ、Firebird{。一旦确定要采用本地存储Q就从成熟的数据库中选择一个,而不要尝试着自己写基于文件的某种~存机制。你会发现到最后你实现了一个山寨版的数据库?/p>

在没有考虑本地存储之前Q与q端的数据访问是直接q接的:

我们上面的例子说明,一旦考虑使用本地存储Q就不能直接讉Kq程服务器,那么需要一个中间的数据层:

数据层的主要职责是维护本地存储与q程服务器之间的数据同步Qƈ提供与应用相关的数据~存、更新机制。数据更新机制有两种Q一U是ProxyQ代理)模式Q一U是自动同步模式?/p>

代理模式比较Ҏ理解。每当需要访问数据的时候,请求发送到q个代理。这个代理会查本地是否可用,如果可用Q如~存处于有效期,那么直接从本地读取数 据,否则它会真正去访问远端服务器Q获取数据,更新~存q返回数据。这U手工处理同步的方式单ƈ且容易控制。当应用处于ȝ模式的时候仍然可以工作的?好?/p>

自动同步模式下,客户端变成都针对本地数据层。有一个健壮的自动同步机制与服务器的保持长q接Q保证数据一直都是更新的。这U方式在应用需要完全本地可q行的时候工作的非常好。如果设计得好,自动同步方式健壮的话Q这U方式会l编E带来极大的便利?/p>

说到同步Q很多h会考虑数据库自带的自动同步机制。我完全不推荐数据库自带的机制。他们的设计初衷本n是ؓ了数据库备䆾Q以及可扩展?QScalabilityQ的考虑。在应用层面Q数据库的同步机制往往不知道具体应用需要进行哪些数据的同步Q同步周期等{。更致命的是Q这U机制或多或 会要求客户端与服务器端具备cM的数据库表结构,q就q样的设计会l客L的缓存表设计带来很大的局限。另外,它对客户?服务器连接也存在一定的局?性,例如需要开攄定端口,特定服务{等。对于纯_的Internet应用Q这U方式更是完全不可行的,你根本不知道q程数据库的l构Q例?Flickr, Google Docs.

当本地存?自动同步机制?#8220;事g理”都实现的时候,应用会是一U全新的架构Q基于数据驱动的事gl构。对于所有本地数据的增删攚w定义ZӞ关?q些数据的视N注册为响应的观察者,d数据的变化于展现隔R界面永q只是被动的响应数据的变化,在我看来Q这是最极致的方式?/p>

l尾

限于幅Q这文章ƈ没有很深入的讨论每一U原?实践。同时还有一些在RichClient中需要考虑的东西我们ƈ没有讨论Q?/p>

  • UInternat应用ȝ模式的实?/strong>。像AdobeAir/Google Gears都有ȝ模式和本地存储的支持Q他们的特点是缓存的不仅仅是数据Q还包括界面。虽然常规的企业应用不太可能包含q些Ҏ,但也具备借鉴意义?
  • 状态的控制。例如管理员能够看到~辑按钮而普通用h法看见,例如不同操作pȝ下的快捷键不同。简单情况下Q通过if-else或者对应编E^C提供的绑定能够完成,然而涉及到更复杂的情况Ӟ特别是网l游戏中大量互斥状态时Q一个设计良好的分层状态机模型能够解决q些问题。如何定义、分析这些状态之间的互斥、ƈ行关p,也是处理复?
  • 试?/strong>。如何对RichClientq行试Q特别是像WPF、JavaFX、Adobe Air{用Runtime+~程实现的框架。它们控制了视图的创E,q且們֐于绑定来q行界面更新。采用传l的MVP/MVC方式会带来巨大的不必要的工作量(我们q么做过Q)Q而且试带来的h值ƈ没有惌那么高?
  • 客户?服务器数据交互模?/strong>。如何进行客h服务器之间的数据交互Q最单的方式是类gHttp Request/Response。这U方式对于单用户E序工作得很好,但当用户之间需要进行交互的时候,会面临巨大挑战。例如,股票代理人关注亚z银行板块,刚好有一新的关于这斚w的评论出玎ͼ股票代理人需要在最?分钟内知道这个消息。如果是Http Request/Response, 你不得不做每?分钟刷一ơ的蠢事Q虽然大多数时候都不会l你数据。项目一旦开始,应当仔l考虑是否存在q样的需求来选择如何q行交互。这部分与本地存储也有密切的关系?
  • 部v方式。RichClient与B/S 直接最大的差异是Q它需要本地安装。如何进行版本检以及自动升U?如何q行分发Q在大规模访问的时候如何进行服务器端分布式部vQ这些问题有些被新技术解决了Q例如Adobe Air以及Google GearsQ但仍然存在考虑的空间。如果是一个安全要求较高的应用Q还需要考虑两端之间的安全加密以及客L正确性验证。新的UI框架层出不穷。开始一个新的RichClient目的时候,作ؓ架构?Tech Lead首先应当x的不是华丽的界面和效果,应当观察如何上q原则和旉华丽的界面框架结合v来。就像我们开始一个web目׃考虑domain 层、持久层、服务层、web层的技术选型一Pq些原则和实践也是项目一开始就考虑的问题?

感谢

感谢我的同事周小强、付莹在我写作过E中提供的无U的和帮助。小强推荐了介绍Google Gears架构的链接,让我能够写作“本地存储”部分有了更深的体会?/p>

q篇文章是我q两q来在RichClient工作、网l游戏、WebGame众多思考的一个集合。我试qJavaFX/WPF/AdobAir 以及相关的文章,然而大多数的例子都是从华丽的界面入手,没有实践相关的内宏V有意思的反而是《大型多人在U游戏开发》这本书Q给了我在企?RichClient开发很多启发。我们曾l犯了很多错误,也获得了许多l验Q以后我们应当能做得更好?/p>

参?/h2>
  • .NET的BackgroundWorker定义Q?a target="_blank">http://msdn.microsoft.com/en-us/library/system.componentmodel.backgroundworker.aspx
  • ActionScript的伪U程Q?a target="_blank">http://blogs.adobe.com/aharui/2008/01/threads_in_actionscript_3.html
  • Google Gears架构Q?a target="_blank">http://code.google.com/apis/gears/architecture.html

相关阅读Q?/strong>

[ ThoughtWorks实践集锦Q?Q] 我和敏捷团队的五个约?/a>?/p>

[ ThoughtWorks实践集锦Q?Q] 如何在敏捷开发中做好数据q移?/p>

[ ThoughtWorks实践集锦Q?Q] RichClient/RIA原则与实践(上)?/p>


作者介l?/strong>Q陈金洲QBuffalo AJAX中文问题 Framework作者,ThoughtWorks咨询师,现居北京。目前的工作主要集中在RichClient开发,同时一直对Web可用性进行观察,q对其实C持兴?/p>

lInfoQ中文站投E或者参与内容翻译工作,请邮件至editors@cn.infoq.com。也Ƣ迎大家加入?a target="_blank">InfoQ中文站用戯论组中与我们的编辑和其他读者朋友交?/p>

二胡 2009-03-18 09:46 发表评论
]]>?web应用的跨访问的解决Ҏhttp://www.tkk7.com/hulizhong/archive/2009/02/17/255084.html二胡二胡Tue, 17 Feb 2009 06:23:00 GMThttp://www.tkk7.com/hulizhong/archive/2009/02/17/255084.htmlhttp://www.tkk7.com/hulizhong/comments/255084.htmlhttp://www.tkk7.com/hulizhong/archive/2009/02/17/255084.html#Feedback0http://www.tkk7.com/hulizhong/comments/commentRss/255084.htmlhttp://www.tkk7.com/hulizhong/services/trackbacks/255084.html 做过跨越多个|站的Ajax开发的朋友都知道,如果在A|站中,我们希望使用Ajax来获得B|站中的特定内容Q如果A|站与B|站不在同一个域中,那么出C跨域讉K问题?br />

      Ajax的跨域访问问题是现有的Ajax开发h员比较常遇到的问题?br />       IE对于跨域讉K的处理是Q弹告框Q提醒用戗如果用户将该网站纳入可信Q|站Q或者调低安全别,那么q个问题IE׃会在提醒你?nbsp; 
         FireFox{其它非微Y的浏览器遇到跨域讉KQ则解决Ҏl一是拒l访问?br />          ? QIE是主浏览器Q只要它能正怋用就好了。此a差已QIE虽然能够处理Q但是是有前提的Q要么用户不厌其烦地在页面弹告框之后点击是(点击? ׃执行该Ajax调用了)Q要么用户将该网站纳入可信Q站点。这两种做法Q在企业理pȝ的应用中倒是比较常见Q因为系l管理员可以以行政手D保证用? 的行为。但是对于互联网上的|站或者门户开发,q种做法则不行?br />          最q遇Cq个问题Q需要在跨域讉Kl束之后完成使主H口出现一些特效,搜烦了一些资料,通过不断试以及在不同浏览器中进行兼Ҏ测试,扑ֈ了几个可行的ҎQ?br />          1? Web代理的方式。即用户讉KA|站时所产生的对B|站的跨域访问请求均提交到A|站的指定页面,p面代替用户面完成交互Q从而返回合适的l果。此 Ҏ可以解决现阶D|能够惛_的多数跨域访问问题,但要求A|站提供Web代理的支持,因此A|站与B|站之间必须是紧密协作的Q且每次交互q程QA|站 的服务器负担增加Q且无法代用户保存session状态?br />          2、on-Demand方式。MYMSN的门户就用的q种方式Q不q? MYMSN中不涉及跨域讉K问题。动态控制script标记的生成,通过修改script标记的src属性完成对跨域面的调用。此Ҏ存在的缺? 是,script的src属性完成该调用旉取的方式时get方式Q如果请求时传递的字符串过大时Q可能会无法正常q行。不q此Ҏ非常适合聚合c门户 用?br />          3、iframe方式。查看过醒来在javaeye上的一关于跨域访问的帖子Q他提到自己已经用iframe的方式解? 了跨域访问问题。数据提交跟获取Q采用iframeq种方式的确可以了,但由于父H口与子H口之间不能交互Q跨域访问的情况下,q种交互被拒l)Q因此无 法完成对父窗口效果的影响?br />          4、用h地{储方式:IE本n依附于windowsq_的特性ؓ我们提供了一U基? iframeQ利用内存来“l行”的方案,即两个window之间可以在客L通过windows剪脓板的方式q行数据传输Q只需要在接受数据的一方设|? Intervalq行轮询Q获得结果后清除Interval卛_。FF的^台独立性决定了它不支持剪脓板这U方式,而以往版本的FF中存在的插g漏洞又被 fixed了,所以FF无法通过内存来完成暗渡陈仓。而由于文件操作FF也没有提供支持(无法通过Cookie跨域完成数据传递)Q致使这U技巧性的方式 只能在IE中用?br />          5、我自己用于解决q类问题的方式:l合了前面几U方式,在访问A|站Ӟ先请求B|站完成数据处理Q再? 据返回的标识来获得所需的结果。这U方法的~点也很明显QB|站的负载增大了。优点,对session也实C保持Q同时A|站与B|站面间的交互能力 增强了。最重要的一点,q种Ҏ满了我的全部需?





二胡 2009-02-17 14:23 发表评论
]]>
վ֩ģ壺 ۺһۺͼ| ͵͵߳վý | ޾Ʒm߹ۿ| ޳aƬ2023| պþëƬ| ëƬ߿ò| Ƶ߿| պƵ| ޾ƷƷԲۿ| avƷʵ| 2022| ѹۿֻ߹ۿ| þþƷѿ| ƷƵվ| AV뾫Ʒһ | þþþþaŷAV| վ߹ۿ| è˳վ߹ۿ| ҹAV| һӰ߿Ƭ| պƷרվ| ˾þں2019| ޳AVӰƬ߹ۿ| Ļѿ| ƷƵһ| ޾Ʒ| ѵӰվ| ޾ƷɫƵ߹ۿԴ | 2022Ļ| 91޾Ʒ߹ۿ| ѹվ߹ۿͼ| 91avƵ߹ۿ| һ| 99þþƷ| ձһѹۿ | ëƬһƵѲ | ҹƷ߹ۿ| ĻӰԺ4hu| ƵС˵ͼƬ| Ѿþþþþþ| ŮƵaƵȫ|