寻觅一只耳朵

javascript匿名函数和闭包

闭包是javascrip等动态解释语言非常重要的一个概念。有了闭包可以很灵活的运用开发。给人一种艺术感。编程也是一种艺术。
闭包是可以包含自由(未绑定到特定对象)变量的代码块;这些变量不是在这个代码块内或者任何全局上下文中定义的,而是在定义代码块的环境中定义(局部变量)。“闭包” 一词来源于以下两者的结合:要执行的代码块(由于自由变量被包含在代码块中,这些自由变量以及它们引用的对象没有被释放)和为自由变量提供绑定的计算环境(作用域)。在 Scala、Scheme、Common Lisp、Smalltalk、Groovy、JavaScript、Ruby 和 Python,objective c 等语言中都能找到对闭包不同程度的支持。

在编程领域我们可以通俗的说:子函数可以使用父函数中的局部变量,这种行为就叫做闭包
闭包的价值在于可以作为函数对象 或者匿名函数,对于类型系统而言这就意味着不仅要表示数据还要表示代码。支持闭包的多数语言都将函数作为第一级对象,就是说这些函数可以存储到变量中、作为参数传递给其他函数,最重要的是能够被函数动态地创建和返回。
举个简单的例子:

function a() {
var i = 0;
function b() {
alert(++i);
}
return b;
}

var c = a();
c(); //1
c(); //2

这段代码有两个特点:
(1)函数b嵌套在函数a内部;
(2)函数a返回函数b。

这样在执行完var c = a()后,变量c实际上是指向了函数b,再执行c()后就会弹出一个窗口显示i的值(第一次为1)。
这段代码其实就创建了一个闭包,为什么?因为函数a外的变量c引用了函数a内的函数b,就是说:当函数a的内部函数b被函数a外的一个变量引用的时候,就创建了一个闭包。
匿名函数的例子:
jquery中我们经常用到这样的代码:
$("#id").click(
function(){
//do somethine
});
其中的function就是匿名函数。




匿名函数和闭包常常结合使用,是代码变得很灵活。

看一个插件的简单例子:
plugins=function(name){
var _name=name||'jack';
return {
nameWithAge:function(age){ alert(_name+"'s age is "+age); },
nameWithSex:function(sex){alert( _name+"'s sex is "+sex);}
};

}
ok,现在我要使用它了,
var myplugin=plugins('rose');

myplugin.nameWithAge(21);
输出结果:rose's age is 21
myplugin.nameWithSex('girl');
输出结果:rose's sex is girl

假如我现在有个新需求增加一个【爱好】的插件;那么你可以很灵活的扩充:

plugins=function(name){
var _name=name||'jack';
return {
nameWithAge:function(age){ alert(_name+"'s age is "+age); },
nameWithSex:function(sex){alert( _name+"'s sex is "+sex);},
nameWithHobby:function(hobby){alert(_name+"'s hobby is "+hobby);}
};

}
修改后调用插件:

myplugin.nameWithHobby('swiming');
输出结果:rose's hobby is swiming


可以看到闭包和匿名函数的结合体现了一定的灵活性。
以上只是自己的一点微不足道的经验,如果有什么不对的地方,欢迎大牛前来拍砖。

IT博客吧IT博客吧IT博客吧IT博客吧IT博客吧IT博客吧IT博客吧

