分享

Web服务客户端的动态方案

 ThinkTank_引擎 2015-01-05

在.net下一般的的Web服务开发是这样的:先规划服务端的服务程序,例如.asmx请求处理程序。然后用disco实用程序生成发现文件,再用wsdl实用程序生成代理类的源文件,将这个源文件编译到客户端应用程序中。如果根据业务需求,修改了Web服务程序,例如增加了一个参数,那么这个过程就必须重复一遍,或者至少客户端程序必须改动。
在某些情况下,这样的要求是苛刻的,或者至少是代价高昂的。例如,我的客户端程序是一个7*24运行的一个Windows服务应用程序,不允许任意停机。这样所有静态的客户端方案虽然美好,却代价高昂。
理论上所有静态的方案都可以很轻松地修改为动态方案。我于是开始设计这个动态方案。
这个方案的实质是,动态获取.wsdl文档,获取Web服务的方法原型,用一个通用的方法来调用,而实际转接到相应的服务方法上。

先计划将代理类不直接从SoapHttpClientProtocol继承,而是从SoapHttpClientProtocol的基类继承,然后仿照SoapHttpClientProtocol的实现做一些实现,结果发现是死路一条。其一是SoapHttpClientProtocol中使用的一些重要的中间类型都是私有的;其二是SoapHttpClientProtocol的SOAP封包是反射器完成的,而自己实现的仿真SoapHttpClientProtocol无法满足私有的反射器的需要。
接下来选择一个合适的动态生成代码的方案。CodeDom稍微简单一些,但是对于代码量较小的方案,不是太实用。
最后只好选择Emit方案了。经过几个小时的努力,终于实现了。与直接使用.wsdl生成的代理比较,速度相差无异。

下面是一个示例,解决一个短信的代理服务需求。其背景是:网络设备维护一个数据库,通过增加记录和标记记录这两种方式接收短信或者发送短信。这个SOAP客户端是一个正常的Windows服务应用程序,定期通过ODBC访问这个数据库中的某个表,当发现未标记的记录时,取出来交给对应的Web服务程序来处理。换句话说,根据短信记录的类别不同,会转到不同的服务中,这个服务按设计是可以通过配置文件来添加的。
例如:

<services>
    <service id="3" name="BartonAgent" sign="Barton" wsdl="bartonagent.wsdl" />
    <service id="4" name="JeffAgent" wsdl="jeffagent.wsdl" />
</services>

由一个工具维护这个配置文件。当配置变动时,配置工具重启Windows服务程序,动态将指定的Web服务加入到服务列表中。

这是客户端的调用模型:

public sealed class ServiceEntry
{
    public ServiceEntry(int id, string name, string sign, string wsdl)
    {
        _ID = id;
        _Name = name;
        _Sign = sign;
        _Type = GetTypeFromWsdl(_Name, wsdl);
        _Instance = Activator.CreateInstance(_Type, new object[] {_Url, _MethodName});
        _Method = _Type.GetMethod(_MethodName, BindingFlags.Instance | BindingFlags.Public);
    }

    private int _ID;
    private string _Name, _Sign;
    private object _Instance;
    private MethodInfo _Method;
    
    public string Execute(params string[] args)
    {
        return _Method.Invoke(_Instance, args).ToString();
    }
    
    private static Type GetTypeFromWsdl(string name, string wsdl)
    {
        
    }
}
于是,就有了这个完整的方案:
1.定义一个基类,动态生成的类将基于这个类工作:
public class SmsAgentServiceClient : SoapHttpClientProtocol
{
    public SmsAgentServiceClient(string url, string methodName)
    {
        Url = url;
        _MethodName = methodName;
    }

    protected string _MethodName;

    protected string HookInvoke(string[] args) 
    {
        object[] results = Invoke(_MethodName, args);
        return results[0].ToString();
    }
}
2.这个分析Wsdl的代码太冗长,这里忽略。以下的代码解决从wsdl获取足够信息后生成动态代理代码:
    
    // 分析基类
    Type super = typeof(SmsAgentServiceClient);
    ConstructorInfo ctorInfo = super.GetConstructor(
        new Type[] {typeof(string), typeof(string)});
    MethodInfo invokeInfo = super.GetMethod("HookInvoke",
        BindingFlags.Instance | BindingFlags.NonPublic);

    // 建立程序集
    AssemblyName aname = new AssemblyName();
    aname.Name = "SmsAgentServiceClient.Attachments";
    aname.Version = new Version("1.0.0.0");
    AssemblyBuilder assembly = AppDomain.CurrentDomain.DefineDynamicAssembly(aname,
        AssemblyBuilderAccess.Run);
    ModuleBuilder module = assembly.DefineDynamicModule("MyModule");

    // 建立类
    TypeBuilder type = module.DefineType(typeName, TypeAttributes.Class, super);
    Type serviceAttributeType = typeof(WebServiceBindingAttribute);
    ConstructorInfo serviceAttributeCtor = serviceAttributeType.GetConstructor(new Type[] {});
    PropertyInfo serviceAttributeNamespace = serviceAttributeType.GetProperty("Namespace",
        BindingFlags.Instance | BindingFlags.Public);
    type.SetCustomAttribute(new CustomAttributeBuilder(serviceAttributeCtor, new object[] {},
        new PropertyInfo[] {serviceAttributeNamespace}, new object[] {namespaceName}));

    // 建立构造器
    ConstructorBuilder ctor = type.DefineConstructor(MethodAttributes.Public,
        CallingConventions.Standard, new Type[] {typeof(string), typeof(string)});
    // 建立构造器体
    ILGenerator ctorGenerator = ctor.GetILGenerator();
    ctorGenerator.Emit(OpCodes.Ldarg_0);
    ctorGenerator.Emit(OpCodes.Ldarg_1);
    ctorGenerator.Emit(OpCodes.Ldarg_2);
    ctorGenerator.Emit(OpCodes.Call, ctorInfo);
    ctorGenerator.Emit(OpCodes.Ret);

    // 建立方法
    MethodBuilder method = type.DefineMethod(methodName,
        MethodAttributes.Public | MethodAttributes.Final, 
        typeof(string), new Type[] {typeof(string[])});
    method.DefineParameter(1, ParameterAttributes.None, mobile);
    method.DefineParameter(2, ParameterAttributes.None, content);
    Type methodAttributeType = typeof(SoapDocumentMethodAttribute);
    ConstructorInfo methodAttributeCtor = methodAttributeType.GetConstructor(
        new Type[] {typeof(string)});
    method.SetCustomAttribute(new CustomAttributeBuilder(methodAttributeCtor,
        new object[] {namespaceName + methodName}));

    // 建立方法体
    ILGenerator methodGenerator = method.GetILGenerator();
    LocalBuilder p1 = methodGenerator.DeclareLocal(typeof(string));
    Label l1 = methodGenerator.DefineLabel();
    methodGenerator.Emit(OpCodes.Ldarg_0);
    methodGenerator.Emit(OpCodes.Ldarg_1);
    methodGenerator.Emit(OpCodes.Call, invokeInfo);
    methodGenerator.Emit(OpCodes.Stloc, p1);
    methodGenerator.Emit(OpCodes.Br_S, l1);
    methodGenerator.MarkLabel(l1);
    methodGenerator.Emit(OpCodes.Ldloc, p1);
    methodGenerator.Emit(OpCodes.Ret);
    
    return type.CreateType();
结论:
一、在使用Emit生成代码时尽可能采用继承自自有基类,将不变的部分写到基类中,Emit只需要调用基类即可。
二、为简化设计,可以将参数及返回值全部改成字符串型。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多