dwr的ScriptSession和HttpSession分析
1、关于ScriptSession?
ScriptSession不会与HttpSession同时创建?
当我们访问一个页面的时候,如果是第一次访问,会创建一个新的HttpSession,之后再访问的时候,就会保持当前的HttpSession,即使是刷新,也能保持当前的HttpSession。?
ScriptSession则不同,第一次访问,会创建一个新的ScriptSession,但是,如果你刷新,就会再创建一个新的ScriptSession. 跟第一次的ScriptSession完全不一样。
?
?
2、如何得到ScriptSession?
在DWR中,我们可以通过WebContextFactory.get()来取得一个WebContext对象,进而通过WebContext的getScriptSession()取得ScriptSession对象。?
?
注意:
在我们自定义的Servlet中,我们也可以通过WebContextFactory.get()来取得一个WebContext,但是这种方法却不能取得ScriptSession对象。
因为,此WebContext对象其实不是通过DWR的上下文环境得到的,所以,就根本没有创建ScriptSession对象。?
假设这种方式也能得到ScriptSession的话,那么我们实现“推”也就可以不局限在DWR的上下文环境中了,那么其灵活性就会大很多了。?
所以,这就是我们不能在Servlet中实现推的原因。?
?
3、关于刷新就创建一个新的ScriptSession问题?
在我们需要推送的页面中,如果你刷新以下,那么就提交一个Http的request,此时,如果是第一次,那么就会创建一个httpSession对象,同时,请求由DwrServlet来处理后,就会创建一个ScriptSession.
这个ScriptSession会和你的request请求的URI绑定放在一个由ScriptSessionManager维护的Map里面(这里面其实是一个URI对应的Set,在Set里面放置的是URI绑定的所有ScriptSession)。
即:Map<URI,ScriptSession> map=HashMap<URI,ScriptSession>();最好是:Map<HttpSessionID,ScriptSession> map=HashMap<HttpSessionID,ScriptSession>();
当你刷新的时候,同样的一个HttpSession,却会创建一个新的ScriptSession,然后绑定到对应的URI上。 即每次刷新,都把新ScriptSession放入的map中,在Map中主键是不能重复的,URI不变的情况下使其对应的ScriptSession为最新。
?
4、向所有的页面访问者推送?
当我们想向所有的页面访问者推送的时候,我们只需要,取得所有的页面访问者,就可以“推”了。?
如何取得所有的页面访问者?
可以通过?
// ? ? ?Collection pages = webContext.getScriptSessionsByPage("/SynMap/map/map.jsp");?
来取得/SynMap/map/map.jsp的所有访问的ScriptSession?
如何推送?
//Util util = new Util(pages);?
//util.addFunctionCall("syningMap",new Double(x),new Double(y),new Integer(zoom));?
通过此方法,就可以实现调用客户端的javascript函数,实现对客户端的操作。?
?
5、在上面的推送中产生的问题?
上面的方法已经可以实现向所有的访问者推送。但是问题是,在客户端,如果用户刷新一次或多次,那么,Collection里面可能就保存了很多的无用的ScriptSession,所以不仅仅会影响性能问题,更重要的是,可能就不能实现你想要的功能。?
比如,你想取得当前再现的有效用户,那么你就需要知道那些ScriptSession是有效的。?
?
6、如何管理有效的ScriptSession?
由于上面的问题,我们就需要自己管理ScriptSession.其实,有效地HttpSession,就是那个和当前的HttpSession匹配的ScriptSession.?
3中提到了这个,Map<HttpSessionID,ScriptSession> map=HashMap<HttpSessionID,ScriptSession>();
我们可以自己维护这个Map,在这个Map里面,我们定义key就是HttpSession的Id,其值就是ScriptSession对象。?
在每一次页面载入的时候,都去注册此ScriptSession,那么就会把新的ScriptSession在注册到HttpSessionId中,从而覆盖掉老的ScriptSession
?