原文出自:IT博客吧(http://www.itblog8.cn/javascript/20130621188.html

 

马云“吐槽”

我比马化腾和李彦宏更会管理-马云退休吐真言

5月10日退休前,马云接受商周独家专访,分享退休的艺术、管理的秘诀、痛苦的经历、自己的变化、人生的思考。马云如何成为马云?有哪些临别赠言?将如何被历史记住?为何“死都不回来”?退休后扮演更重要角色?

 

 

原文链接:IT博客吧详细文章请:看这里

马云-细节与格局哪个重要:赢在细节,输在格局

 

阿里巴巴董事局CEO马云(资料图)

  “格”是人格,“局”是胸怀,细节好的人格局一般都差,格局好的人从来不重细节,两个都干好,那叫太有才

  2011年9月10日,在第八届网商大会上,阿里巴巴集团CEO马云发表了主题演讲。马云表示,对企业家而言,注重细节与兼具胸怀是将企业做好的关键。他指出,细节能成就一个人或一家企业,但如果没有包容的胸怀,细节对成功则不再具备推动力。

原文出自:IT博客吧详细文章请看这里

 

 

oracle回滚

oracle 使用DBMS_FLASHBACK恢复意外删除的数据  

dbms_flashback
1> 获得当前SCN

select dbms_flashback.get_system_change_number from dual;
SQL> select dbms_flashback.get_system_change_number from dual;

GET_SYSTEM_CHANGE_NUMBER
------------------------
4672259


2>如果能够确切知道删除之前SCN最好,如果不知道,可以进行闪回查询尝试
1* select count(*) from wrcdb.templetdic as of scn 4672259
SQL> /

COUNT(*)
----------
0

1* select count(*) from wrcdb.templetdic as of scn 4670800
SQL> /

COUNT(*)
----------
42
3>该scn中的数据和删除前一样,就可以用该scn恢复了
SQL> edit
已写入 file afiedt.buf

1* insert into wrcdb.templetdic select * from wrcdb.templetdic as of scn 4670800
SQL> /

已创建42行。
太棒了!
delete from answer_data;
commit;
insert into answer_data select * from answer_data as of scn 1928300000;
commit;

 

//////////////////////////////////////////////////////////////////

select dbms_flashback.get_system_change_number from dual;
--------1933613010
select count(*) from COMPANY_DETECTION_PARAMETERS as of scn 1933613010;
delete from COMPANY_DETECTION_PARAMETERS;
commit;
insert into COMPANY_DETECTION_PARAMETERS select * from COMPANY_DETECTION_PARAMETERS as of scn 1933610000;
commit;
select count(*) from COMPANY_DETECTION_PARAMETERS;

 

原文出自:http://www.itblog8.cn/sql/20130612160.html

 

码农的恋爱观:只有程序员能看懂

/**
* 在加勒比海盗的初次相遇,
* so I write some
code to show my love to you.
*/
Boy
i = new Boy("BF");
Girl u = new Girl("GF");
// December 25, 2011, I told you I love you. 
i.love(u);
// but..., what you said that
meaning we are still good friends.
u.sayOtherthing();
// Since then, I ask you for the
reason.
var reason=i.ask(u);
// you say we were not understand
enough for each other .
$("body").append(reason);
// You say that it is too quickily
to turn our relation of lover.
// And take care of u and our
love.
i.takeCareOf(u);
// So I keep waiting and I have confidence that you
will.
boolean isAccept
= false;
while (!isAccept) {
i.waitFor(u);
// I think it is an important decision
// and you should forgot the unhappy things that happended before.
isAccept = 
u.thinkOver();
}
// 
After a please sound of accept, we will live happily ever
after.
u.accept(i);
i.liveHappilyWith(u);


原文出自:IT博客吧(http://www.itblog8.cn/coderthings/20130609141.html)

非小型电子商务网站数据库设计

转载自:IT博客吧:http://www.itblog8.cn/sql/20130604139.html

前言 

做了两年多针对淘宝的电子商务数据线下数据系统,越到后面越觉得自己还没入门,不管技术上还是业务上,这篇文章既是对自己的积累的一次梳理,更想的是能在和各位朋友交流中,互相进步。

ps:所有字段并不是正式项目所使用字段,请根据自己的业务需求进行酌情查看处理 

,类目属性,商品,订单结构可以参考淘宝API数据接口进行查看具体字段。
商品模块设计 

商品模块是支撑整个架构的核心,如果这块没设计好,那么所有后期的复杂的统计需求基本都满足不了。

 

为什么这样子设计属性看这里这里,把品牌从类目中剥离出来是为了降低程序针对商品属性这块的复杂度。这里通过淘宝的添加宝贝的操作来说明上面的数据结构如何满足下面的需求:

 

 

 

PS:本来要截玉兰油沐浴露的图,结果发现淘宝取消了以前选择毫升*买的多送得多组合SKU的添加商品方式,改成了一个SKU就是一个宝贝的编辑手段,呵呵,没办法,只有上面截个衣服的图,下面的数据却是快消品的。淘宝这样做这也是没办法的,这种快消品不同SKU,图片还不能用一样的,而且大部分用户搜索的时候呢,会喜欢直接搜索具体的毫升数,这也给我们提了个醒,不同的类目可能会是不一样的处理方式,就算是服装这种SKU相对标准的类目,也会有说在展示和搜索结果中,会放置一个产品的多个SKU,比如凡客的网站,一件衣服的几个颜色都会出现在类目搜索结果中,增加曝光度,吸引用户点击购买。

页面属性的编程实现可以参考这里。SKU存放在产品SKU表中,按我们的实际需求增加修改字段,比如我的表中多了ProductCode和BarCode字段,SKU的属性会拆分后存入产品基本属性值表,便于搜索或统计等需求。商品的基本属性全部打横存入商品的基本属性表中,那么SKU表的存储如下:

 

那么这个item是4013的产品在基本属性值表中的数据存储如下:

 

这里我是把所有的属性都打成一条一条存储在这个表中,那么能满足我们在日常业务的属性搜索,统计等需求。按属性搜索,这里必须要注意以下几点:

1.不可能所有的属性都开放给用户或者我们的客户进行搜索,所以我们会在属性名表中有个字段(是否搜索字段)来人工控制哪些属性是搜索属性

2.基本属性是同一个宝贝下面所有SKU都共有的,SKU属性是单个SKU独有的,所以搜索的时候还必须分清楚销售属性(销售属性组成SKU)和基本属性。

3.属性图片的存储我并没有设计,因为我们是做快消品,没有这个需求。但是,如果我做的话我还会是在基本属性值表中加上”是否图片属性,是否使用默认图片,图片URL“3个字段来记录颜色属性。做属性搜索的时候比较方便。

4.产品通过关键字搜索和属性搜索是分开的,两种搜索并不是一种解决方式,比如淘宝,在首页的搜索框是通过分词匹配宝贝标题的关键字,通过关键字的匹配程度,店铺的dsr评分权重来决定搜索结果,而属性搜索的时候则是匹配满足属性条件的宝贝。那属性又分第1点和第2点,所以还是挺麻烦的。

那到了这里产品的存储已经说完了,其它的运费什么的,就懒得说了。

这里你会发现有打包品表,打包品子表,最终商品表,商品变更记录表。这里需要详细说明一下。

首先说一下打包品概念:

打包品:为了各种运营上的需求,很多时候我们会人为的把多个SKU组合成一个商品进行组合销售,我们在淘宝购物的时候,经常会看到这样的情况,A产品+B产品组合销售,AB的组合在淘宝上面表现为一个宝贝,你看看这里或者这里或者这里,这些就是拉。这种销售数据在订单里表现是一个淘宝商品,但是我们要做库存管理,数据分析等需要拆分出来。这是必须考虑的!

PS:有那种出厂打包品,比如一个包装盒里面有香皂,有沐浴露,但是它们本身就是一个SKU,出厂就这样,所以不能和打包品混为一谈。

由于我们运营上的需要,我们可能卖单个SKU,也可能卖多个SKU的组合,那么在我这里把单个SKU和多个SKU组合都看成打包品,单个的SKU打包品它的子项只有它自己,这样做的好处是,系统中只需要一种方式来处理这种关系。在打包品表中通过类型来区分。

这里有一个关键问题要注意,我们在出售商品过程中,价格是可能会随时人工或者系统来干预变化的,比如产品A标题叫B洗发水+C护发素直降20元,但是我们根据实际的流量和转换率价格可以上下浮动,那么我们就要及时的调整价格,所以我们的标题,价格都需要进行更改,这里牵涉2个问题,我们是新建一个打包品或者我们是另外放在最终商品表,我们就需要修改对应的标题和价格,同时呢,在商品变更记录表中记录添加一个上次修改的备份,作为我们对不同价格的转换率的一个分析基础数据。第二个问题就是由于修改了打包品或者创建了新的打包品(SKU子项,SKU数量一样)价格,那么对应的分配到每个具体SKU的价格发生了变化,这里如果是新建了打包品就没问题,但是如果是修改打包品,那么我们对打包品SKU子项的价格就必须通过相应的公式进行计算。比如A+B+C今天是100元,A是30,B是50,C是20,如果价格变成了90或者110,那么对应到具体的子项价格也需要更改,因为很经常的需求就是统计某产品或者某SKU的销售量和销售额。

所以最终是我们网站上是出售打包品还是最终产品,是每次新建打包品还是修改,这要看自己权衡。但是商品的价格变更是一定要记录的。一是留备份,二是分析价格对销售的影响等等。

这样设计遇到的问题

1.起初产品维护人员意见很大,觉得很复杂,工作量很大。因为这种结构需要维护人员维护属性,并且需要他们懂一些专业知识和熟悉整个流程,各种名词搞得他们头晕,后面甚至出现了相当大的负面抵触情绪。这个没办法,因为我们这个不是网站,不是说让你简单的舒服的就能添加一个商品,这个需要掌握分类-产品-属性-打包品之间的业务关系以及这样维护的好处。解决办法:1.慢慢沟通,说明增加的工作量是为了他们在出复杂的报表的时候不需要手动去进行筛选,而且基础数据维护好了,一劳永逸。2.一定要培训好产品维护人员,让他们在有相关业务背景人员指导下能清晰的分清楚属性的意义,以及根据业务规则维护好属性基础表和录入产品等信息。

2.由于一开始数据的关联检查机制没做好,导致后面乱了很多数据,所以在后面又来花时间检验数据和建立起相应的检查机制,浪费了很多时间。

订单模块结构 

 

这里关系很简单,我想着重说明3个问题,

1个就是订单主表中存储地址库ID+买家具体地址组合成购物地址,不是依赖用户收货地址的信息,因为用户的收货地址是可能发生人为的修改的。

2.地址库的城市级别,这是一个统计维度,最好加上

3.在订单子表中,不仅存储了打包品的ID,还会把当时网站上该商品的标题和SKU名字,以及当时的价格存储进来,这是很有必要,不能直接使用关联的打包品或者商品的价格,标题,前面说了,随时可能改变的。

4.促销信息表,这里就是记录所有促销活动信息,一个商品可能对应多个促销活动,比如使用了优惠券+(满200-20)+ 满100包邮+VIP优惠10元+XXX。这样的设计是比较好的。从订单角度来看这个订单应用了多少个活动模型,能准确的抽取某种促销活动的所有订单。

不要把这种东西设置产品表中,或者与产品表进行关联,先不考虑其它原因,首先把业务模型和产品模型混在一起就乱了。

活动模块设计 

 

由于我们的订单表有订单活动促销信息表与其关联,那么实际上我们统计一般的需求只需要关联过来活动模型表就能得到说某个活动或者某类活动的数据情况,这里对于前台商品活动信息是个悲剧,一套活动缓存跟新机制,前台所有商品显示的时候和所有订单提交时检查是否满足所在时间内的活动模型来展示不同的UI。

还会有很多活动模型,这里只是列了几个,另外,必须注意一个问题,只要涉及到包邮的地方就要考虑有的地方不能包邮。也可以单独存储不包邮的城市。这个就要看业务上如何决定这个模型怎么建立。

访问跟踪模块设计和CRM

访问数据这一块我们是结合量子统计和自己跟踪的数据进行合并数据出数,这块我就过掉了,因为这块感觉我们并没有做太深,今年会单独把这块加强。我只想说,这块很重要。这一块是电子商务运营过程中的重中之中,没有他,几乎所有的带有指导性的报表数据你都没办法出。没有他就没有转换率,没有它,你就不知道站外推广的效果如何,没有他,你就不知道网站栏目,活动标题,图片等怎么去改版,甚至商品怎么放置!!!等等等等….

CRM也是一样,我们现在的弱项,我们现在着重统计用户回购率,对品牌忠诚度等一些现在业务上比较关注的面,没有钻深,但是这样光这样说好像有点太那啥,我放点收集的资料吧,这也是我们下一步努力的目标。

建立CRM的最本质目就是获取、保持和增加可获利客户(消费者)。

 

 

我认为好的数据报表展现 

我认为好的二维数据统计分析报表必要的条件:

1.多种直观的,美观的图形报表展现 (图1)

2.对应的数据表格展现 (图1)

3.对应的名词解释 (图1)

4.相关业务报表的关联查看 (图2)

5.能窜起业务流程,(图3)

图1:

 

图2:

 

图3:

 

我自认为我们公司的报表的美观,直观,清晰度都做的不错了,但是看到另一家公司的报表之后(就是后面2张),我直接给他们跪下了。相关数据一目了然,通过数据窜起了整个站点流程。多好的产品经理啊~!建议大家做的时候可以参考这家,量子后台的也不错。

后记 

现在越来越多关注运营,营销推广和各种商业模式,技术反倒变成相对不太重要的东西。路漫漫,何其修远兮,努力吧。

原文出自:http://www.cnblogs.com/mmmjiang13/archive/2012/07/05/2575538.html#2695940

 

j2ee异常处理

原文出自:IT博客吧(http://www.itblog8.cn/java/20130530133.html

在实际的j2ee项目中,系统内部难免会出现一些异常,如果把异常放任不管直接打印到浏览器可能会让用户感觉莫名其妙,也有可能让某些用户找到破解系统的方法。

  出来工作一年时间了,我也大概对异常处理有了一些了解,在这呢小弟简单介绍下个人对异常处理的见解,抛砖引玉,希望各位大神提出宝贵的意见和建议。


   就拿spring+struts2+hibernate项目说明:通常一个页面请求到后台以后,首先是到action(也就是所谓mvc的 
controller),在action层会调用业务逻辑service,servce层会调用持久层dao获取数据。最后执行结果会汇总到 
action,然后通过action控制转发到指定页面,执行流程如下图所示:


 


   而这三层其实都有可能发生异常,比如dao层可能会有SQLException,service可能会有 
NullPointException,action可能会有IOException,一但发生异常并且程序员未做处理,那么该层不会再往下执行,而是向 
调用自己的方法抛出异常,如果dao、service、action层都未处理异常的话,异常信息会抛到服务器,然后服务器会把异常直接打印到页面,结果 
就会如下图所示:


 


  其实这种错误对于客户来说毫无意义,因为他们通常是看不懂这是什么意思的。


  刚学java的 
时候,我们处理异常通常两种方法:①直接throws,放任不管;②写try...catch,在catch块中不作任何操作,或者仅仅 
printStackTrace()把异常打印到控制台。第一种方法最后就造就了上图的结果;而第二种方法更杯具:页面不报错,但是也不执行用户的请求, 
简单的说,其实这就是bug(委婉点:通常是这样)!


  那么发生异常到底应该怎么办呢?我想在大家对java异常有一定了解以后,会知道:异常应该在action控制转发之前尽量处理,同时记录log日志,然后在页面以友好的错误提示告诉用户出错了。大家看下面的代码:



  如果按照上面的方式处理异常以后,我们用户最后看到的页面可能就会是下面这种形式(我想这种错误提示应该稍微友好点了吧):


 


  然后我们回到刚才处理异常的地方,如果大家积累了一些项目经验以后会发现使用上面那种处理异常的方式可能还不够灵活:


  ①因为spring把大多数非运行时异常都转换成运行时异常(RuntimeException)最后导致程序员根本不知道什么地方应该进行try...catch操作


  ②每个方法都重复写try...catch,而且catch块内的代码都很相似,这明显做了很多重复工作而且还很容易出错,同时也加大了单元测试的用例数(项目经理通常喜欢根据代码行来估算UT case)


  ③发生异常有很多种情况:可能有数据库增删改查错误,可能是文件读写错误,等等。用户觉得每次发生异常都是“访问过程中产生错误,请重试”的提示完全不能说明错误情况,他们希望让异常信息更详尽些,比如:在执行数据删除时发生错误,这样他们可以更准确地给维护人员提供bug信息。



  如何解决上面的问题呢?我是这样做的:JDK异常或自定义异常+异常拦截器


  struts2拦截器的作用在网上有很多资料,在此不再赘述,我的异常拦截器原理如下图所示:


 


  首先我的action类、service类和dao类如果有必要捕获异常,我都会try...catch,catch块内不记录log,通常是抛出一个新异常,并且注明错误信息:




  然后在异常拦截器对异常进行处理,看下面的代码:


public String intercept(ActionInvocation actioninvocation) 
{

String result = null; // Action的返回值
try {
// 
运行被拦截的Action,期间如果发生异常会被catch住
result = 
actioninvocation.invoke();
return result;
} catch (Exception e) 
{
/**
* 处理异常
*/
String errorMsg = 
"未知错误!";
//通过instanceof判断到底是什么异常类型
if (e instanceof BaseException) 
{
BaseException be = (BaseException) e;
be.printStackTrace(); 
//开发时打印异常信息,方便调试
if(be.getMessage()!=null||Constants.BLANK.equals(be.getMessage().trim())){
//获得错误信息
errorMsg
= be.getMessage().trim();
}
} else if(e instanceof
RuntimeException){
//未知的运行时异常
RuntimeException re = 
(RuntimeException)e;
re.printStackTrace();

else{
//未知的严重异常
e.printStackTrace();
}
//把自定义错误信息
HttpServletRequest
request = (HttpServletRequest) 
actioninvocation
.getInvocationContext().get(StrutsStatics.HTTP_REQUEST);

/**

* 发送错误消息到页面
*/
request.setAttribute("errorMsg", 
errorMsg);

/**
* log4j记录日志
*/
Log log = 
LogFactory
.getLog(actioninvocation.getAction().getClass());
if
(e.getCause() != null){
log.error(errorMsg, 
e);
}else{
log.error(errorMsg, e);
}

return
"error";
}// ...end of catch
}



  需要注意的是:在使用instanceof判断异常类型的时候一定要从子到父依次找,比如BaseException继承与RuntimeException,则必须首先判断是否是BaseException再判断是否是RuntimeException。


  最后在error JSP页面显示具体的错误消息即可:




  以上方式可以拦截后台代码所有的异常,但如果出现数据库连接异常时不能被捕获的,大家可以使用struts2的全局异常处理机制来处理:




  上面这是一个很简单的异常拦截器,大家可以使用自定义异常,那样会更灵活一些。


  以上异常拦截器可以使用其它很多技术替换:比如spring aop,servlet filter等,根据项目实际情况处理。


  【补充】ajax也可以进行拦截,但是因为ajax属于异步操作,action通过response形式直接把数据返回给ajax回调函数,如果发生异常,ajax是不会执行页面跳转的,所以必须把错误信息返回给回调函数,我针对json数据的ajax是这样做的:



 

 

关于ajax原理

原文出自:IT博客吧(http://www.itblog8.cn/it/javascript/20130528132.html

在写这篇文章之前,曾经写过一篇关于AJAX技术的随笔,不过涉及到的方面很窄,对AJAX技
术的背景、原理、优缺点等各个方面都很少涉及null。
这次写这篇文章的背景是因为公司需要对内部程序员做一个培训。项目经理找到了我,并且
征询我培训的主题,考虑到之前Javascript、CSS等WEB开发技术都已经讲解过了,
所以决定针对AJAX这一块做一个比较系统的培训,所以这篇文章实际上是一个培训的材料。

在这篇文章中,我将从10个方面来对AJAX技术进行系统的讲解。

1、ajax技术的背景

不可否认,ajax技术的流行得益于google的大力推广,正是由于google
earth、google suggest以及gmail等对ajax技术的广泛应用,催生了ajax的流行。
而这也让微软感到无比的尴尬,因为早在97年,微软便已经发明了ajax中的关键技术
,并且在99年IE5推出之时,它便开始支持XmlHttpRequest对象,
并且微软之前已经开始在它的一些产品中应用ajax,比如说MSDN网站菜单中的一些应
用。遗憾的是,不知道出于什么想法,当时微软发明了ajax的核心技术之后,
并没有看到它的潜力而加以发展和推广,而是将它搁置起来。对于这一点来说,我个
人是觉得非常奇怪的,因为以微软的资源和它的战略眼光来说,
应该不会看不到ajax技术的前景,唯一的解释也许就是因为当时它的主要竞争对手Ne
tscape的消失反而使它变得麻痹和迟钝,毕竟巨人也有打盹的时候,
比如IBM曾经在对微软战略上的失误。正是这一次的失误,成就了它现在的竞争对手g
oogle在ajax方面的领先地位,
而事实上google目前在ajax技术方面的领先是微软所无法达到的,这一点在后面我讲
述ajax缺陷的时候也会提到。现在微软也意识到了这个问题,
因此它也开始在ajax领域奋起直追,比如说推出它自己的ajax框架atlas,并且在.NE
T2.0也提供了一个用来实现异步回调的接口,即ICallBack接口。
那么微软为什么对自己在ajax方面的落后如此紧张呢?现在就让我们来分析一下ajax
技术后面隐藏的深刻意义。


2、ajax技术的意义

我们在平时的开发中都多多少少的接触或者应用到了ajax,谈到ajax技术的意义,我们
关注得最多的毫无疑问是提升用户的体验。但是,
如果我们结合将来电脑和互联网的发展趋势,我们会发现ajax技
术在某些方面正好代表了这种趋势。为什么这样说呢?我们知道,自从电脑出现以来,
一直是桌面软件占据着绝对主导的地位,但是互联网的出现和成功使这一切开
始发生着微妙的变化。
相当一部分的人都相信,迟早有一天,数据和电脑软件将会从桌面转移到互联网。也就
是说,将来的电脑有可能抛弃笨重的硬盘,而直接从互 联网来获取数据和服务,
我记得我念大学的时候,有位教授给我们上课的时候,曾经设想过这样一种情景,也许
在将来的电脑桌面上,没有任何多余的软件和程序, 而仅仅只有一个IE,
虽然现在看起来我们距离这一天还很遥远,并且这其中还有很多的问题需要解决,但是
我觉得这个并非梦想,而是迟早将实现的现实。那么,
这其中的主要问题就是互联网的连接不稳定,谁也不愿意看着自己的电脑从服务器一点
一滴的下载数据,那么,ajax是不是解决了这个问题呢,
说实话,与其说ajax解决了这个问题,倒不如它只是掩盖了这个问题,它只是在服务器
和客户端之间充当了一个缓冲器,让用户误以为服务没有中断。
精确的说,ajax并不能提高从服务器端下载数据的速度,而只是使这个等待不那么令人
沮丧。但是正是这一点就足以产生巨大的影响和震动,
它实际上也对桌面软件产生了巨大的冲击。这一点我用一个例子来说明,我们可以比较
一下Outlook Express和Gmail,前者是典型的桌面软件,
后者是ajax所实现的B/S模式,实际上后者目前已经在慢慢取代前者了,Gmail在收发邮
件的时候已经和Outlook Express的功能几乎没有差别了,
而且它不需要安装客户端程序。这就是为什么微软对ajax所带来的冲击有着如此的恐惧
心理,并且在它前不久所进行的调查之中,
将google看做他们未来十年内的主要竞争对手的主要原因之一。当然,这种变化也并不
会将桌面软件全部淘汰,
现有的浏览器还没有一个能像PhotoShop等桌面程序那样处理复杂的图像。但是我们也
不能忽视它带来的影响和冲击。

3、关于ajax的名字

ajax 的全称是Asynchronous Javascript and XML,其中,Asynchronous
是异步的意思,它有别于传统web开发中采用的同步的方式。

4、关于同步和异步

异步传输是面向字符的传输,它的单位是字符;而同步传输是面向比特的传输,它的单位是
桢, 它传输的时候要求接受方和发送方的时钟是保持一致的。

具体来说,异步传输是将比特分成小组来进行传送。一般每个小组是一个8位字符,在每个小
组的头部和尾部都有一个开始位和一个停止位,
它在传送过程中接收方和发送方的时钟不要求一致,也就是说,发送方可以在任何时刻发送
这些小组,而接收方并不知道它什么时候到达。
一个最明显的例子就是计算机键盘和主机的通信,按下一个键的同时向主机发送一个8比特位
的ASCII代 码,键盘可以在任何时刻发送代码,
这取决于用户的输入速度,内部的硬件必须能够在任何时刻接收一个键入的字符。这是一个
典型的异步传输过程。 异步传输存在
一个潜在的问题,即接收方并不知道数据会在什么时候到达。在它检测到数据并做出响应之
前,第一个比特已经过去了。 这就像有人出乎意料地从后面走上来跟你说
话,而你没来得及反应过来,漏掉了最前面的几个词。因此,每次异步传输的信息都以一个
起始位开头, 它通知接收方数据已经到达了,这就给了接收方响应、接收
和缓存数据比特的时间;在传输结束时,
一个停止位表示该次传输信息的终止。按照惯例,空闲(没有传送数据)的线路实际携带着
一个代表二进制1的信号。
步传输的开始位使信号变成0,其他的比特位使信号随传输的数据信息而变化。最后,停止位
使信号重新变回1,
该信号一直保持到下一个开始位到达。例如在键盘上数字“1”,按照8比特位的扩展ASCII编码
,将发送“00110001”, 同时需要在8比特位的前面加一个起始位,后面一个停止位。

同步传输的比特分组要大得多。它不是独立地发送每个字符,每个字符都有自己的开始位和
停止位,而是把它们组合起来一起发送。 我们将这些组合称为数据帧,或简称为帧。

  数据帧的第一部分包含一组同步字符,它是一个独特的比特组合,类似于前面提到的起
始位,用于通知接收方一个帧已经到达,
但它同时还能确保接收方的采样速度和比特的到达速度保持一致,使收发双方进入同步。

  帧的最后一部分是一个帧结束标记。与同步字符一样,它也是一个独特的比特串,类似
于前面提到的停止位, 用于表示在下一帧开始之前没有别的即将到达的数据了。

  同步传输通常要比异步传输快速得多。接收方不必对每个字符进行开始和停止的操作。
一旦检测到帧同步字符,
它就在接下来的数据到达时接收它们。另外,同步传输的开销也比较少。例如,一个典型的
帧可能有500字节(即4000比特)的数据,
其中可能只包含100比特的开销。这时,增加的比特位使传输的比特总数增加2.5%,这与异步
传输中25 %的增值要小得多。
随着数据帧中实际数据比特位的增加,开销比特所占的百分比将相应地减少。但是,数据比
特位越长,缓存数据所需要的缓冲区也越大
,这就限制了一个帧的大小。另外,帧越大,它占据传输媒体的连续时间也越长。在极端的
情况下, 这将导致其他用户等得太久。

了解了同步和异步的概念之后,大家应该对ajax为什么可以提升用户体验应该比较清晰
了,它是利用异步请求方式的。打个比方, 如果现在你家里所在的小区因
某种情况而面临停水,现在有关部门公布了两种方案,一是完全停水8个小时,
在这8个小时内完全停水,8个小时后恢复正常。二是不完全停水10
个小时,在这10个小时内水没有完全断,
只是流量比原来小了很多,在10个小时后恢复正常流量,那么,如果是你你会选择哪种
方式呢?显然是后者。

5、ajax所包含的技术

大家都知道ajax并非一种新的技术,而是几种原有技术的结合体。它由下列技术组合而
成。

1.使用CSS和XHTML来表示。

2. 使用DOM模型来交互和动态显示。

3.使用XMLHttpRequest来和服务器进行异步通信。

4.使用javascript来绑定和调用。

在上面几中技术中,除了XmlHttpRequest对象以外,其它所有的技术都是基于web标准并且已
经得到了广泛使用的,
XMLHttpRequest虽然目前还没有被W3C所采纳,但是它已经是一个事实的标准,因为目前几乎
所有的主流浏览器都支持它。

6、ajax原理和XmlHttpRequest对象

  Ajax的原理简单来说通过XmlHttpRequest对象来向服务器发异步请求,从服务器获得数
据,然后用javascript来操作DOM而更新页面。这其中最关键的一步就是从服务器获得请求数
据。要清楚这个过程和原理,我们必须对 XMLHttpRequest有所了解。

 XMLHttpRequest是ajax的核心机制,它是在IE5中首先引入的,是一种支持异步请求的技术
。简单的说,也就是javascript可以及时向服务器提出请求和处理响应,而不阻塞用户。达
到无刷新的效果。

 所以我们先从XMLHttpRequest讲起,来看看它的工作原理。

 首先,我们先来看看XMLHttpRequest这个对象的属性。

  它的属性有:

  onreadystatechange 每次状态改变所触发事件的事件处理程序。

  responseText 从服务器进程返回数据的字符串形式。

  responseXML 从服务器进程返回的DOM兼容的文档数据对象。

  status
从服务器返回的数字代码,比如常见的404(未找到)和200(已就绪)

  status Text 伴随状态码的字符串信息

  readyState 对象状态值

    0 (未初始化) 对象已建立,但是尚未初始化(尚未调用open方法)

    1 (初始化) 对象已建立,尚未调用send方法

    2 (发送数据) send方法已调用,但是当前的状态及http头未知

    3 (数据传送中)
已接收部分数据,因为响应及http头不全,这时通过responseBody和responseText获取部分
数据会出现错误,

    4 (完成)
数据接收完毕,此时可以通过通过responseXml和responseText获取完整的回应数据

  但是,由于各浏览器之间存在差异,所以创建一个XMLHttpRequest对象可能需要不同的
方法。这个差异主要体现在IE和其它浏览器之间。下面是一个比较标准的创建XMLHttpReque
st对象的方法。

function CreateXmlHttp() { 
//非IE浏览器创建XmlHttpRequest对象 if
(window.XmlHttpRequest) { 
xmlhttp = new XmlHttpRequest(); 

//IE浏览器创建XmlHttpRequest对象 
if (window.ActiveXObject) { 
try { 
xmlhttp = new ActiveXObject("Microsoft.XMLHTTP"); 

catch (e) { 
try { 
xmlhttp = new ActiveXObject("msxml2.XMLHTTP"); 
} catch (ex) {


}
}
function Ustbwuyi() { 
var data = document.getElementById("username").value; 
CreateXmlHttp(); 
if (!xmlhttp)

alert("创建xmlhttp对象异常!"); 
return false; 

xmlhttp.open("POST", url, false); 
xmlhttp.onreadystatechange = function () { 
if (xmlhttp.readyState == 4) { 
document.getElementById("user1").innerHTML = "数据正在加载..."; 
if
(xmlhttp.status == 200) { 
document.write(xmlhttp.responseText); 



xmlhttp.send();
}

  如上所示,函数首先检查XMLHttpRequest的整体状态并且保证它已经完成(readyStatu
s=4),即数据已经发送完毕。然后根据服务器的设定询问请求状态,如果一切已经就绪(s
tatus=200),那么就执行下面需要的操作。

对于XmlHttpRequest的两个方法,open和send,其中open方法指定了:

a、向服务器提交数据的类型,即post还是get。

b、请求的url地址和传递的参数。
c、传输方式,false为同步,true为异步。默认为true。如果是异步通信方式(true)
,客户机就不等待服务器的响应;如果是同步方式(false),客户机就要等到服务器返
回消息后才去执行其他操作。我们需要根据实际需要来指定同步方式,在某些页面中
,可能会发出多个请求,甚至是有组织有计划有队形大规模的高强度的request,而后
一个是会覆盖前一个的,这个时候当然要指定同步方式。

Send方法用来发送请求。
  知道了XMLHttpRequest的工作流程,我们可以看出,XMLHttpRequest是完全用来向服务
器发出一个请求的,它的作用也局限于此,但它的作用是整个ajax实现的关键,因为ajax无
非是两个过程,发出请求和响应请求。并且它完全是一种客户端的技术。而XMLHttpRequest
正是处理了服务器端和客户端通信的问题所以才会如此的重要。

  现在,我们对ajax的原理大概可以有一个了解了。我们可以把服务器端看成一个数据接
口,它返回的是一个纯文本流,当然,这个文本流可以是XML格式,可以是Html,可以是Jav
ascript代码,也可以只是一个字符串。这时候,XMLHttpRequest向服务器端请求这个页面,
服务器端将文本的结果写入页面,这和普通的web开发流程是一样的,不同的是,客户端在异
步获取这个结果后,不是直接显示在页面,而是先由javascript来处理,然后再显示在页面
。至于现在流行的很多ajax控件,比如magicajax等,可以返回DataSet等其它数据类型,只
是将这个过程封装了的结果,本质上他们并没有什么太大的区别。

7、ajax的优点

1、最大的一点是页面无刷新,在页面内与服务器通信,给用户的体验非常好。

  2、使用异步方式与服务器通信,不需要打断用户的操作,具有更加迅速的响应能力。

 
3、可以把以前一些服务器负担的工作转嫁到客户端,利用客户端闲置的能力来处理,减轻服
务器和带宽的负担,节约空间和宽带租用成本。并且减轻服务器的负担,ajax的原则是“按需
取数据”,可以最大程度的减少冗余请求,和响应对服务器造成的负担。

4、基于标准化的并被广泛支持的技术,不需要下载插件或者小程序。

8、ajax的缺点

  下面我着重讲一讲ajax的缺陷,因为平时我们大多注意的都是ajax给我们所带来的好处
诸如用户体验的提升。而对ajax所带来的缺陷有所忽视。

  下面所阐述的ajax的缺陷都是它先天所产生的。

 1、ajax干掉了back按钮,即对浏览器后退机制的破坏。后退按钮是一个标准的web站点
的重要功能,但是它没法和js进行很好的合作。这是ajax所带来的一个比较严重的问题,
因为用户往往是希望能够通过后退来取消前一次操作的。那么对于这个问题有没有办法?
答案是肯定的,用过Gmail的知道,Gmail下面采用的ajax技术解决了这个问题,在Gmail
下面是可以后退的,但是,它也并不能改变ajax的机制,它只是采用的一个比较笨但是有
效的办法,即用户单击后退按钮访问历史记录时,通过创建或使用一个隐藏的IFRAME来重
现页面上的变更。(例如,当用户在Google
Maps中单击后退时,它在一个隐藏的IFRAME中进行搜索,然后将搜索结果反映到Ajax元素
上,以便将应用程序状态恢复到当时的状态。)

但是,虽然说这个问题是可以解决的,但是它所带来的开发成本是非常高的,和ajax框架所
要求的快速开发是相背离的。这是ajax所带来的一个非常严重的问题。

2、安全问题

技术同时也对IT企业带来了新的安全威胁,ajax技术就如同对企业数据建立了一个直接通道
。这使得开发者在不经意间会暴露比以前更多的数据和服务器逻辑。ajax的逻辑可以对客户
端的安全扫描技术隐藏起来,允许黑客从远端服务器上建立新的攻击。还有ajax也难以避免
一些已知的安全弱点,诸如跨站点脚步攻击、SQL注入攻击和基于credentials的安全漏洞等

3、对搜索引擎的支持比较弱。
4、破坏了程序的异常机制。至少从目前看来,像ajax.dll,ajaxpro.dll这些ajax框架
是会破坏程序的异常机制的。关于这个问题,我曾经在开发过程中遇到过,但是查了一
下网上几乎没有相关的介绍。后来我自己做了一次试验,分别采用ajax和传统的form提
交的模式来删除一条数据……给我们的调试带来了很大的困难。
5、另外,像其他方面的一些问题,比如说违背了url和资源定位的初衷。例如,我给你
一个url地址,如果采用了ajax技术,也许你在该url地址下面看到的和我在这个url地
址下看到的内容是不同的。这个和资源定位的初衷是相背离的。
6、一些手持设备(如手机、PDA等)现在还不能很好的支持ajax,比如说我们在手机的
浏览器上打开采用ajax技术的网站时,它目前是不支持的,当然,这个问题和我们没太
多关系。
9、ajax的几种框架
  目前我们采用的比较多的ajax框架主要有ajax.dll,ajaxpro.dll,magicajax.dll
以及微软的atlas框架。Ajax.dll和Ajaxpro.dll这两个框架差别不大,而magicajax.dll只是
封装得更厉害一些,比如说它可以直接返回DataSet数据集,前面我们已经说过,ajax返回的
都是字符串,magicajax只是对它进行了封装而已。但是它的这个特点可以给我们带来很大的
方便,比如说我们的页面有一个列表,而列表的数据是不断变化的,那么我们可以采用magi
cajax来处理,操作很简单,添加magicajax之后,将要更新的列表控件放在magicajax的控件
之内,然后在pageload里面定义更新间隔的时间就ok了,atlas的原理和magicajax差不多。
但是,需要注意的一个问题是,这几种框架都只支持IE,没有进行浏览器兼容方面的处理,
用反编译工具察看他们的代码就可以知道。
除了这几种框架之外,我们平时用到的比较多的方式是自己创建xmlHttpRequest对象,这种
方式和前面的几种框架相比更具有灵活性。另外,在这里还提一下aspnet2.0自带的异步回
调接口,它和ajax一样也可以实现局部的无刷新,但它的实现实际上也是基于xmlhttprequ
est对象的,另外也是只支持IE,当然这是微软的一个竞争策略。

10、ajax示例

验证用户名是否注册。

采用两种方式

  1 ajax.dll

  2 自己写xmlhttprequest对象

 

java 模式之构建模式

原文出自:IT博客吧(http://www.itblog8.cn/java/20130527130.html

构建模式和工厂模式有点类似,不过关注点不同。工厂模式往往只关心你要的是什么,二不关心这个东西的具体细节是什么。而创建模式则关心的是这个东西的具体细节的创建。

 

构建模式和工厂模式有点类似,不过关注点不同。工厂模式往往只关心你要的是什么,二不关心这个东西的具体细节是什么。而创建模式则关心的是这个东西的具体细节的创建。拿创建人物来说,我们关心的不仅是创建一个人物,还要关心他的性别,肤色和名字,则可以使用构建模式:
1.package builder; 
2.
3./**
4. * 
5. * DOC 种族角色
6. * 
7. */
8.public class Race { 
9.
10. private String name;// 名字
11.
12. private String skinColor;// 肤色
13.
14. private String sex;// 性别
15.
16. public String getName() { 
17. return this.name; 
18. } 
19.
20. public void setName(String name) { 
21. this.name = name; 
22. } 
23.
24. public String getSkinColor() { 
25. return this.skinColor; 
26. } 
27.
28. public void setSkinColor(String skinColor) { 
29. this.skinColor = skinColor; 
30. } 
31.
32. public String getSex() { 
33. return this.sex; 
34. } 
35.
36. public void setSex(String sex) { 
37. this.sex = sex; 
38. } 
39.
40.}

1.package builder; 
2.
3./**
4. * 
5. * DOC 我们关心的不仅仅是创建一个人物,还要关心其特征的创建
6. * 
7. */
8.public class RaceBuilder { 
9.
10. private Race race; 
11.
12. /**
13. * DOC 创建一个种族
14. * 
15. * @return
16. */
17. public RaceBuilder builder() { 
18. this.race = new Race(); 
19. return this; 
20. } 
21.
22. /**
23. * DOC 取名字
24. * 
25. * @return
26. */
27. public RaceBuilder setName(String name) { 
28. this.race.setName(name); 
29. return this; 
30. } 
31.
32. /**
33. * DOC 选择性别
34. * 
35. * @return
36. */
37. public RaceBuilder setSex(String sex) { 
38. this.race.setSex(sex); 
39. return this; 
40. } 
41.
42. /**
43. * DOC 选择肤色
44. * 
45. * @return
46. */
47. public RaceBuilder setSkinColor(String skinColor) { 
48. this.race.setSkinColor(skinColor); 
49. return this; 
50. } 
51.
52. /**
53. * 
54. * DOC 返回这个创建好的种族
55. * 
56. * @return
57. */
58. public Race create() { 
59. return this.race; 
60. } 
61.}

测试:
1.package builder; 
2.
3.public class Main { 
4.
5. public static void main(String[] args) { 
6. Race race = new RaceBuilder().builder().setName("张三").setSex("男").setSkinColor("白色").create(); 
7. } 
8.
9.}

 

两步掌握freemarker模板引擎之语法学习

原文出自:IT博客吧(http://itblog8.cn/album/2012120773.html) 

一、基本:
1、scalars :存储单值
字符串:简单文本由单或双引号括起来。
数字:直接使用数值。
日期:通常从数据模型获得
布尔值:true或false,通常在<#if …>标记中使用

 

2、hashes:充当其它对象的容器,每个都关联一个唯一的查询名字

 

具有一个唯一的查询名字和他包含的每个变量相关联。

 

3、sequences:充当其它对象的容器,按次序访问

 

使用数字和他包含的每个变量相关联。索引值从0开始。

 

4、集合变量:

 

除了无法访问它的大小和不能使用索引来获得它的子变量:集合可以看作只能由<#list...>指令使用的受限sequences。

 

5、方法:通过传递的参数进行计算,以新对象返回结果

 

方法变量通常是基于给出的参数计算值在数据模型中定义。

 

6、用户自定义FTL指令:宏和变换器

 

7、节点

 

节点变量表示为树型结构中的一个节点,通常在XML处理中使用。

 

模板:

 

使用FTL(freeMarker模板语言)编写

 

组成部分

 

一、整体结构

 

1、注释:<#--注释内容-->,不会输出。

 

2、文本:直接输出。

 

3、interpolation:由 ${var} 或 #{var} 限定,由计算值代替输出。

 

4、FTL标记

 

二、指令:
freemarker指令有两种:
1、预定义指令:引用方式为<#指令名称>
2、用户定义指令:引用方式为<@指令名称>,引用用户定义指令时须将#换为@。
注意:如果使用不存在的指令,FreeMarker不会使用模板输出,而是产生一个错误消息。

 

freemarker指令由FTL标记来引用,FTL标记和HTML标记类似,名字前加#来加以区分。如HTML标记的形式为<h1></h1>则FTL标记的形式是<#list></#list>(此处h1标记和list指令没有任何功能上的对应关系,只是做为说明使用一下)。

 

有三种FTL标记:
1)、开始标记:<#指令名称>
2)、结束标记:</#指令名称>
3)、空标记:<#指令名称/>

 

注意:

 

1) FTL会忽略标记之中的空格,但是,<#和指令 与 </#和指令之间不能有空格。
2)FTL标记不能够交叉,必须合理嵌套。每个开始标记对应一个结束标记,层层嵌套。如:
<#list>
<li>
${数据}
<#if 变量>
<p>game over!</p>
</#if>
</li>
</#list>

 

注意事项:
1)、FTL对大小写敏感。所以使用的标记及interpolation要注意大小写。name与NAME就是不同的对象。<#list>是正确的标记,而<#List>则不是。
2)、interpolation只能在文本部分使用,不能位于FTL标记内。如<#if${var}>是错误的,正确的方法是:<#ifvar>,而且此处var必须为布尔值。
3)、FTL标记不能位于另一个FTL标记内部,注释例外。注释可以位于标记及interpolation内部。

 


三、表达式

 

1、直接指定值:

 

1-1、字符串:
由双引号或单引号括起来的字符串,其中的特殊字符(如' "等)需要转义。

 


1-2、raw字符串:
有一种特殊的字符串称为raw字符串,被认为是纯文本,其中的和{等不具有特殊含义,该类字符串在引号前面加r,下面是一个例子:
${r"/${data}"year""}屏幕输出结果为:/${data}"year"

 


转义 含义
序列 

" 双引号(u0022)

' 单引号(u0027)

\ 反斜杠(u005C)

n 换行(u000A)

r Return (u000D)

t Tab (u0009)

b Backspace (u0008)

f Form feed (u000C)

l <

g >

a &

{ {

xCode 4位16进制Unicode代码

1-3、数字:直接输入,不需要引号

 

1)、精度数字使用“.”分隔,不能使用分组符号
2)、目前版本不支持科学计数法,所以“1E3”是错误的
3)、不能省略小数点前面的0,所以“.5”是错误的
4)、数字8、+8、08和8.00都是相同的

 

1-4、布尔值:true和false,不使用引号

 

1-5、序列:由逗号分隔的子变量列表,由[]方括号限定。
1)、子变量列表可以是表达式
2)、可以使用数字范围定义数字序列,<b>不需要方括号限定</b>,例如2..5等同于[2,3, 4, 5],但是更有效率,可以定义反递增范围如:5..2。

 

1-6、散列(hash)
1)、由逗号分隔的键/值列表,由{}大括号限定,键和值之间用冒号分隔,如:{"key1":valu1,"key2":"characterstring"....}
2)、键和值都是表达式,但是键必须是字符串。

 

2、获取变量:

 

2-1、顶层变量:${变量名}

 

变量名只能是字母、数字、下划线、$、#、@的组合,且不能以数字开头。

 

2-2、散列:有两种方法

 

1)、点语法:变量名字和顶层变量的名字受同样的限制
2)、方括号语法:变量名字无限制,可以是任意的表达式的结果
book.author.name
book.author.["name"]
book["author"].name
book["author"]["name"]
以上是等价的。

 

