几乎所有的Web开发语言都支持Session功能,Servlet也不例外。 Servlet/JSP中的Session功能是通过作用域(scope)这个概念来实现的。
作用域分为四种,分别为:

page 在当前页面有效(仅用于JSP中)
request 在当前请求中有效
session 在当前会话中有效
application 在所有应用程序中有效

是不是看不太明白?page因为仅用于JSP中,这里只讲述其他三种作用域。首先要声明的一点,所谓“作用域”就是“信息共享的范围”,也就是说一个信息能够在多大的范围内有效。
话说武松一日来到景阳岗,见一旗帜迎风飘扬,旗子上书五个大字“三碗不过岗”。武松叫道:“店家,拿三碗酒来,再切两斤熟牛肉!”店小二应声道:“三碗好酒,二斤熟牛肉啰~~”里面厨师赶忙当当当当切好牛肉,店小二倒上三碗酒,店小二端上前来。
武松咕咚咕咚连干三碗,叫一声“好酒!店家,再来三碗!”小二忙又倒上三碗好酒,武松一饮而尽。就这样前前后后武松一共喝了十八大腕。付了帐刚要走,店小二道: “客官,这前面山上有大虫,客官刚刚喝完十八碗酒恐怕过不得岗,不如在小店暂住一夜,待明天和猎户一同过岗岂不是好?”
之后武松说什么就留待各位看官自己去回忆啦。在这段武松打虎中,大家有没有看到些熟悉的东西?

  • 武松: 浏览器。
  • 酒馆: 服务器。
  • 店小二、厨师: Servlet或者JSP。
  • 来三碗好酒!: 浏览器向服务器发出HTTP请求。
  • 店小二上酒: 服务器的响应。
  • 武松从进店到离开: 一个HTTP对话。
    我们可以看到,Web交互的最基本单位为HTTP请求(武松点菜)。每个用户从进入网站到离开网站这段过程称为一个HTTP会话 (武松进店到出店),一个服务器的运行过程中会有多个用户访问,就是多个HTTP会话(酒馆当然不可能只接待武松一个客人)。那么作用域就可以理解为:
request HTTP请求开始到结束这段时间
session HTTP会话开始到结束这段时间
application 服务器启动到停止这段时间

request
一个HTTP请求的处理可能需要多个Servlet合作(武松点菜时店小二就要吩咐厨房做菜),几个Servlet之间可以通过某种方式传递信息(店小二就用吆喝的方式通知厨房),但这个信息在请求结束后就无效了(厨房在做完菜之后就不用再管这道菜的事儿了)。
Servlet之间的信息共享是通过HttpServletRequest接口的两个方法来实现的:
void setAttribute(String name, Object value)
将对象 value 以 name 为名称保存到request作用域中。
Object getAttribute(String name)
从request作用域中取得指定名字的信息。
doGet()、doPost()函数的第一个参数就是 HttpServletRequest 对象,使用这个对象的 setAttribute 即可传递信息。
那么设置好信息之后,如何将信息传给其他Servlet?这就要用到 RequestDispatcher 接口的 forward 方法,将请求转发给其他Servlet。
RequestDispatcher ServletContext.getRequestDispatcher(String path)
取得Dispatcher以便转发。path为转发的目的Servlet。
void RequestDispatcher.forward(ServletRequest request, ServletResponse response)
将request和response转发。
因此,只要在当前Servlet中先 setAttribute,然后forward,最后在forward到的Servlet中 getAttribute即可实现信息传递。
PHP的程序员可能不太好理解这一段,因为PHP中没有转发的概念,一个请求只能由一个PHP文件来处理,所以PHP中根本没有request作用域的概念。而Servlet则不同,请求可以在应用程序中任意转发,所以用request作用域在不同Servlet之间传递信息。需要注意两点:
转发不是重定向,转发是在Web应用内部进行的。PHP支持重定向但没有转发。
转发对浏览器是透明的,也就是说,无论在服务器上如何转发,浏览器地址栏中显示的仍然是最初那个Servlet的地址。
session
session作用域比较容易理解,同一浏览器访问多次,在这多次访问之间传递信息,就是session作用域。 ('''武松每次点菜帐房先生都要记一笔账,等武松走之前结帐用。这笔帐在武松吃饭过程中始终有效,即位于session作用域中''')
session是通过HttpSession接口实现的。
Object HttpSession.getAttribute(String name)
从session中获取信息
void HttpSession.setAttribute(String name, Object value)
向session中保存信息
而通过HttpServletRequest.getSession()方法可以获得HttpSession对象。
HttpSession HttpServletRequest.getSessio()
获取当前请求所在的session的对象。
session的开始容易判断(浏览器发出第一个HTTP请求即可认为会话开始),但结束就不好判断了(因为浏览器关闭时不会通知服务器“我关了,会话可以结束了”),所以只能通过这种方法判断:如果一定的时间内客户端没有反应,则认为会话结束。 Tomcat的默认值为120分钟,但这个值也可以通过 HttpSession 的 setMaxInactiveInterval 方法来设置。
void setMaxInactiveInterval(int interval)
设置绘画的超时值。
如果想主动让会话结束,如用户单击“注销”的时候,可以使用 HttpSession 的 invalidate 方法:
vooid invalidate()
强制结束当前session。
application
application作用域就是服务器启动到关闭的整段时间,在这个作用域内设置的信息可以被所有应用程序使用。 (餐馆打烊后结帐,用到的即是开张到打烊之间的所有信息。)
application作用域上的信息传递就是通过ServetContext实现的。

