咖啡's profileSeparate LivesPhotosBlogLists Tools Help

Separate Lives

Both sides of the Story---Never Give up !
2/10/2009

Nickelback Gotta Be Somebody 2009-02-10 00:20:15

  
 
This time, I wonder what it feels like
To find the one in this life, the one we all dream of
But dreams just aren’t enough
So I’ll be waiting for the real thing,
I’ll know it by the feeling
The moment when we’re meeting,
will play out like a scene
Straight off the silver screen
So I’ll be holding my own breath, right up ’til the end
Until that moment when,
I find the one that I’ll spend forever with

这一次,我想知道是什么感觉
为了找到这一生,我们所拥有的梦想
但现在仅仅只是个梦想
所以要等候它实现的时候,
我就知道它的感觉
我们像一个玩的筋疲力尽的小鹿
我们需要喘息
所以我要带着我的呼吸,直到结束
直到那一刻,
我找到这种感觉了,直到永远

Cause nobody wants to be the last one there
Cause everyone wants to feel like someone cares
Someone to love with my life in their hands
There’s gotta be somebody for me like that
Cause nobody wants to do it all on their own
And everyone wants to know they’re not alone
There’s somebody else that feels the same somewhere
There’s gotta be somebody for me out there

没有人想成为最后一个
因为每个人都关心他的人
每个人的命运在自己手里
还有别人为我
没人愿意做事业都靠自己
每个人都要知道,他们就不会感到孤独
有其他人认为相同的地方
那一定是不属于这个地方

Tonight, out on the street, out in the moonlight
And dammit this means too right,
it’s just like deja vu
Me standing here with you
So I’ll be holding my own breath,
could this be the end
Is it that moment when,
I find the one that I’ll spend forever with

今晚,露宿街头,在月光下
这句话太棒了
它就像似曾相识
我只喜欢与你在一起时的感觉
所以我要留住这种感觉,
这可能是结局
直到那一刻,
我找到这种感觉了,直到永远

Cause nobody wants to be the last one there
Cause everyone wants to feel like someone cares
Someone to love with my life in their hands
There’s gotta be somebody for me like that
Cause nobody wants to do it all on their own
And everyone wants to know they’re not alone
There’s somebody else that feels the same somewhere
There’s gotta be somebody for me out there

没有人想成为最后一个
因为每个人都关心他
每个人的命运在自己手里
还有别人为我
没人愿意做事都靠自己
每个人都要知道,他们就不会感到孤独
有人不这样认为的话
那一定是不属于这个地方

You can’t give up, looking for a diamond in the rough
The wind shows up, (make sure you’re holding on)
Cause it could be the one, the one you’re waiting on
Cause nobody wants to be the last one there
And everyone wants to feel like someone cares
Someone to love with my life in their hands
There’s gotta be somebody for me, oh

你不能放弃,寻找一颗未经雕琢的钻石
风的出现,确保你的坚持
原因可能是其中之一,你还在等待
没有人想成为最后一个
因为每个人都关心他
每个人的命运在自己手里
这可以使我出人头地

Nobody wants to do it all on their own
And everyone wants to know they’re not alone
There’s somebody else that feels the same somewhere
There’s gotta be somebody for me out there
Nobody wants to be the last one there
Cause everyone wants to feel like someone cares
There’s somebody else that feels the same somewhere
There’s gotta be somebody for me out there

没人愿意做所有的事情
每个人都想要知道,他们不会孤独
有其他人认为相同的地方
那一定是不属于这个地方
没有人想过去的人站在那里
因为每个人都让他感到有人关心他
有其他人认为相同的地方
那一定是不属于这个地方
3/27/2008

几种排序算法分析

晚上没事可做,为了打发时间,无聊呀.便写了几个程序试验了一下几种快速排序算法,生成了分析报告图

 SORT_PLOT-ALMOST-SORTEDSORT_PLOT-RANDOMSORT_PLOT-REVERSESORT_PLOT-SORTEDchart

Testing performance for sets of 10000, 10 trials...
SELECTION SORTED 10000 224.8
SELECTION ALMOST-SORTED 10000 224.7
SELECTION REVERSE 10000 218.3
SELECTION RANDOM 10000 215.3
INSERTION SORTED 10000 0.0
INSERTION ALMOST-SORTED 10000 17.1
INSERTION REVERSE 10000 171.7
INSERTION RANDOM 10000 88.7
BUBBLE SORTED 10000 0.0
BUBBLE ALMOST-SORTED 10000 422.8
BUBBLE REVERSE 10000 474.2
BUBBLE RANDOM 10000 691.3
MERGE SORTED 10000 4.6
MERGE ALMOST-SORTED 10000 4.7
MERGE REVERSE 10000 3.0
MERGE RANDOM 10000 1.5
SHELLSORT-A SORTED 10000 3.1
SHELLSORT-A ALMOST-SORTED 10000 4.8
SHELLSORT-A REVERSE 10000 1.5
SHELLSORT-A RANDOM 10000 3.1
SHELLSORT-B SORTED 10000 0.0
SHELLSORT-B ALMOST-SORTED 10000 1.5
SHELLSORT-B REVERSE 10000 0.0
SHELLSORT-B RANDOM 10000 3.1
QUICKSORT-A SORTED 10000 0.0
QUICKSORT-A ALMOST-SORTED 10000 3.2
QUICKSORT-A REVERSE 10000 0.0
QUICKSORT-A RANDOM 10000 1.5
QUICKSORT-B SORTED 10000 0.0
QUICKSORT-B ALMOST-SORTED 10000 0.0
QUICKSORT-B REVERSE 10000 0.0
QUICKSORT-B RANDOM 10000 1.6
QUICKSORT-C SORTED 10000 0.0
QUICKSORT-C ALMOST-SORTED 10000 0.0
QUICKSORT-C REVERSE 10000 1.6
QUICKSORT-C RANDOM 10000 0.0
QUICKSORT-D SORTED 10000 0.0
QUICKSORT-D ALMOST-SORTED 10000 1.5
QUICKSORT-D REVERSE 10000 34.3
QUICKSORT-D RANDOM 10000 3.1

3/24/2008

How to get control of your time and your life

学会去管理时间
从你开始重视时间的时候,你的人生开始发生变化。
如果能活到70岁的话一生有25550天
时间是有限的,成功的人要学会去管理时间。

不能决定生命的长度,但我能决定生命的宽度。
时间成本是最高的,可是我们却没有计算进去。

浪费时间的原因:
1.没有明确的目标
2.拖延
3.缺乏优先顺序
4.做事有头无尾
5,重过程,不重结果
6。缺乏条理和整洁
7。不懂得授权
8。盲目行动
9。简单的问题负责化
10。不会拒绝别人的请求。

所有高科技都是为了节省时间。

管理方法:
1。策略第一
2。分清轻重缓急 20%/80%原则
3。马上行动,要不然以后你就要付出很多的代价。(是指你一定要做的事)
4。明确的目标和行动计划
5。每个人时间都是有价值的。(工作时间243天)年收入2万(每天价值82。3块)(每时10.29 块)(每分0.17块)
6。珍惜今天,当日事当日毕。每天问自己我今天生而无悔,死而无怨。
7。第一次做好,追求零缺点。(有价值的一次做好,不要反复做。一件事一定2分钟解决,不能你要说明你还要多长时间。口头的话,一定要用1分钟讲清楚。电梯原则。所有问题的解决方案都不能超过3个。)取决你定的标准。
8。要求工作时,一定要求时限。(你什么时间有空?你什么时候把电脑给我呢?-----一定要求时间)
9。及时授权,明确分工。要判断他有没有能力去做,不要叫他去做。要评估。要注重结果。
不能重复授权。要从简到繁。出现困难时要给他解决。不要姑息“倒授权 ”。你要负责。
10。养成条理和整洁的习惯。(那里拿的东西那里放)复杂问题简单化。
11。养成快速的节奏感
12。结果导向。
13。猴子管理。(告诉方法,不要让人浪费你的时间,有人的时间是没有价值的。)任何事情都要拉下面子,要敢于说不,(我现在确实是没有时间)我站起来,不要坐着。你想要我做什么。如何用零碎的时间。
14。第一胜过更好。它的价值空间是最大的。不是第一就很难生存。
15。记录每一天你做的每一件事的时间。
16。业余时间的管理。睡眠要品质要好,随时随地入睡的习惯。
1/18/2008

别让深爱变成一种疼痛