2-3、序列:使用散列的方括号语法获取变量,方括号中的表达式结果必须为数字。注意:第一个项目的索引为0。可以使用
[startindex..endindex]语法获取序列片段。

 

2-4、特殊变量:FreeMarker内定义变量,使用.variablename语法访问。

 

3、字符串操作

 

3-1、interpolation:使用${}或#{}在文本部分插入表达式的值,例如:

 

${"hello${username}!"}
${"${username}${username}${username}"}

 

也可以使用+来获得同样的结果:
${"hello"+username+"!"}
${username+username+username}

 

注意:${}只能用于文本部分而不能出现于标记内。

 

<#if ${user.login}>或<#if"${user.login}">都是错误的;
<#if user.login>是正确的。
本例中user.login的值必须是布尔类型。

 

3-2、子串:
举例说明:假如user的值为"Big Joe"
${user[0]}${user[4]}结果是:BJ
${user[1..4]}结果是:ig J

 

4、序列操作

 

4-1、连接操作:可以使用+来操作,例如:
["title","author"]+["month","day"]

 

5、散列操作
5-1、连接操作:可以使用+来操作,如果有相同的KEY,则右边的值会替代左边的值,例如:
{"title":散列,"author":"emma"}+{"month":5,"day":5}+{"month":6}结果month的值就是6。

 