Object getAttribute(String name)
从application中获取信息。
void setAttribute(String name, Object value)
向application作用域中设置信息。

总结

可以看到,每个作用域除了实现接口不同、意义不同之外,它们的使用方法和作用都是相同的,都是通过 getAttribute 和 setAttribute 方法进行信息传递。

作用域 意义 实现接口
request HTTP请求内 HttpServletRequest
session HTTP会话内 HttpSession
application 服务器生命周期内 ServletContext

示例程序

这一节的示例程序是一个用户登录的模拟程序。文件较多。

  • login.html 登录表单
  • DoLogin.java 处理登录动作的Servlet
  • LoginSuccess.java 用于显示登录成功信息的Servlet
  • SessionTest.java 登录后的处理程序
  • DoLogout.java 注销的处理程序

为了演示 request、application、session 各个作用域的使用方法, Servlet之间进行了数据传递,数据传递方式如下:

数据产生 数据接受 数据内容 作用域
DoLogin LoginSuccess 登录时间 request
DoLogin SessionTest 登录用户名 session
DoLogin SessionTest 系统登录次数 application

原文:http://hi.baidu.com/jgszx/blog/item/23e7304b1b1d7a2b08f7efec.html 

posted @ 2012-05-15 09:43 Anlycp 阅读(14) 评论(0) 编辑

列表某列如何设置成下拉框,如何实现数字校验,可通过下面代码实现

效果图:

 

 

