之前遇到一个要求,需要能够取消一个正在进行中的Web Service。这也是我第一次遇到这个功能,不过不难,我想。既然ASP.NETAJAX的客户端与服务器端通信完全通过Microsoft AJAX Library的异步通信层进行,那么我们只要得到正在请求WebService的Sys.Net.WebRequest对象,调用其abort方法就可以了。但是究竟应该如何得到这个对象呢?于是我粗略地阅读了一下代码。   首先假设有如下的Web Service方法定义(DemoService.asmx):íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
[ScriptService]íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
public class DemoService  : System.Web.Services.WebServiceíÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
{íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    [WebMethod]íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    public string DemoMethod()íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    {íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        return "Hello World";íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    }    íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
}íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  访问DemoService.asmx/jsdebug(或者将其使用ScriptManager引用到页面中之后)就能够得到如下的代理(片断、经过排版)类。íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
var DemoService = function()íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
{íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    DemoService.initializeBase(this);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    this._timeout = 0;íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    this._userContext = null;íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    this._succeeded = null;íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    this._failed = null;íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
}íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
DemoService.prototype =íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
{íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    DemoMethodunction(succeededCallback, failedCallback, userContext)íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    {íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        return this._invoke(íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            DemoService.get_path(),íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            'DemoMethod',íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            false,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            {},íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            succeededCallback,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            failedCallback,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
            userContext);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    }íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
}íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
DemoService.registerClass('DemoService',Sys.Net.WebServiceProxy);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
...íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  显然,这个代理类继承了Sys.Net.WebServiceProxy类,于是我就把目光转向了其中的_invoke方法:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
function Sys$Net$WebServiceProxy$_invoke(íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    servicePath, methodName, useGet, params, onSuccess, onFailure, userContext) {íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    // validation omittedíÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    ...íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    return Sys.Net.WebServiceProxy.invoke(íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        servicePath,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        methodName,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        useGet,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        params,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        onSuccess,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        onFailure,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        userContext,íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        this.get_timeout());íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
}íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  这下又将操作委托给了Sys.Net.WebServiceProxy.invoke静态方法,继续看代码:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
Sys.Net.WebServiceProxy.invoke = function Sys$Net$WebServiceProxy$invoke(íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    servicePath, methodName, useGet, params, onSuccess, onFailure, userContext, timeout) {íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    // validation omittedíÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    ...íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    // Create a web request to make the method callíÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    var request = new Sys.Net.WebRequest();íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    // preparing request omittedíÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    ...íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    request.invoke();íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    function onComplete(response, eventArgs) {íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
        // method body omittedíÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    }íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    return request;íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
}íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  嗨,这不就是我所需要的Sys.Net.WebRequest对象吗?原来想要得到这个对象那么简单,于是我就写下了下面的代码:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
var request = DemoService.DemoMethod(onComplete);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  然后在必要时:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
request.abort();íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  执行,出现了错误:request为undefined,为什么DemoMethod方法调用没有返回request对象?跟踪了代码之后,不大不小地晕了一下,原来问题出在这里:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
DemoService._staticInstance = new DemoService();íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
...íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
DemoService.DemoMethod = function(onSuccess,onFailed,userContext)íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
{íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
    DemoService._staticInstance.DemoMethod(onSuccess,onFailed,userContext);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
}íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  虽然早就知道WebService代理会在类上创建一个Singleton对象,并且创建静态方法再委托给那个实例上的相应方法,却一直没有意识到这个细节。在上面的静态方法中,居然是直接调用了DemoMethod方法,却没有将结果返回出来,真让我哭笑不得了一下。íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  不过问题时非常容易解决的,只要使用如下的方式在客户端调用WebService方法就可以了:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
var request = DemoService._staticInstance.DemoMethod(onComplete);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  不过这个做法似乎……有些奇怪?那么您也可以这样:íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
var demoService = new DemoService();íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
var request = demoService.DemoMethod(onComplete);íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ
  在这里,重新创建一个demoService对象似乎有些多余,不过在某些时候也是非常有用的做法。例如,您需要将操作分为两类,一类的超时时间为5秒,而另一类为10秒,因此您就可以创建两个代理对象,分别设置不同的超时时间。因为超时时间我们只能在Service的级别上设置,而不能在调用方法时指定。íÉ ò±?Cçwww.netcsharp.cnMá!¿WeëßfÈ