记不得是从哪里看到了这句话:如果你不爱一个人,请放手,好让别人有机会爱她。如果你爱的人放弃了你,请放开自己,好让自己有机会爱别人。这话直白但很有道理,也从一个侧面教会了人们如何对待情感。 
  有的东西你再喜欢也不会属于你的,有的东西你再留恋也注定要放弃的,爱是人生中一首永远也唱不完的歌。人一生中也许会经历许多种爱,但千万别让爱成为一种伤害。 
  生活中到处都存在着缘分,缘聚缘散好像都是命中注定的事情;有些缘分一开始就注定要失去,有些缘分是永远都不会有好结果;可是我却偏偏渴望创造一种奇迹。爱一个人不一定要拥有,但拥有一个人就一定要好好的去爱她。话说着容易,可一旦做时就真的很难,不信你试试。 
  如果真诚是一种伤害,请选择谎言;如果谎言是一种伤害,请选则沉默;如果沉默是一种伤害,请选择离开。如果爱是一种伤害,请不要靠近。可是好多的情况下并不是如此,因为不由得你选择。 
  如果失去是苦,你怕不怕付出?如果痴迷是苦,你会不会选择结束?如果追求是苦,你会不会选择执迷不悟?如果分离是苦,你要向谁倾诉?好多事情都是后来才看清楚,好多事情当时一点也不觉得苦,然而我已经找不到来时的路。 
  有一种爱,明明是深爱,却表达不完美。有一种爱,明知道要放弃,却不甘心就此离开。有一种爱,明知是煎熬,却又躱不掉。有一种爱,明知无前路,心却早已收不回来。 
  爱情不是游戏,因为我们玩不起它。爱是真心付出,要忘记真地做不到。不管归处将是哪里,我想都该在心底留有一份纯真的美好。从来没有轻易对别人动心,突然发现自己深深地爱上了你,那种滋味真是难以用言语表达,是喜悦?是悲哀?你叫我忘记,难道爱说收就可以收得回吗?可以的话也不叫爱了。 
  也许我没有足够的勇气面对现实的残酷,那么什么是勇气?是哭着要你爱我?还是哭着让你离开?估计此时没有一个正确的答案。 
  男人的自信来自一个女人对他的崇拜,女人的高傲来自一个男人对她的倾慕。那么为什么我们总是不懂得珍惜眼前人?在未可预知的重逢里,我们以为总会重逢,总会有缘再会,总以为有机会说一声对不起,却从没想过每一次挥手道别,都可能是诀离。 
  我常常有如此的感慨,也许爱情只是因为寂寞,需要找一个人来爱,即使没有任何结局。可是爱为什么也如此的脆弱?有时它易碎的程度比玻璃花瓶还容易。它又如此的坚强,坚强到即便已把自己弄的遍体鳞伤,依然痴心的爱着,从不后悔。 
  爱可以是一瞬间的事情,也可以是一辈子的事情。每个人都可以在不同的时间爱上不同的人,为什么我的爱就这么一次呢?我也知道不是谁离开了谁就无法生活,可是要真正的遗忘却是一件万难的事情,也许正因如此我才不够坚强。 
  现在的一切,看似不经意,却是我苦心经营的结果,此刻我特别希望来一场风雨,因为那样我身在其中,即使泪流满面也不会被人发现。 
  世事难料,其实凡事都是在它适当的时候降临,只是我们没有适当的心情去迎接它或是没在意。正如有人说:无论是谁只要在错的时间里做了对的事,其结果是可想而知的,其代价是显而意见的。 
  因为爱所以离开,因为爱所以放弃。听起来这话很伟大,很洒脱,可是有谁为了爱真正的能够离开呢?也许你能,我却不能轻易做到放弃。尽管有些感情如此直接和残酷,容不下任何迂回曲折的温暖.......... 
  晚风轻轻地吹过,心田慢慢在放松,可我对你的牵挂却依然无法放下。一个人不寂寞,想一个人才寂寞。心情随着风儿在空中轻轻的飘荡,不再去想那么多的事情,于是,只好放纵自己的大脑,任夜风带着我的思绪随处飘荡。 
  感受着你的离去,心里有一种刺痛,霎时间内心变得空荡荡的,感觉人生真的了无意义。其实,自己也很明白你的想法,你的处境,只是太牵挂一个人的时候,爱也会成为一种负担。 
  如果你真的爱了,那么不要轻言放弃,即便她让你伤心了,试着去牵挂她,倾听她,让她明白你依然关爱她;如果你真的爱了,那么不要轻言放弃,即便她让你失望了,试着去包容她,让她知道你依然在乎她。爱情真的是个很奇妙的东西,具有无穷的魔力,让人为之着迷。我坚信,爱一个人 ,就会爱她的所有,不会因为一些世俗的东西而改变。虽然明白,喜欢一个人并为她付出一切,也许这付出没有收获,许多故事也都是没有结局的,但是,我依然甘心付出我的努力,尽量不让她受到伤害。

有人说过:人生难得放纵自己一次,那么就让我放纵一下自己的感情吧,不必在意结果,当真心爱过之后,就会淡然的去面对人生的很多挫折。 经历了一段过去,一直以来,我都是这样认! 
  遇到你最爱的人,然后体会到爱的感觉; 
  因为了解被爱的感觉,所以才能发现最爱你的人 
  当你经历过爱人与被爱,学会了爱,才会知道什么是你需要的,也才会找到最适合你,能够相处一辈子的人。 你最爱的,往往没有选择你; 
  最爱你的,往往不是你最爱的; 
  而最长久的,偏偏不是你最爱也不是最爱你的, 
  只是在最适合的时间出现的那个人。 没有人是故意要变心的,他爱你的时候是真的爱你, 
  可是他不爱你的时候也是真的不爱你了, 
  他爱你的时候没有办法假装不爱你; 
  同样的,他不爱你的时候也没有办法假装爱你 。 
  当一个人不爱你要离开你, 
  你要问自己还爱不爱他, 
  如果你也不爱他了,千万别为了可怜的自尊而不肯离开; 
  如果你还爱他,你应该会希望他过得幸福快乐, 
  希望他跟真正爱的人在一起,绝不会阻止, 
  你要是阻止他得到真正的幸福,就表示你已经不爱他了, 
  而如果你不爱他,你又有什么资格指责他变心呢? 爱不是占有, 
  你喜欢月亮,不可能把月亮拿下来放在脸盆里, 
  但月亮的光芒仍可照进你的房间。 
  换句话说,你爱一个人,也可以用另一种方式拥有, 
  让爱人成为生命里的永恒回忆, 
  如果你真爱一个人,就要爱他原来的样子─爱他的好,也爱他的坏: 
  爱他的优点,也爱他的缺点, 
  绝不能因为爱他,就希望他变成自己所希望的样子, 
  万一变不成就不爱他了。 
  真正爱一个人是无法说出原因的, 
  你只知道无论何时何地、心情好坏,你都希望这个人陪著你; 
  真正的感情是两人能在最艰苦中相守,也就是没有丝毫要求。 
  毕竟,感情必须付出,而不是只想获得; 
  分开是一种必然的考验, 
  如果你们感情不够稳固,只好认输!

10/6/2007

有点“找不着北”的感觉

一直有点“找不着北”的感觉,尽看着日子一天天地过,都不知道究竟想要什么样得生活!

    今天自己闷在屋子里,空暇之余,读到一位成功人士的经历深受启发.茫然间产生对未来的思考。转眼间,自己从事IT已经一年了有许多收获,也留下了不少遗憾。收获是:从学生转变为职业人。遗憾是:读书太少:一直没有完完整整地读几本书。虽然距离现在才一年光景,不过就这一年时间却足够令我的心情浮浮沉沉、起伏不定:“ 自信--激情--失落--希望--奋斗--懒惰--迷茫--郁闷 ”,用这条路线来形容我一年来的变化可能再适合不过了。
    毕业到现在也有一年了,看着身边的朋友换工作的换工作、升职的升职、加工资的加工资,找女朋友的找女朋友,好象每个人都忙得不亦乐乎,惟独我依旧还是孤独的一人。相比之下,我是很失败的。可就在我真正静下心来想好好考虑下是否该换份新工作或者为自己的将来为个人发展问题作打算的时候,我才猛然发觉原来自己浪费了不少时间。整整一年时间里,都搞不清楚究竟学到了什么又失去了什么,这不能不说是我的另一种悲哀。

    在有一次和朋友的聊天过程中,他的高瞻远瞩他的个人远见让我觉得自己确实很惭愧。记得他说:“虽然现在有了份较安定的工作,每个月也有固定的工资可以拿,但是如果不求发展的话,5年后或者10年后又能怎样呢?试想如果没办法做到主管、经理或者老板之类的话。”当时我还笑他太过于杞人忧天了?没想到他现在工作的目的比我高了很多不仅是为了生活更为了未来的人生。你说,相比之下我能不惭愧么?这不得不让我好好考虑下人生的意义了。

    我想是时候该改变下方向了,人虽不变环境也不变但心态却变了,也许迷惘与郁闷仍将继续纠缠着我,但我却不能在一味的永远沉默下去。

   “天下本无事,烦人自恼之”,也许你会觉得我确实是想得太多了,好端端地工作干吗要胡思乱想,这不是给自己施加没必要的压力给自己添乱么?

   呜呼唉哉!还是让自己好好沉静一下吧,最好闭上眼睛,什么都不要想,耳朵里塞着MP3,任凭音乐飘散,那是最舒服不过的事了。人没变,环境也没变,只是听的音乐该换下而已。记得很久没听过许巍了,或许该好好听上一把,也许只有他的歌声才能让我重新找回思考的方向……

                                                                  仅以此文纪念我已逝去的青春 --- 田波 于2007-10-06  晚01:15