View Code
<?xml version="1.0" encoding="utf-8"?>
<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009" 
               xmlns:s="library://ns.adobe.com/flex/spark" 
               xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600" >
    <s:layout>
        <s:VerticalLayout/>
    </s:layout>    
    <fx:Declarations>
        <!-- 将非可视元素(例如服务、值对象)放在此处 -->
    </fx:Declarations>
    <fx:Script>
        <![CDATA[
            import mx.collections.ArrayCollection;
            import mx.controls.Alert;
            import mx.events.DataGridEvent;
            
            private var dpFormula:ArrayCollection=new ArrayCollection([
                {NAME:"AA",OPERATOR:">",OVALUE:5},
                {NAME:"BB",OPERATOR:">=",OVALUE:9},
                {NAME:"CC",OPERATOR:"<",OVALUE:6}
            ])
            protected function btnsave_clickHandler(event:MouseEvent):void
            {
                for(var i:Number=0;i<dpFormula.length;i++){        
                    if(dpFormula[i].OVALUE==""){
                        Alert.show("请输入算法值");
                        return;
                    }
                    if(!isNumber(dpFormula[i].OVALUE)){
                        dgFormula.editedItemPosition={ columnIndex:2, rowIndex:i};
                        return
                    }
                }
                //校验完毕,可进行保存处理
            }
            
            private function isNumber(value:String):Boolean{            
                var num:Number=Number(value);
                if(num.toString()=="NaN"){
                    return false;    
                }
                else{
                    return true;
                }                
            }

        ]]>
    </fx:Script>

    
    <mx:DataGrid id="dgFormula" dataProvider="{dpFormula}" width="300" height="200" editable="true" >
        <mx:columns>
            <mx:DataGridColumn dataField="NAME" headerText="中文名" editable="false"/>
            <!--内部渲染  编辑时呈现ComboBox控件-->
            <mx:DataGridColumn dataField="OPERATOR" headerText="运算符" editorDataField="value" >                        
                <mx:itemEditor>
                    <fx:Component>
                        <mx:VBox verticalGap="0" width="100%" height="100%" verticalAlign="middle">
                            <fx:Script>
                                <![CDATA[
                                    import mx.collections.ArrayCollection;
                                    [Bindable]
                                    public var arrmark:ArrayCollection=new ArrayCollection([
                                        {lable:">"},
                                        {lable:">="},
                                        {lable:"="},
                                        {lable:"<"},
                                        {lable:"<="},
                                    ]); 
                                    
                                    public function get value():Object{
                                        if(cbxMark.selectedItem == null){
                                            return null;
                                        }
                                        var obj:Object=cbxMark.selectedItem;
                                        return obj.lable;
                                    }
                                    override public function set data(value:Object):void{
                                        super.data = value;
                                        //给combobox赋初始值
                                        for each(var obj:Object in arrmark){
                                            if(obj.lable==value.OPERATOR){
                                                cbxMark.selectedItem = obj;
                                                break;
                                            }
                                        }
                                    }
                                ]]>
                            </fx:Script>
                            <s:ComboBox id="cbxMark" labelField="lable" dataProvider="{arrmark}" 
                                        width="100%" height="100%" />
                        </mx:VBox>
                    </fx:Component>
                </mx:itemEditor>                
            </mx:DataGridColumn>
            <!--内部渲染  编辑时呈现TextInput控件-->
            <mx:DataGridColumn dataField="OVALUE" headerText="值"  editorDataField="value" >
                <mx:itemEditor>
                    <fx:Component>
                        <mx:VBox verticalGap="0" width="100%" height="100%" verticalAlign="middle">
                            <fx:Declarations>
                                <mx:NumberValidator id="numVal" source="{txtvalue}" property="text"
                                                    allowNegative="true" domain="real" invalidCharError="请输入数字"/>
                            </fx:Declarations>
                            <fx:Script>
                                <![CDATA[
                                    public function get value():Object{
                                        return txtvalue.text;
                                    }
                                    override public function set data(value:Object):void{
                                        super.data = value;
                                        txtvalue.text=value.OVALUE;
                                    }
                                ]]>
                            </fx:Script>
                            <s:TextInput id="txtvalue" width="100%" height="100%" change="numVal.validate()"  />
                        </mx:VBox>
                    </fx:Component>
                </mx:itemEditor>
            </mx:DataGridColumn>
        </mx:columns>
    </mx:DataGrid>
    <s:Button id="btnsave"  label="保存" click="btnsave_clickHandler(event)"/>
</s:Application>

 

 

posted @ 2012-05-03 10:27 Anlycp 阅读(43) 评论(0) 编辑

 

 

随着我们的项目页面日渐增多,编译速度也越来越蜗牛了。

网上有很多办法教如何提高编译速度的,试了好几个,发现下面方法比较好。

 

修改方法:

Flex项目右键—》 属性—》Flex应用程序—》把不需要编译的mxml页面删除(等需要编译的时可在此位置添加回来)

Flex项目右键—》 属性—》Flex 模块 —》把已经编译好的模块移除

 

posted @ 2012-03-06 09:52 Anlycp 阅读(79) 评论(0) 编辑

 

Tree、DataGrid经常要设置自己制定样式或特殊的UI,我们需要呈现器,经常使用外部呈现器(作为项目渲染器使用的自定义组件在MXML或ActionScript编写),我们需要用到itemRenderer属性来进行设置

 

使用方法:

<mx:Tree id="IDTree"  labelField="@NAME" dataProvider="{DPtree}"
       width="200" height="159"  itemRenderer="components.TreeCheckBoxRenderer"
       click="TreeOnClick(event)"  showRoot="true"  x="0" y="0">       
 </mx:Tree> 

 

如希望动态设置itemRenderer,写法如下:

import components.TreeCheckBoxRenderer;  

//方法中下入下面代码:

var renderer:IFactory=new ClassFactory(TreeCheckBoxRenderer);   
IDTree.itemRenderer=renderer;

 

posted @ 2012-02-14 11:20 Anlycp 阅读(129) 评论(0) 编辑

 