6、算术运算

 

6-1、操作符:+、-、*、/、%
除+号以外的其他操作符两边的数据,必须都是数字类型。
如果+号操作符一边有一个字符型数据,会自动将另一边的数据转换为字符型数据,运算结果为字符型数据。

 

6-2、比较操作符:
1)、=
2)、==
3)、!=
4)、<
5)、<=
6)、>
7)、>=
1-3的操作符,两边的数据类型必须相同,否则会产生错误
4-7的操作符,对于日期和数字可以使用,字符串不可以使用。

 

注意:
1)、FreeMarker是精确比较,所以"x" "x " "X"是不等的。
2)、因为<和>对FTL来说是开始和结束标记,所以,可以用两种方法来避免这种情况:
一种是使用括号<#if (a<b)>
另一是使用替代输出,对应如下:
< lt
<= lte
> gt
>= gte

 

6-3、逻辑操作符:只能用于布尔值,否则会出现错误。

 

&&(and)与运算
||(or)或运算
!(not)非运算

 

6-4、内建函数:使用方法类似于访问散列的子变量,只是使用?代替.例如:${test?upper_case?html}

 

常用的内建函数列举如下:

 

1)、字符串使用:

 

html:对字符串进行HTML编码
cap_first:字符串第一个字母大写
lower_first:字符串第一个字母小写
upper_case:将字符串转换成大写
trim:去年字符前后的空白字符

 