8/23/2007

exjs and dwr and spring

ExtJs 在Web UI已经获得很大的殊荣,但单凭借它 Client UI 还无法霸占BS 那么丰富的应用。。他还要选择 一个Web层与后台一起来交互完成一个完美的系统。

Extjs的天生丽质的, 完美了融合JQuery,Prototype,YUI, 和她结合当然也需要出类拔萃,生出名门。 如果采用Java为开发主语言,那么 ExtJS + DWR + SPRING 算是门当户对。

可惜Extjs 的DataStore 的 prxy 并没有DWR一席,因为DWR返回的是Java对象与JSON非常像, 然而DWR 的在于它与Web远程Java方法的直接会话,因此有着非常灵活的一面。所以,有时候使用DWR对于J2EE来说,要更好于JSON,他做了JSON做不到事.

extjs官方论坛里已经有人编写了这样的扩展,后经人丰富,现在基本能满足要求了。
DWR扩展代码如下,创建 Ext.data.DWRProxy类。同样也继承Ext.data.DataProxy ,拿来show着解读一下。

dwr.js

Ext.data.DWRProxy = function(dwrCall, pagingAndSort){
Ext.data.DWRProxy.superclass.constructor.call(this);
this.dwrCall = dwrCall;
//this.args = args;
this.pagingAndSort = (pagingAndSort!=undefined ? pagingAndSort : true);
};

Ext.extend(Ext.data.DWRProxy, Ext.data.DataProxy, {
load : function(params, reader, callback, scope, arg) {
if(this.fireEvent("beforeload", this, params) !== false) {
var sort;
if(params.sort && params.dir) sort = params.sort + ' ' + params.dir;
else sort = '';
var delegate = this.loadResponse.createDelegate(this, [reader, callback, scope, arg], 1);
var callParams = new Array();
if(arg.arg) {
callParams = arg.arg.slice();
}

if(this.pagingAndSort) {
callParams.push(params.start);
callParams.push(params.limit);
callParams.push(sort);
}

callParams.push(delegate);
this.dwrCall.apply(this, callParams);
} else {
callback.call(scope || this, null, arg, false);
}
},

loadResponse : function(listRange, reader, callback, scope, arg) {
var result;
try {
result = reader.read(listRange);
} catch(e) {
this.fireEvent("loadexception", this, null, response, e);
callback.call(scope, null, arg, false);
return;
}
callback.call(scope, result, arg, true);
},

update : function(dataSet){},

updateResponse : function(dataSet)
{}
});