Flex页面初始化creationComplete事件中,加入焦点设置语句 : focusManager.setFocus("TextInput组件ID");

页面运行后并看不到效果,将焦点设置语句放到按钮事件中,TextInput倒是焦点设置上了。

 

主要原因是我们把Flash嵌到HTML页面中,打开HTML页面时并没有聚焦,所以最好在HTML页面上加个JavaScript脚本使其聚焦即可。

 

步骤:

1、页面初始化creationComplete事件中,加入焦点设置语句 : focusManager.setFocus("TextInput组件ID");

2、bin-debug目录中找到此页面,如 Login.html (Login对于的swf ID 就是Login), 在body中添加js方法

<body onload="document.getElementById('要加载的swf ID').focus()">

 

通过这两个步骤即可实现初始化焦点

 

 

同样的,我们也可以使用ExternalInterface.call调用,讲JS写在Flex的AS脚本里,代码如下:

   protected function init():void{
    focusManager.setFocus(txtLoginname); //txtLoginname是TextInput组件ID
    ExternalInterface.call("document.getElementById('Login').focus()");//Login是要加载的swf ID
   }

 

posted @ 2012-02-11 14:41 Anlycp 阅读(116) 评论(0) 编辑
摘要: Flex的Modules技术是可以被flex程序使用的一个swf文件,它不能脱离程序独立运行,但是多个程序之间可以共享它。flex的Modules技术将应用程序分割成小块、模块,主程序动态的加载所需要的模块。主程序在启动时并不全部加载所有的模块。当用户和模块没有交互的时候它不需要加载模块,同时它在模块不需要的时候可以卸载模块并释放内存和资源。flex的Modules技术主要有如下的优点:让swf文件初始下载尺寸更小让加载时间更短对应用程序更好的封装性。界面呈现Application开发添加ModuleLoader控件,并传入的计算参数值View Code <?xmlversion=&qu阅读全文
posted @ 2012-02-02 14:44 Anlycp 阅读(80) 评论(0) 编辑
摘要: 下面是效果图,中间的计算结果是一个Group,A、B 填充和计算结果是在Application中,下面功能实现Application与Group通信的相互通信。界面呈现Application开发View Code <?xmlversion="1.0"encoding="utf-8"?><s:Applicationxmlns:fx="http://ns.adobe.com/mxml/2009"xmlns:s="library://ns.adobe.com/flex/spark"xmlns:mx=&q阅读全文
posted @ 2012-01-13 11:38 Anlycp 阅读(61) 评论(0) 编辑
摘要: Flex界面和Java后台进行Socket交互,socket服务器端总接收到 <policy-file-request/>信息,而无法进行下一步通信。原因是Adobe Flash Player 9.0.124版本后,安全策略被更改,原来Socket或XmlSocket的应用里的http方式加载安全策略的手段不能继续使用。我们可以通过下面方法来实现。 Socket服务器端检验得到安全策略<policy-file-request/>,如接收则发送策略串检验没有获得安全策略<policy-file-request/>,则进行业务逻辑处理客户端 初始化界面时调用So阅读全文
posted @ 2012-01-13 11:24 Anlycp 阅读(96) 评论(0) 编辑
摘要: LineChart控件如何动态添加线,并设置线的颜色呢。找了很多资料都没合适的,不容易呀,下面直接上代码。<?xmlversion="1.0"encoding="utf-8"?><s:Applicationxmlns:fx="http://ns.adobe.com/mxml/2009"xmlns:s="library://ns.adobe.com/flex/spark"xmlns:mx="library://ns.adobe.com/flex/mx"minWidth="阅读全文
posted @ 2011-12-12 18:09 Anlycp 阅读(284) 评论(0) 编辑
摘要: 查了很多四舍五入的取值方式,悲剧了 都只介绍toFixed,但取得值好像不是四舍五入。怎么大家就没想过用round来求得呢,round取得的是整数进行四舍五入下面例子是取两位小数,如果取三位小数,值需要改为Math.round(a*1000)/1000,以此类推。。。vara:Number=newNumber();a=1.1456//保留两位小数,并且四舍五入a=Math.round(a*100)/100//a值为1.15a=1.1456//保留两位小数,不进行四舍五入varb:String=currencyFormatter.format(a);//a值为1.14//toFixed取两位小数阅读全文
posted @ 2011-12-07 15:46 Anlycp 阅读(158) 评论(0) 编辑