2)、序列使用:
size:获得序列中元素的数目

 

3)、数字使用:
int:取得数字的整数部分

 

7、操作符的优先顺序:

 

后缀:[subbarName][subStringRange].(mathodParams)
一元:+expr、-expr、! (not)
内建:?
乘法:*、/、%
加法:+、-
关系:<、<=、>、>= (lt、lte、gt、gte)
相等:=、==、!=
逻辑与:&& (and)
逻辑或:|| (or)
数字范围:..

 

四、interpolation

 

inperpolation只能用于文本,有两种类型:通用interpolation及数字interpolation

 

1、通用interpolation

 

如${expr}

 

1-1、插入字符串值:直接输出表达式结果。
1-2、插入数字值:根据缺省格式(由setting指令设置)将表达式结果转换成文本输出;可以使用内建函数string来格式化单个interpolation

 

如:
<#setting number_format="currency" />
<#assign answer=42 />
${answer} <#-- ¥42.00 -->
${answer?string} <#-- ¥42.00 -->
${answer?string.number} <#-- 42 -->
${answer?string.currency} <#-- ¥42.00-->
${answer?string.percent} <#-- 42,00%-->

 

1-3、插入日期值:根据缺省格式(由setting指令设置)将表达式结果转换成文本输出;可以使用内建函数string来格式化单个interpolation

 