Ext.data.ListRangeReader = function(meta, recordType){
Ext.data.ListRangeReader.superclass.constructor.call(this, meta, recordType);
this.recordType = recordType;
};
Ext.extend(Ext.data.ListRangeReader, Ext.data.DataReader, {
getJsonAccessor: function(){
var re = /[\[\.]/;
return function(expr) {
try {
return(re.test(expr))
? new Function("obj", "return obj." + expr)
: function(obj){
return obj[expr];
};
} catch(e){}
return Ext.emptyFn;
};
}(),

read : function(o){
var recordType = this.recordType, fields = recordType.prototype.fields;

//Generate extraction functions for the totalProperty, the root, the id, and for each field
if (!this.ef) {
if(this.meta.totalProperty) {
this.getTotal = this.getJsonAccessor(this.meta.totalProperty);
}

if(this.meta.successProperty) {
this.getSuccess = this.getJsonAccessor(this.meta.successProperty);
}

if (this.meta.id) {
var g = this.getJsonAccessor(this.meta.id);
this.getId = function(rec) {
var r = g(rec);
return (r === undefined || r === "") ? null : r;
};
} else {
this.getId = function(){return null;};
}
this.ef = [];
for(var i = 0; i < fields.length; i++){
f = fields.items[i];
var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
this.ef[i] = this.getJsonAccessor(map);
}
}

var records = [];
var root = o.data, c = root.length, totalRecords = c, success = true;

if(this.meta.totalProperty){
var v = parseInt(this.getTotal(o), 10);
if(!isNaN(v)){
totalRecords = v;
}
}

if(this.meta.successProperty){
var v = this.getSuccess(o);
if(v === false || v === 'false'){
success = false;
}
}

for(var i = 0; i < c; i++){
var n = root[i];
var values = {};
var id = this.getId(n);
for(var j = 0; j < fields.length; j++){
f = fields.items[j];
var v = this.ef[j](n);
values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
}
var record = new recordType(values, id);
records[i] = record;
}

return {
success : success,
records : records,
totalRecords : totalRecords
};
}
});



经过改造之后,那么DWR可以像JSON一样,返回的数据作为DataStore
代码片断

ds = new Ext.data.Store({
proxy: new Ext.data.DWRProxy(TodoService.getItems, true), //TotoService.getItems dwr开放出来的Java获取数据方法
reader: new Ext.data.ListRangeReader(
{id:'id', totalProperty:'totalSize'}, recordType),
remoteSort: true
});



Spring实现了强大的管理了后台的JavaBean,干干净净的注入创建了每个对象, DWR天生又是支持Spring,无缝的讲Spring的后台Bean 发布到Web层作为JavaScript对象,和JavaScript方法。中途不需要人工干预写Servlet.


不多说了, 跑了下代码,通过! 
7/25/2007

电影 K 歌情人 Way Back Into Love

 Way Back Into Love

I have been sleeping with a cloud above my bed I have been living with a shadow overhead I have been lonely for so long Trapped in the past, I just can not seem to move on I have been hiding all my hopes and dreams away
Just in case I ever need them again someday I have been setting aside time
To clear a little space in the corners of my mind All I want to do is find a way back into love I can not make it through without a way back into love
Oh oh oh
I have been watching but the stars refuse to shine I have been searching but I just do not see the signs I know that it is out there There is got to be something for my soul somewhere I have been looking for someone to shed some light Not just somebody just to get me throught the night
I could use some direction And I am open to your suggestions
All I want to do is find a way back into love I can not make it through without a way back into love And if I open my heart againI guess I am hoping you will be there for me in the end There are moments when I do not know if it is real
Or if anybody feels the way I feel I need inspiration
Not just another negotiation All I want to do is find a way back into love
I can not make it through without a way back into love And if I open my heart to you I am hoping you will show me what to do
And if you help me to start again You know that I will be there for you in the end 

  

翻译了一下....嘿嘿...大体意思是...

你知道最后我会在那里等你 我活在阴影之下 乌云伴我共枕 一直孤单好久
困在过去,似乎就是无法往前走 我把所有的希望和梦想藏起来
只为了有一天可能又需要他们 我总留些时间 净化心里的小角落 我只要重新找到爱的感觉 若没有爱的感觉我办不到 我留心观察然而繁星不愿发亮
我一直搜寻但看不到任何征兆 我知道它不在那里 但一定有个为我灵魂而生的东西在某处 我一直寻找着让我发光的人 而非只是个陪我过夜的家伙 我能使用一些指示 而且我可能是你的暗示 我只要找到重新爱的感觉
若没有爱的感觉我办不到 我想我希望最后你会在那里等我 有时候我不确定它是不是真的 或许有人和我心有戚焉 我需要妙计 而非又一个妥协 我只要重新找到爱的感觉 若没有爱的感觉我办不到 若我为你打开心房 希望你会告诉我该怎么做 若你帮我重新开始

7/6/2007

追MM与程序模式

 1、FACTORY一追MM少不了请吃饭了,麦当劳的鸡翅和肯德基的鸡翅都是MM爱吃的东西,虽然口味有所不同,但不管你带MM去麦当劳或肯德基,只管向服务员说“来四个鸡翅”就行了。麦当劳和肯德基就是生产鸡翅的Factory

    工厂模式:客户类和工厂类分开。消费者任何时候需要某种产品,只需向工厂请求即可。消费者无须修改就可以接纳新产品。缺点是当产品修改时,工厂类也要做相应的修改。如:如何创建及如何向客户端提供。

    2、BUILDER—MM最爱听的就是“我爱你”这句话了,见到不同地方的MM,要能够用她们的方言跟她说这句话哦,我有一个多种语言翻译机,上面每种语言都有一个按键,见到MM我只要按对应的键,它就能够用相应的语言说出“我爱你”这句话了,国外的MM 也可以轻松搞掂,这就是我的“我爱你”builder。(这一定比美军在伊拉克用的翻译机好卖)

    建造模式:将产品的内部表象和产品的生成过程分割开来,从而使一个建造过程生成具有不同的内部表象的产品对象。建造模式使得产品内部表象可以独立的变化,客户不必知道产品内部组成的细节。建造模式可以强制实行一种分步骤进行的建造过程。

    3、FACTORY METHOD—请MM去麦当劳吃汉堡,不同的MM有不同的口味,要每个都记住是一件烦人的事情,我一般采用Factory?Method模式,带着MM到服务员那儿,说“要一个汉堡”,具体要什么样的汉堡呢,让MM直接跟服务员说就行了。

    工厂方法模式:核心工厂类不再负责所有产品的创建,而是将具体创建的工作交给子类去做,成为一个抽象工厂角色,仅负责给出具体工厂类必须实现的接口,而不接触哪一个产品类应当被实例化这种细节。

    4、PROTOTYPE—跟MM用QQ聊天,一定要说些深情的话语了,我搜集了好多肉麻的情话,需要时只要copy出来放到QQ里面就行了,这就是我的情话prototype了。(100块钱一份,你要不要)

    原始模型模式:通过给出一个原型对象来指明所要创建的对象的类型,然后用复制这个原型对象的方法创建出更多同类型的对象。原始模型模式允许动态的增加或减少产品类,产品类不需要非得有任何事先确定的等级结构,原始模型模式适用于任何的等级结构。缺点是每一个类都必须配备一个克隆方法。

    5、SINGLETON—俺有6个漂亮的老婆,她们的老公都是我,我就是我们家里的老公Sigleton,她们只要说道“老公”,都是指的同一个人,那就是我(刚才做了个梦啦,哪有这么好的事)

    单例模式:单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例单例模式。单例模式只应在有真正的“单一实例”的需求时才可使用。

    结构型模式

    6、ADAPTER—在朋友聚会上碰到了一个美女Sarah,从香港来的,可我不会说粤语,她不会说普通话,只好求助于我的朋友kent了,他作为我和Sarah之间的Adapter,让我和Sarah可以相互交谈了(也不知道他会不会耍我)

    适配器(变压器)模式:把一个类的接口变换成客户端所期待的另一种接口,从而使原本因接口原因不匹配而无法一起工作的两个类能够一起工作。适配类可以根据参数返还一个合适的实例给客户端。

    7、BRIDGE—早上碰到MM,要说早上好,晚上碰到MM,要说晚上好;碰到MM穿了件新衣服,要说你的衣服好漂亮哦,碰到MM新做的发型,要说你的头发好漂亮哦。不要问我“早上碰到MM新做了个发型怎么说”这种问题,自己用BRIDGE组合一下不就行了

    桥梁模式:将抽象化与实现化脱耦,使得二者可以独立的变化,也就是说将他们之间的强关联变成弱关联,也就是指在一个软件系统的抽象化和实现化之间使用组合/聚合关系而不是继承关系,从而使两者可以独立的变化。

    8、COMPOSITE—Mary今天过生日。“我过生日,你要送我一件礼物。”“嗯,好吧,去商店,你自己挑。”“这件T恤挺漂亮,买,这条裙子好看,买,这个包也不错,买。”“喂,买了三件了呀,我只答应送一件礼物的哦。”“什么呀,T恤加裙子加包包,正好配成一套呀,小姐,麻烦你包起来。”“……”,MM都会用Composite模式了,你会了没有?

    合成模式:合成模式将对象组织到树结构中,可以用来描述整体与部分的关系。合成模式就是一个处理对象的树结构的模式。合成模式把部分与整体的关系用树结构表示出来。合成模式使得客户端把一个个单独的成分对象和由他们复合而成的合成对象同等看待。

    9、DECORATOR—Mary过完轮到Sarly过生日,还是不要叫她自己挑了,不然这个月伙食费肯定玩完,拿出我去年在泰山顶上照的照片,在背面写上“最好的的礼物,就是爱你的Fita”,再到街上礼品店买了个像框(卖礼品的MM也很漂亮哦),再找隔壁搞美术设计的Mike设计了一个漂亮的盒子装起来……,我们都是Decorator,最终都在修饰我这个人呀,怎么样,看懂了吗?

    装饰模式:装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案,提供比继承更多的灵活性。动态给一个对象增加功能,这些功能可以再动态的撤消。增加由一些基本功能的排列组合而产生的非常大量的功能。

    10、FACADE—我有一个专业的Nikon相机,我就喜欢自己手动调光圈、快门,这样照出来的照片才专业,但MM可不懂这些,教了半天也不会。幸好相机有Facade设计模式,把相机调整到自动档,只要对准目标按快门就行了,一切由相机自动调整,这样 MM也可以用这个相机给我拍张照片了。

    门面模式:外部与一个子系统的通信必须通过一个统一的门面对象进行。门面模式提供一个高层次的接口,使得子系统更易于使用。每一个子系统只有一个门面类,而且此门面类只有一个实例,也就是说它是一个单例模式。但整个系统可以有多个门面类。11、FLYWEIGHT—每天跟MM发短信,手指都累死了,最近买了个新手机,可以把一些常用的句子存在手机里,要用的时候,直接拿出来,在前面加上MM的名字就可以发送了,再不用一个字一个字敲了。共享的句子就是Flyweight,MM的名字就是提取出来的外部特征,根据上下文情况使用。

    享元模式:FLYWEIGHT在拳击比赛中指最轻量级。享元模式以共享的方式高效的支持大量的细粒度对象。享元模式能做到共享的关键是区分内蕴状态和外蕴状态。内蕴状态存储在享元内部,不会随环境的改变而有所不同。外蕴状态是随环境的改变而改变的。外蕴状态不能影响内蕴状态,它们是相互独立的。将可以共享的状态和不可以共享的状态从常规类中区分开来,将不可以共享的状态从类里剔除出去。客户端不可以直接创建被共享的对象,而应当使用一个工厂对象负责创建被共享的对象。享元模式大幅度的降低内存中对象的数量。

    12、PROXY —跟MM在网上聊天,一开头总是“hi,你好”,“你从哪儿来呀?”“你多大了?”“身高多少呀?”这些话,真烦人,写个程序做为我的Proxy吧,凡是接收到这些话都设置好了自动的回答,接收到其他的话时再通知我回答,怎么样,酷吧。?

    代理模式:代理模式给某一个对象提供一个代理对象,并由代理对象控制对源对象的引用。代理就是一个人或一个机构代表另一个人或者一个机构采取行动。某些情况下,客户不想或者不能够直接引用一个对象,代理对象可以在客户和目标对象直接起到中介的作用。客户端分辨不出代理主题对象与真实主题对象。代理模式可以并不知道真正的被代理对象,而仅仅持有一个被代理对象的接口,这时候代理对象不能够创建被代理对象,被代理对象必须有系统的其他角色代为创建并传入。

    行为模式

    13、CHAIN?OF?RESPONSIBLEITY—晚上去上英语课,为了好开溜坐到了最后一排,哇,前面坐了好几个漂亮的MM哎,找张纸条,写上“Hi,可以做我的女朋友吗?如果不愿意请向前传”,纸条就一个接一个的传上去了,糟糕,传到第一排的M M把纸条传给老师了,听说是个老处女呀,快跑!

    责任链模式:在责任链模式中,很多对象由每一个对象对其下家的引用而接

    起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理此请求。客户并不知道链上的哪一个对象最终处理这个请求,系统可以在不影响客户端的情况下动态的重新组织链和分配责任。处理者有两个选择:承担责任或者把责任推给下家。一个请求可以最终不被任何接收端对象所接受。

    14、COMMAND—俺有一个MM家里管得特别严,没法见面,只好借助于她弟弟在我们俩之间传送信息,她对我有什么指示,就写一张纸条让她弟弟带给我。这不,她弟弟又传送过来一个COMMAND,为了感谢他,我请他吃了碗杂酱面,哪知道他说:“我同时给我姐姐三个男朋友送COMMAND,就数你最小气,才请我吃面。”,:-(

    命令模式:命令模式把一个请求或者操作封装到一个对象中。命令模式把发出命令的责任和执行命令的责任分割开,委派给不同的对象。命令模式允许请求的一方和发送的一方独立开来,使得请求的一方不必知道接收请求的一方的接口,更不必知道请求是怎么被接收,以及操作是否执行,何时被执行以及是怎么被执行的。系统支持命令的撤消。

    15、INTERPRETER—俺有一个《泡MM真经》,上面有各种泡MM的攻略,比如说去吃西餐的步骤、去看电影的方法等等,跟MM约会时,只要做一个Interpreter,照着上面的脚本执行就可以了。

    解释器模式:给定一个语言后,解释器模式可以定义出其文法的一种表示,并同时提供一个解释器。客户端可以使用这个解释器来解释这个语言中的句子。解释器模式将描述怎样在有了一个简单的文法后,使用模式设计解释这些语句。在解释器模式里面提到的语言是指任何解释器对象能够解释的任何组合。在解释器模式中需要定义一个代表文法的命令类的等级结构,也就是一系列的组合规则。每一个命令对象都有一个解释方法,代表对命令对象的解释。命令对象的等级结构中的对象的任何排列组合都是一个语言。

    16、ITERATOR—我爱上了Mary,不顾一切的向她求婚。

    Mary:“想要我跟你结婚,得答应我的条件”

    我:“什么条件我都答应,你说吧”

    Mary:“我看上了那个一克拉的钻石”

    我:“我买,我买,还有吗?”

    Mary:“我看上了湖边的那栋别墅”

    我:“我买,我买,还有吗?”

    Mary:“你的小弟弟必须要有50cm长”

    我脑袋嗡的一声,坐在椅子上,一咬牙:“我剪,我剪,还有吗?”

    ……

    迭代子模式:迭代子模式可以顺序访问一个聚集中的元素而不必暴露聚集的内部表象。多个对象聚在一起形成的总体称之为聚集,聚集对象是能够包容一组对象的容器对象。迭代子模式将迭代逻辑封装到一个独立的子对象中,从而与聚集本身隔开。迭代子模式简化了聚集的界面。每一个聚集对象都可以有一个或一个以上的迭代子对象,每一个迭代子的迭代状态可以是彼此独立的。迭代算法可以独立于聚集角色变化。

    17、MEDIATOR—四个MM打麻将,相互之间谁应该给谁多少钱算不清楚了,幸亏当时我在旁边,按照各自的筹码数算钱,赚了钱的从我这里拿,赔了钱的也付给我,一切就OK啦,俺得到了四个MM的电话。

    调停者模式:调停者模式包装了一系列对象相互作用的方式,使得这些对象不必相互明显作用。从而使他们可以松散偶合。当某些对象之间的作用发生改变时,不会立即影响其他的一些对象之间的作用。保证这些作用可以彼此独立的变化。调停者模式将多对多的相互作用转化为一对多的相互作用。调停者模式将对象的行为和协作抽象化,把对象在小尺度的行为上与其他对象的相互作用分开处理。18、MEMENTO—同时跟几个MM聊天时,一定要记清楚刚才跟MM说了些什么话,不然MM发现了会不高兴的哦,幸亏我有个备忘录,刚才与哪个MM说了什么话我都拷贝一份放到备忘录里面保存,这样可以随时察看以前的记录啦。

    备忘录模式:备忘录对象是一个用来存储另外一个对象内部状态的快照的对象。备忘录模式的用意是在不破坏封装的条件下,将一个对象的状态捉住,并外部化,存储起来,从而可以在将来合适的时候把这个对象还原到存储起来的状态。

    19、OBSERVER—想知道咱们公司最新MM情报吗?加入公司的MM情报邮件组就行了,tom负责搜集情报,他发现的新情报不用一个一个通知我们,直接发布给邮件组,我们作为订阅者(观察者)就可以及时收到情报啦

    观察者模式:观察者模式定义了一种一队多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使他们能够自动更新自己。

    20、STATE—跟MM交往时,一定要注意她的状态哦,在不同的状态时她的行为会有不同,比如你约她今天晚上去看电影,对你没兴趣的MM就会说“有事情啦”,对你不讨厌但还没喜欢上的MM就会说“好啊,不过可以带上我同事么?”,已经喜欢上你的MM就会说“几点钟?看完电影再去泡吧怎么样?”,当然你看电影过程中表现良好的话,也可以把MM的状态从不讨厌不喜欢变成喜欢哦。

    状态模式:状态模式允许一个对象在其内部状态改变的时候改变行为。这个对象看上去象是改变了它的类一样。状态模式把所研究的对象的行为包装在不同的状态对象里,每一个状态对象都属于一个抽象状态类的一个子类。状态模式的意图是让一个对象在其内部状态改变的时候,其行为也随之改变。状态模式需要对每一个系统可能取得的状态创立一个状态类的子类。当系统的状态变化时,系统便改变所选的子类。

    21、STRATEGY—跟不同类型的MM约会,要用不同的策略,有的请电影比较好,有的则去吃小吃效果不错,有的去海边浪漫最合适,单目的都是为了得到MM的芳心,我的追MM锦囊中有好多Strategy哦。

    策略模式:策略模式针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以相互替换。策略模式使得算法可以在不影响到客户端的情况下发生变化。策略模式把行为和环境分开。环境类负责维持和查询行为类,各种算法在具体的策略类中提供。由于算法和环境独立开来,算法的增减,修改都不会影响到环境和客户端。

    22、TEMPLATE?METHOD——看过《如何说服女生上床》这部经典文章吗?女生从认识到上床的不变的步骤分为巧遇、打破僵局、展开追求、接吻、前戏、动手、爱抚、进去八大步骤(Template method),但每个步骤针对不同的情况,都有不一样的做法,这就要看你随机应变啦(具体实现);

    模板方法模式:模板方法模式准备一个抽象类,将部分逻辑以具体方法以及具体构造子的形式实现,然后声明一些抽象方法来迫使子类实现剩余的逻辑。不同的子类可以以不同的方式实现这些抽象方法,从而对剩余的逻辑有不同的实现。先制定一个顶级逻辑框架,而将逻辑的细节留给具体的子类去实现。

    23、VISITOR—情人节到了,要给每个MM送一束鲜花和一张卡片,可是每个MM送的花都要针对她个人的特点,每张卡片也要根据个人的特点来挑,我一个人哪搞得清楚,还是找花店老板和礼品店老板做一下Visitor,让花店老板根据MM的特点选一束花,让礼品店老板也根据每个人特点选一张卡,这样就轻松多了;

    访问者模式:访问者模式的目的是封装一些施加于某种数据结构元素之上的操作。一旦这些操作需要修改的话,接受这个操作的数据结构可以保持不变。访问者模式适用于数据结构相对未定的系统,它把数据结构和作用于结构上的操作之间的耦合解脱开,使得操作集合可以相对自由的演化。访问者模式使得增加新的操作变的很容易,就是增加一个新的访问者类。访问者模式将有关的行为集中到一个访问者对象中,而不是分散到一个个的节点类中。当使用访问者模式时,要将尽可能多的对象浏览逻辑放在访问者类中,而不是放到它的子类中。访问者模式可以跨过几个类的等级结构访问属于不同的等级结构的成员类。

7/3/2007

如何做个极品单身男人

对生活要充满激情,其他的都会因此而得到。
钱很重要,努力赚钱,有了钱你可以得到很多东西,包括女人的身体。
如果没钱,就要学会得到很多女人的心。
如果钱和心都无法得到,那只能说明你无能。
身体最重要,无论什么都不应该用损害健康去获得。
要学会微笑,对每一个人。
让周围的人因为有你的存在而感到快乐。
善待家人,孝顺父母。
学会装傻,学会漠视,学会忍耐。
不要因为寂寞而去寻求安慰,男人要会享受寂寞。
谈太多恋爱会让人倦怠,要适当控制。
见一个爱一个的男人,他和他的爱都会跟着贬值。
事业远比爱情重要。如果说事业都不能永恒,那么爱情只能是昙花一现。
“我爱你”三个字是关键时刻用来堵住所有废话用的,平时还是多用行动表现吧。
比那三个字更好的办法是适时的拥抱和接吻。女人期待被男人强吻,前提是她喜欢你。
不要和爱慕虚荣的女人在一起,她迟早会离开你。看她爱不爱虚荣,只需要打扮的很邋遢的和她走在一起,或者送她一件很普通的礼物,然后留意她的表情。
如果一个女人认为别人就应该对她好,那趁早离开她,不懂感恩的人也不懂得珍惜。
一夜情和嫖娼没什么两样,不过一定要慎重选择对象。
不要招惹玩不起游戏的人,不管男人还是女人。
适时的送花和小礼物给女人,让她们开心其实很简单。
不要欺骗善良的女人,这样的女人数量很少。
傻女孩很多,聪明的也不少,不要盲目自信。
如果不打算骗女人一生一世,那就不要骗她。如果是善意的谎言,要设法让她明白你的良苦用心。
婚姻应该由自己主动选择,一个理解你的女人是最优的选择。
娶一个你不爱的女人之前,要确信和保证自己能一生对她很好。
背叛了女人,就要设法补偿,补偿最好的方法是钱,连钱也不要的女人就要小心了。
分手最好设法让对方先提出来。
世界上没有永远,珍惜现在才能把握未来。
对女人一定要坦诚,就算当流氓也要当坦坦荡荡的流氓。
没有一个女人,能比你的母亲更爱你。
不要吃回头草,伤你第一次的人还会有第二次,破镜重圆的几率仅仅是在电视上比较高。
能和前女友做朋友,只有两种可能,一是你们当初都只是玩玩而已,没付出彼此最真的感情。或者:必定有个人是在默默的付出无怨无悔。
要有至少一个了解你的女人,作为你的密友。要有至少一个了解你的男人,能没事陪你喝酒。
在女人面前,永远不要解释什么,直接认错不会有坏处。
第一辆车应该是二手的,先开好了再说。
一定要有一所用自己钱买的房子。
30岁之前,搞垮几个公司是好事,失败的经验比成功更能让男人心智成熟。
再大的打击都会过去,只要你还活着就永远都有机会,乐观的态度是力量的源泉。
一定要从摔倒的地方爬起来。
如果没有才华,就要有气魄,没有好的外貌,就要学会有品味。
会工作也要会休闲,有时候适当的不修边幅会让你更性感。
能哄人是出色的本事,这一点都不丢人。哄上司,哄同事,哄老人,哄女人,哄孩子。如果以上你通通都能哄到,那么恭喜你,你的人生一定值得你炫耀。
先有实力,才有魅力。让人愿意跟随你,是你的魅力所在。
要看起来像个君子,即使你不是。
要有温和的脾气和好斗的性格。
不要跟一般人一般见识,不要随便骂脏话。
侮辱一个人最好的方式就是不予理睬。
尊重任何服务行业的人,他们和你的生活息息相关。
有自己的立场和观点,不一定要抢着标明。学会沉默,有些话不必说出来。
自然的发型最好,不要整天用发胶和定型水。服装整洁就好,不必过于光鲜。皮鞋干净就好,不要擦的油光瓦亮。
厨艺高超,但不轻易下厨。厨艺不佳就要刀工精湛。
知道那里有好的餐馆,女人会记得带她吃到好东西的男人。
可以抽烟喝酒,但是不要上瘾。对身体不好的任何爱好,都应该浅尝辄止。
至少得爱一种体育运动,喜欢并参与进去。
6/29/2007

what session....?

摘要:虽然session机制在web应用程序中被采用已经很长时间了,但是仍然有很多人不清楚session机制的本质,以至不能正确的应用这一技术。本文将详细讨论session的工作机制并且对在Java web application中应用session机制时常见的问题作出解答。

目录:
一、术语session
二、HTTP协议与状态保持
三、理解cookie机制
四、理解session机制
五、理解javax.servlet.http.HttpSession
六、HttpSession常见问题
七、跨应用程序的session共享
八、总结
参考文档

一、术语session
在我的经验里,session这个词被滥用的程度大概仅次于transaction,更加有趣的是transaction与session在某些语境下的含义是相同的。

session,中文经常翻译为会话,其本来的含义是指有始有终的一系列动作/消息,比如打电话时从拿起电话拨号到挂断电话这中间的一系列过程可以称之为一个 session。有时候我们可以看到这样的话“在一个浏览器会话期间,...”,这里的会话一词用的就是其本义,是指从一个浏览器窗口打开到关闭这个期间 ①。最混乱的是“用户(客户端)在一次会话期间”这样一句话,它可能指用户的一系列动作(一般情况下是同某个具体目的相关的一系列动作,比如从登录到选购商品到结账登出这样一个网上购物的过程,有时候也被称为一个transaction),然而有时候也可能仅仅是指一次连接,也有可能是指含义①,其中的差别只能靠上下文来推断②。

然而当session一词与网络协议相关联时,它又往往隐含了“面向连接”和/或“保持状态”这样两个含义, “面向连接”指的是在通信双方在通信之前要先建立一个通信的渠道,比如打电话,直到对方接了电话通信才能开始,与此相对的是写信,在你把信发出去的时候你并不能确认对方的地址是否正确,通信渠道不一定能建立,但对发信人来说,通信已经开始了。“保持状态”则是指通信的一方能够把一系列的消息关联起来,使得消息之间可以互相依赖,比如一个服务员能够认出再次光临的老顾客并且记得上次这个顾客还欠店里一块钱。这一类的例子有“一个TCP session”或者 “一个POP3 session”③。

而到了web服务器蓬勃发展的时代,session在web开发语境下的语义又有了新的扩展,它的含义是指一类用来在客户端与服务器之间保持状态的解决方案④。有时候session也用来指这种解决方案的存储结构,如“把xxx保存在session 里”⑤。由于各种用于web开发的语言在一定程度上都提供了对这种解决方案的支持,所以在某种特定语言的语境下,session也被用来指代该语言的解决方案,比如经常把Java里提供的javax.servlet.http.HttpSession简称为session⑥。

鉴于这种混乱已不可改变,本文中session一词的运用也会根据上下文有不同的含义,请大家注意分辨。
在本文中,使用中文“浏览器会话期间”来表达含义①,使用“session机制”来表达含义④,使用“session”表达含义⑤,使用具体的“HttpSession”来表达含义⑥

二、HTTP协议与状态保持
HTTP 协议本身是无状态的,这与HTTP协议本来的目的是相符的,客户端只需要简单的向服务器请求下载某些文件,无论是客户端还是服务器都没有必要纪录彼此过去的行为,每一次请求之间都是独立的,好比一个顾客和一个自动售货机或者一个普通的(非会员制)大卖场之间的关系一样。

然而聪明(或者贪心?)的人们很快发现如果能够提供一些按需生成的动态信息会使web变得更加有用,就像给有线电视加上点播功能一样。这种需求一方面迫使HTML逐步添加了表单、脚本、DOM等客户端行为,另一方面在服务器端则出现了CGI规范以响应客户端的动态请求,作为传输载体的HTTP协议也添加了文件上载、 cookie这些特性。其中cookie的作用就是为了解决HTTP协议无状态的缺陷所作出的努力。至于后来出现的session机制则是又一种在客户端与服务器之间保持状态的解决方案。

让我们用几个例子来描述一下cookie和session机制之间的区别与联系。笔者曾经常去的一家咖啡店有喝5杯咖啡免费赠一杯咖啡的优惠,然而一次性消费5杯咖啡的机会微乎其微,这时就需要某种方式来纪录某位顾客的消费数量。想象一下其实也无外乎下面的几种方案:
1、该店的店员很厉害,能记住每位顾客的消费数量,只要顾客一走进咖啡店,店员就知道该怎么对待了。这种做法就是协议本身支持状态。
2、发给顾客一张卡片,上面记录着消费的数量,一般还有个有效期限。每次消费时,如果顾客出示这张卡片,则此次消费就会与以前或以后的消费相联系起来。这种做法就是在客户端保持状态。
3、发给顾客一张会员卡,除了卡号之外什么信息也不纪录,每次消费时,如果顾客出示该卡片,则店员在店里的纪录本上找到这个卡号对应的纪录添加一些消费信息。这种做法就是在服务器端保持状态。

由于HTTP协议是无状态的,而出于种种考虑也不希望使之成为有状态的,因此,后面两种方案就成为现实的选择。具体来说cookie机制采用的是在客户端保持状态的方案,而session机制采用的是在服务器端保持状态的方案。同时我们也看到,由于采用服务器端保持状态的方案在客户端也需要保存一个标识,所以session机制可能需要借助于cookie机制来达到保存标识的目的,但实际上它还有其他选择。

三、理解cookie机制 
cookie机制的基本原理就如上面的例子一样简单,但是还有几个问题需要解决:“会员卡”如何分发;“会员卡”的内容;以及客户如何使用“会员卡”。

正统的cookie分发是通过扩展HTTP协议来实现的,服务器通过在HTTP的响应头中加上一行特殊的指示以提示浏览器按照指示生成相应的cookie。然而纯粹的客户端脚本如JavaScript或者VBScript也可以生成cookie。

而cookie 的使用是由浏览器按照一定的原则在后台自动发送给服务器的。浏览器检查所有存储的cookie,如果某个cookie所声明的作用范围大于等于将要请求的资源所在的位置,则把该cookie附在请求资源的HTTP请求头上发送给服务器。意思是麦当劳的会员卡只能在麦当劳的店里出示,如果某家分店还发行了自己的会员卡,那么进这家店的时候除了要出示麦当劳的会员卡,还要出示这家店的会员卡。

cookie的内容主要包括:名字,值,过期时间,路径和域。
其中域可以指定某一个域比如.google.com,相当于总店招牌,比如宝洁公司,也可以指定一个域下的具体某台机器比如www.google.com或者froogle.google.com,可以用飘柔来做比。
路径就是跟在域名后面的URL路径,比如/或者/foo等等,可以用某飘柔专柜做比。
路径与域合在一起就构成了cookie的作用范围。
如果不设置过期时间,则表示这个cookie的生命期为浏览器会话期间,只要关闭浏览器窗口,cookie就消失了。这种生命期为浏览器会话期的 cookie被称为会话cookie。会话cookie一般不存储在硬盘上而是保存在内存里,当然这种行为并不是规范规定的。如果设置了过期时间,浏览器就会把cookie保存到硬盘上,关闭后再次打开浏览器,这些cookie仍然有效直到超过设定的过期时间。

存储在硬盘上的cookie 可以在不同的浏览器进程间共享,比如两个IE窗口。而对于保存在内存里的cookie,不同的浏览器有不同的处理方式。对于IE,在一个打开的窗口上按 Ctrl-N(或者从文件菜单)打开的窗口可以与原窗口共享,而使用其他方式新开的IE进程则不能共享已经打开的窗口的内存cookie;对于 Mozilla Firefox0.8,所有的进程和标签页都可以共享同样的cookie。一般来说是用javascript的window.open打开的窗口会与原窗口共享内存cookie。浏览器对于会话cookie的这种只认cookie不认人的处理方式经常给采用session机制的web应用程序开发者造成很大的困扰。

下面就是一个goolge设置cookie的响应头的例子
HTTP/1.1 302 Found
Location: http://www.google.com/intl/zh-CN/
Set-Cookie: PREF=ID=0565f77e132de138:NW=1:TM=1098082649:LM=1098082649:S=KaeaCFPo49RiA_d8; expires=Sun, 17-Jan-2038 19:14:07 GMT; path=/; domain=.google.com
Content-Type: text/html




这是使用HTTPLook这个HTTP Sniffer软件来俘获的HTTP通讯纪录的一部分




浏览器在再次访问goolge的资源时自动向外发送cookie




使用Firefox可以很容易的观察现有的cookie的值
使用HTTPLook配合Firefox可以很容易的理解cookie的工作原理。




IE也可以设置在接受cookie前询问




这是一个询问接受cookie的对话框。

四、理解session机制
session机制是一种服务器端的机制,服务器使用一种类似于散列表的结构(也可能就是使用散列表)来保存信息。

当程序需要为某个客户端的请求创建一个session的时候,服务器首先检查这个客户端的请求里是否已包含了一个session标识 - 称为 session id,如果已包含一个session id则说明以前已经为此客户端创建过session,服务器就按照session id把这个 session检索出来使用(如果检索不到,可能会新建一个),如果客户端请求不包含session id,则为此客户端创建一个session并且生成一个与此session相关联的session id,session id的值应该是一个既不会重复,又不容易被找到规律以仿造的字符串,这个 session id将被在本次响应中返回给客户端保存。

保存这个session id的方式可以采用cookie,这样在交互过程中浏览器可以自动的按照规则把这个标识发挥给服务器。一般这个cookie的名字都是类似于SEEESIONID,而。比如weblogic对于web应用程序生成的cookie,JSESSIONID= ByOK3vjFD75aPnrF7C2HmdnV6QZcEbzWoWiBYEnLerjQ99zWpBng!-145788764,它的名字就是 JSESSIONID。

由于cookie可以被人为的禁止,必须有其他机制以便在cookie被禁止时仍然能够把session id传递回服务器。经常被使用的一种技术叫做URL重写,就是把session id直接附加在URL路径的后面,附加方式也有两种,一种是作为URL路径的附加信息,表现形式为http://...../xxx;jsessionid= ByOK3vjFD75aPnrF7C2HmdnV6QZcEbzWoWiBYEnLerjQ99zWpBng!-145788764
另一种是作为查询字符串附加在URL后面,表现形式为http://...../xxx?jsessionid=ByOK3vjFD75aPnrF7C2HmdnV6QZcEbzWoWiBYEnLerjQ99zWpBng!-145788764
这两种方式对于用户来说是没有区别的,只是服务器在解析的时候处理的方式不同,采用第一种方式也有利于把session id的信息和正常程序参数区分开来。
为了在整个交互过程中始终保持状态,就必须在每个客户端可能请求的路径后面都包含这个session id。

另一种技术叫做表单隐藏字段。就是服务器会自动修改表单,添加一个隐藏字段,以便在表单提交时能够把session id传递回服务器。比如下面的表单
<form name="testform" action="/xxx">
<input type="text">
</form>
在被传递给客户端之前将被改写成
<form name="testform" action="/xxx">
<input type="hidden" name="jsessionid" value="ByOK3vjFD75aPnrF7C2HmdnV6QZcEbzWoWiBYEnLerjQ99zWpBng!-145788764">
<input type="text">
</form>
这种技术现在已较少应用,笔者接触过的很古老的iPlanet6(SunONE应用服务器的前身)就使用了这种技术。
实际上这种技术可以简单的用对action应用URL重写来代替。

在谈论session机制的时候,常常听到这样一种误解“只要关闭浏览器,session就消失了”。其实可以想象一下会员卡的例子,除非顾客主动对店家提出销卡,否则店家绝对不会轻易删除顾客的资料。对session来说也是一样的,除非程序通知服务器删除一个session,否则服务器会一直保留,程序一般都是在用户做log off的时候发个指令去删除session。然而浏览器从来不会主动在关闭之前通知服务器它将要关闭,因此服务器根本不会有机会知道浏览器已经关闭,之所以会有这种错觉,是大部分session机制都使用会话cookie来保存session id,而关闭浏览器后这个 session id就消失了,再次连接服务器时也就无法找到原来的session。如果服务器设置的cookie被保存到硬盘上,或者使用某种手段改写浏览器发出的HTTP请求头,把原来的session id发送给服务器,则再次打开浏览器仍然能够找到原来的session。

恰恰是由于关闭浏览器不会导致session被删除,迫使服务器为seesion设置了一个失效时间,当距离客户端上一次使用session的时间超过这个失效时间时,服务器就可以认为客户端已经停止了活动,才会把session删除以节省存储空间。

五、理解javax.servlet.http.HttpSession
HttpSession是Java平台对session机制的实现规范,因为它仅仅是个接口,具体到每个web应用服务器的提供商,除了对规范支持之外,仍然会有一些规范里没有规定的细微差异。这里我们以BEA的Weblogic Server8.1作为例子来演示。

首先,Weblogic Server提供了一系列的参数来控制它的HttpSession的实现,包括使用cookie的开关选项,使用URL重写的开关选项,session持久化的设置,session失效时间的设置,以及针对cookie的各种设置,比如设置cookie的名字、路径、域, cookie的生存时间等。

一般情况下,session都是存储在内存里,当服务器进程被停止或者重启的时候,内存里的session也会被清空,如果设置了session的持久化特性,服务器就会把session保存到硬盘上,当服务器进程重新启动或这些信息将能够被再次使用, Weblogic Server支持的持久性方式包括文件、数据库、客户端cookie保存和复制。

复制严格说来不算持久化保存,因为session实际上还是保存在内存里,不过同样的信息被复制到各个cluster内的服务器进程中,这样即使某个服务器进程停止工作也仍然可以从其他进程中取得session。

cookie生存时间的设置则会影响浏览器生成的cookie是否是一个会话cookie。默认是使用会话cookie。有兴趣的可以用它来试验我们在第四节里提到的那个误解。

cookie的路径对于web应用程序来说是一个非常重要的选项,Weblogic Server对这个选项的默认处理方式使得它与其他服务器有明显的区别。后面我们会专题讨论。

关于session的设置参考[5] http://e-docs.bea.com/wls/docs70/webapp/weblogic_xml.html#1036869

六、HttpSession常见问题
(在本小节中session的含义为⑤和⑥的混合)


1、session在何时被创建
一个常见的误解是以为session在有客户端访问时就被创建,然而事实是直到某server端程序调用 HttpServletRequest.getSession(true)这样的语句时才被创建,注意如果JSP没有显示的使用 <% @page session="false"%> 关闭session,则JSP文件在编译成Servlet时将会自动加上这样一条语句 HttpSession session = HttpServletRequest.getSession(true);这也是JSP中隐含的 session对象的来历。

由于session会消耗内存资源,因此,如果不打算使用session,应该在所有的JSP中关闭它。

2、session何时被删除
综合前面的讨论,session在下列情况下被删除a.程序调用HttpSession.invalidate();或b.距离上一次收到客户端发送的session id时间间隔超过了session的超时设置;或c.服务器进程被停止(非持久session)

3、如何做到在浏览器关闭时删除session
严格的讲,做不到这一点。可以做一点努力的办法是在所有的客户端页面里使用javascript代码window.oncolose来监视浏览器的关闭动作,然后向服务器发送一个请求来删除session。但是对于浏览器崩溃或者强行杀死进程这些非常规手段仍然无能为力。

4、有个HttpSessionListener是怎么回事
你可以创建这样的listener去监控session的创建和销毁事件,使得在发生这样的事件时你可以做一些相应的工作。注意是session的创建和销毁动作触发listener,而不是相反。类似的与HttpSession有关的listener还有 HttpSessionBindingListener,HttpSessionActivationListener和 HttpSessionAttributeListener。

5、存放在session中的对象必须是可序列化的吗
不是必需的。要求对象可序列化只是为了session能够在集群中被复制或者能够持久保存或者在必要时server能够暂时把session交换出内存。在 Weblogic Server的session中放置一个不可序列化的对象在控制台上会收到一个警告。我所用过的某个iPlanet版本如果 session中有不可序列化的对象,在session销毁时会有一个Exception,很奇怪。

6、如何才能正确的应付客户端禁止cookie的可能性
对所有的URL使用URL重写,包括超链接,form的action,和重定向的URL,具体做法参见[6]
http://e-docs.bea.com/wls/docs70/webapp/sessions.html#100770

7、开两个浏览器窗口访问应用程序会使用同一个session还是不同的session
参见第三小节对cookie的讨论,对session来说是只认id不认人,因此不同的浏览器,不同的窗口打开方式以及不同的cookie存储方式都会对这个问题的答案有影响。

8、如何防止用户打开两个浏览器窗口操作导致的session混乱
这个问题与防止表单多次提交是类似的,可以通过设置客户端的令牌来解决。就是在服务器每次生成一个不同的id返回给客户端,同时保存在session里,客户端提交表单时必须把这个id也返回服务器,程序首先比较返回的id与保存在session里的值是否一致,如果不一致则说明本次操作已经被提交过了。可以参看《J2EE核心模式》关于表示层模式的部分。需要注意的是对于使用javascript window.open打开的窗口,一般不设置这个id,或者使用单独的id,以防主窗口无法操作,建议不要再window.open打开的窗口里做修改操作,这样就可以不用设置。

9、为什么在Weblogic Server中改变session的值后要重新调用一次session.setValue
做这个动作主要是为了在集群环境中提示Weblogic Server session中的值发生了改变,需要向其他服务器进程复制新的session值。

10、为什么session不见了
排除session正常失效的因素之外,服务器本身的可能性应该是微乎其微的,虽然笔者在iPlanet6SP1加若干补丁的Solaris版本上倒也遇到过;浏览器插件的可能性次之,笔者也遇到过3721插件造成的问题;理论上防火墙或者代理服务器在cookie处理上也有可能会出现问题。
出现这一问题的大部分原因都是程序的错误,最常见的就是在一个应用程序中去访问另外一个应用程序。我们在下一节讨论这个问题。

七、跨应用程序的session共享

常常有这样的情况,一个大项目被分割成若干小项目开发,为了能够互不干扰,要求每个小项目作为一个单独的web应用程序开发,可是到了最后突然发现某几个小项目之间需要共享一些信息,或者想使用session来实现SSO(single sign on),在session中保存login的用户信息,最自然的要求是应用程序间能够访问彼此的session。

然而按照Servlet规范,session的作用范围应该仅仅限于当前应用程序下,不同的应用程序之间是不能够互相访问对方的session的。各个应用服务器从实际效果上都遵守了这一规范,但是实现的细节却可能各有不同,因此解决跨应用程序session共享的方法也各不相同。

首先来看一下Tomcat是如何实现web应用程序之间session的隔离的,从 Tomcat设置的cookie路径来看,它对不同的应用程序设置的cookie路径是不同的,这样不同的应用程序所用的session id是不同的,因此即使在同一个浏览器窗口里访问不同的应用程序,发送给服务器的session id也可以是不同的。


  

根据这个特性,我们可以推测Tomcat中session的内存结构大致如下。




笔者以前用过的iPlanet也采用的是同样的方式,估计SunONE与iPlanet之间不会有太大的差别。对于这种方式的服务器,解决的思路很简单,实际实行起来也不难。要么让所有的应用程序共享一个session id,要么让应用程序能够获得其他应用程序的session id。

iPlanet中有一种很简单的方法来实现共享一个session id,那就是把各个应用程序的cookie路径都设为/(实际上应该是/NASApp,对于应用程序来讲它的作用相当于根)。
<session-info>
<path>/NASApp</path>
</session-info>

需要注意的是,操作共享的session应该遵循一些编程约定,比如在session attribute名字的前面加上应用程序的前缀,使得 setAttribute("name", "neo")变成setAttribute("app1.name", "neo"),以防止命名空间冲突,导致互相覆盖。


在Tomcat中则没有这么方便的选择。在Tomcat版本3上,我们还可以有一些手段来共享session。对于版本4以上的Tomcat,目前笔者尚未发现简单的办法。只能借助于第三方的力量,比如使用文件、数据库、JMS或者客户端cookie,URL参数或者隐藏字段等手段。

我们再看一下Weblogic Server是如何处理session的。


  

从截屏画面上可以看到Weblogic Server对所有的应用程序设置的cookie的路径都是/,这是不是意味着在Weblogic Server中默认的就可以共享session了呢?然而一个小实验即可证明即使不同的应用程序使用的是同一个session,各个应用程序仍然只能访问自己所设置的那些属性。这说明Weblogic Server中的session的内存结构可能如下




对于这样一种结构,在 session机制本身上来解决session共享的问题应该是不可能的了。除了借助于第三方的力量,比如使用文件、数据库、JMS或者客户端 cookie,URL参数或者隐藏字段等手段,还有一种较为方便的做法,就是把一个应用程序的session放到ServletContext中,这样另外一个应用程序就可以从ServletContext中取得前一个应用程序的引用。示例代码如下,

应用程序A
context.setAttribute("appA", session); 

应用程序B
contextA = context.getContext("/appA");
HttpSession sessionA = (HttpSession)contextA.getAttribute("appA"); 

值得注意的是这种用法不可移植,因为根据ServletContext的JavaDoc,应用服务器可以处于安全的原因对于context.getContext("/appA");返回空值,以上做法在Weblogic Server 8.1中通过。

那么Weblogic Server为什么要把所有的应用程序的cookie路径都设为/呢?原来是为了SSO,凡是共享这个session的应用程序都可以共享认证的信息。一个简单的实验就可以证明这一点,修改首先登录的那个应用程序的描述符weblogic.xml,把cookie路径修改为/appA 访问另外一个应用程序会重新要求登录,即使是反过来,先访问cookie路径为/的应用程序,再访问修改过路径的这个,虽然不再提示登录,但是登录的用户信息也会丢失。注意做这个实验时认证方式应该使用FORM,因为浏览器和web服务器对basic认证方式有其他的处理方式,第二次请求的认证不是通过 session来实现的。具体请参看[7] secion 14.8 Authorization,你可以修改所附的示例程序来做这些试验。

八、总结
session机制本身并不复杂,然而其实现和配置上的灵活性却使得具体情况复杂多变。这也要求我们不能把仅仅某一次的经验或者某一个浏览器,服务器的经验当作普遍适用的经验,而是始终需要具体情况具体分析。
摘要:虽然session机制在web应用程序中被采用已经很长时间了,但是仍然有很多人不清楚session机制的本质,以至不能正确的应用这一技术。本文将详细讨论session的工作机制并且对在Java web application中应用session机制时常见的问题作出解答。 
 
china

咖啡 田

Occupation
Photo 1 of 6