如:

 

${lastupdata?string("yyyy-MM-dd HH:mm:ss zzzz")} <#-- 2003-04-08 21:24:44 Pacific Daylight Time -->
${lastupdata?string("EEE,MMM d, ''yy")} <#--tue,Apr 8, '03 -->
${lastupdata?string("EEEE,MMMM dd, yyyy,hh:mm:ss a'('zzz')'")} <#-- Tuesday,April 08, 2003,09:24:44 PM (PDT)-->

 

1-4、插入布尔值:根据缺省格式(由setting指令设置)将表达式结果转换成文本输出;可以使用内建函数string来格式化单个interpolation

 

如:
<#assign foo=ture />
${foo?string("yes","no")} <#-- yes -->

 

2、数字interpolation:

 

有两种形式:
1)、#{expr}
2)、#{expr;format}:format可以用来格式化数字,format可以是如下:
mX:小数部分最小X位
MX:小数部分最大X位

 

例如:
<#assign x=2.582 />
<#assign y=4 />
#{x;M2} <#-- 2.58 -->
#{y;M2} <#-- 4 -->
#{x;m1} <#-- 2.582 -->
#{y;m1} <#-- 4.0 -->
#{x;m1M2} <#-- 2.58 -->
#{y;m1M2} <#-- 4.0 -->

 


杂项

 

一、用户定义指令

 

宏和变换器变量是两种不同类型的用户自定义指令,他们的区别是:

 

宏可以在模板中用macro指令来定义
变换器是在模板外由程序定义

 

1、宏:和某个变量关联的模板片段,以便在模板中通过用户自定义指令使用该变量
1-1、基本用法:
例如:
<#macro greet>
<font size="+2"> Hello JOE!</font>
</#macro>

 


使用时:
<@greet></@greet>
如果没有体内容也可以用
<@greet />

 

1-2、变量:

 

1)、可以在宏定义之后定义参数,宏参数是局部变量,只在宏定义中有效。如:

 

<#macro greet person>
<font size="+2"> Hello ${person}!</font>
</#macro>
使用时:
<@greet person="emma"> and <@greet person="LEO">
输出为:
<font size="+2"> Hello emma!</font>
<font size="+2"> Hello LEO!</font>

 

注意:宏的参数是FTL表达式,所以,person=emma和上面的例子中具有不同的意义,这意味着将变量emma的值传给person,这个值可能是任意一种数据类型,甚至是一个复杂的表达式。

 


宏可以有多个参数,使用时参数的次序是无关的,但是只能使用宏中定义的参数,并且对所有参数赋值。如:
<#macro greet person color>
<font size="+2" color="${color}"> Hello${person}!</font>
</#macro>

 

使用时:
<@greet color="black" person="emma" />正确
<@greet person="emma"/>错误,color没有赋值,此时,如果在定义宏时为color定义缺省值<#macrogreet personcolor="black">这样的话,这个使用方法就是正确的。
<@greet color="black" person="emma" bgcolor="yellow"/>错误,宏greet定义中未指定bgcolor这个参数

 

2、嵌套内容:

 

2-1、自定义指令可以有嵌套内容,使用<#nested>指令,执行自定义指令开始和结束标记之间的模板片段。例如:
<#macro greet>
<p>
<#nested>
</p>
</#macro>

 


<@greet>hello Emma!</@greet>

 

输出为
<p>hello Emma!</p>

 

2-2、<#nested>指令可以被多次调用,例如
<#macro greet>
<p>
<#nested>
<#nested>
<#nested>
<#nested>
</p>
</#macro>

 

<@greet>hello Emma!</@greet>

 

输出为
<p>
hello Emma!
hello Emma!
hello Emma!
hello Emma!
</p>

 

2-3、嵌套的内容可以是有效的FTL,例如:
<#macro welcome>
<p>
<#nested>
</p>
</#macro>

 

<#macro greet person color="black">
<font size="+2" color="${color}"> Hello${person}!</font>
</#macro>

 

<@welcome>
<@greet person="Emma" color="red" />
<@greet person="Andrew" />
<@greet person="Peter" />
</@welcome>

 

输出为:
<p>
<font size="+2" color="red"> Hello Emma!</font>
<font size="+2" color="black"> HelloAndrew!</font>
<font size="+2" color="black"> HelloPeter!</font>
</p>

 

2-4、宏定义中的局部变量对嵌套内容是不可见的,例如:

 

<#macro repeat count>
<#local y="test" />
<#list 1..count as x>
${y}${count}/${x}:<#nested />
</#list>
</#macro>

 

<@repeat count=3>
${y?default("?")}
${x?default("?")}
${count?default("?")}
</@repeat>

 

输出结果为
test 3/1:???
test 3/2:???
test 3/3:???

 

2-5、在宏定义中使用循环变量,通常用来重复嵌套内容,基本用法为:作为nested指令的参数,传递循环变量的实际值,而在调用自定义指令时,在标记的参数后面指定循环变量的名字。
例如:
<#macro repeat count>
<#list 1..count as x>
<#nested x,x/2,x==count />
</#list>
</#macro>

 

<@repeat count=4;c,halfc,last>
${c}. ${halfc}
<#if last>
last!
</#if>
</@repeat>

 

输出结果是

 

1. 0.5
2. 1
3. 1.5
4. 2last!

 

注意:指定循环变量的数目和用户定义指令开始标记指定的不同不会有问题
调用时,少指定循环变量,多指定的值会不见
调用时,多指定循环变量,多余的循环变量不会被创建

 

二、在模板中定义变量

 

1、在模板中定义的变量有三种类型

 

1-1、plain变量:可以在模板的任何地方访问,包括使用include指令插入的模板,使用assign指令创建和替换。
1-2、局部变量:在宏定义体中有效,使用local指令创建和替换。
1-3、循环变量:只能存在于指令的嵌套内容,由指令(如list)自动创建。

 

注意:
1)、宏的参数是局部变量,不是循环变量。
2)、局部变量隐藏同名的plain变量
3)、循环变量隐藏同名的plain变量和局部变量。

 

例如:

 

<#assign x="plain">
1. ${x} <#-- plain -->

 

<@test />

 

6. ${x}
<#list ["loop"] as x>
7. ${x} <#-- loop -->
<#assign x="plain2">
8. ${x} <#-- loop -->
</#list>
9. ${x} <#-- plain2 -->

 

<#macro test>
2. ${x} <#-- plain -->
<#local x="local">
3. ${x} <#-- local -->
<#list ["loop"] as x>
4. ${x} <#-- loop -->
</#list>
5. ${x} <#-- local -->
</#macro>

 

4)、内部循环变量隐藏同名的外部循环变量

 

<#list ["loop1"] as x>
${x} <#-- loop1 -->
<#list ["loop2"] as x>
${x} <#-- loop2 -->
<#list ["loop3"] as x>
${x} <#-- loop3 -->
</#list>
${x} <#-- loop2 -->
</#list>
${x} <#-- loop1 -->
</#list>

 

5)、模板中的变量会隐藏数据模型中的同名变量,如果需访问数据模型中的变量,使用特殊变量global。

 

例如:
假设数据模型中的user值为Emma
<#assign user="Man">
${user} <#-- Man -->
${.global.user}

 

网易美工了不起啊

网易美工了不起啊,lofter做的太让人喜欢了。