C#中动态生成WebServer
1.前言
(以下内容是根据别人的博客复制和总结出来的。如果有问题请联系我删除)
(url:https://www.cnblogs.com/dotnet261010/p/12461930.html)这个是有关于WebServer动态调用
(https://blog.csdn.net/liuganhaha/article/details/94719956?ops_request_misc=&request_id=&biz_id=102&utm_term=C?ops_request_misc=&request_id=&biz_id=102&utm_term=C&utm_medium=distribute.pc_search_result.none-task-blog-2~all~sobaiduweb~default-3-94719956.pc_search_result_control_group&spm=1018.2226.3001.4187#%E4%BF%AE%E6%94%B9ConfigurationManager.AppSe&utm_medium=distribute.pc_search_result.none-task-blog-2~all~sobaiduweb~default-3-94719956.pc_search_result_control_group) C# 修改config appSettings节点
2.动态调用
如果发布的WebService地址改变,那么就要重新添加WebService的引用。如果是现有的WebService发生了改变,也要更新现有的服务引用,这需要把代码放到现场才可以。那么有没有什么方式可以解决这种问题呢?那就是使用动态调用WebService的方法。
2.1我们在配置文件里面添加配置,把WebService的地址、WebService提供的类名、要调用的方法名称,都写在配置文件里
添加配置文件
<appSettings> <!--WebService地址--> <add key="WebServiceAddress" value="http://localhost:9008/TestWeb.asmx"/> <!--WebService提供的类名--> <add key="ClassName" value="TestWeb"/> <!--WebService方法名--> <add key="MethodName" value="Test"/> <!--存放dll文件的地址--> <add key="FilePath" value="E:\Test"/> </appSettings>配置文件信息
2.2动态生成Webserver的一个帮助类
using System; using System.CodeDom; using System.CodeDom.Compiler; using System.IO; using System.Net; using System.Text; using System.Web; using System.Web.Caching; using System.Web.Services.Description; using System.Xml.Serialization; namespace WebServiceDemo { public class WebServiceHelper { /// <summary> /// 生成dll文件保存到本地 /// </summary> /// <param name="url">WebService地址</param> /// <param name="className">类名</param> /// <param name="methodName">方法名</param> /// <param name="filePath">保存dll文件的路径</param> public static void CreateWebServiceDLL(string url,string className, string methodName,string filePath ) { // 1. 使用 WebClient 下载 WSDL 信息。 WebClient web = new WebClient(); Stream stream = web.OpenRead(url + "?WSDL"); // 2. 创建和格式化 WSDL 文档。 ServiceDescription description = ServiceDescription.Read(stream); //如果不存在就创建file文件夹 if (Directory.Exists(filePath) == false) { Directory.CreateDirectory(filePath); } if (File.Exists(filePath + className + "_" + methodName + ".dll")) { //判断缓存是否过期 var cachevalue = HttpRuntime.Cache.Get(className + "_" + methodName); if (cachevalue == null) { //缓存过期删除dll File.Delete(filePath + className + "_" + methodName + ".dll"); } else { // 如果缓存没有过期直接返回 return; } } // 3. 创建客户端代理代理类。 ServiceDescriptionImporter importer = new ServiceDescriptionImporter(); // 指定访问协议。 importer.ProtocolName = "Soap"; // 生成客户端代理。 importer.Style = ServiceDescriptionImportStyle.Client; importer.CodeGenerationOptions = CodeGenerationOptions.GenerateProperties | CodeGenerationOptions.GenerateNewAsync; // 添加 WSDL 文档。 importer.AddServiceDescription(description, null, null); // 4. 使用 CodeDom 编译客户端代理类。 // 为代理类添加命名空间,缺省为全局空间。 CodeNamespace nmspace = new CodeNamespace(); CodeCompileUnit unit = new CodeCompileUnit(); unit.Namespaces.Add(nmspace); ServiceDescriptionImportWarnings warning = importer.Import(nmspace, unit); CodeDomProvider provider = CodeDomProvider.CreateProvider("CSharp"); CompilerParameters parameter = new CompilerParameters(); parameter.GenerateExecutable = false; // 可以指定你所需的任何文件名。 parameter.OutputAssembly = filePath + className + "_" + methodName + ".dll"; parameter.ReferencedAssemblies.Add("System.dll"); parameter.ReferencedAssemblies.Add("System.XML.dll"); parameter.ReferencedAssemblies.Add("System.Web.Services.dll"); parameter.ReferencedAssemblies.Add("System.Data.dll"); // 生成dll文件,并会把WebService信息写入到dll里面 CompilerResults result = provider.CompileAssemblyFromDom(parameter, unit); if (result.Errors.HasErrors) { // 显示编译错误信息 System.Text.StringBuilder sb = new StringBuilder(); foreach (CompilerError ce in result.Errors) { sb.Append(ce.ToString()); sb.Append(System.Environment.NewLine); } throw new Exception(sb.ToString()); } //记录缓存 var objCache = HttpRuntime.Cache; // 缓存信息写入dll文件 objCache.Insert(className + "_" + methodName, "1", null, DateTime.Now.AddMinutes(5), TimeSpan.Zero, CacheItemPriority.High, null); } } }
2.3调用帮助类生成动态dll文件,和如何使用生成的动态Webserver接口
当你要使用的时候,要把生成和使用分开写!
/// <summary> /// 动态调用WebService /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void btn_Dynamic_Click(object sender, EventArgs e) { // 读取配置文件,获取配置信息 string url = ConfigurationManager.AppSettings["WebServiceAddress"]; string className = ConfigurationManager.AppSettings["ClassName"]; string methodName = ConfigurationManager.AppSettings["MethodName"]; string filePath = ConfigurationManager.AppSettings["FilePath"]; // 调用WebServiceHelper(这里是生程动态dll文件) WebServiceHelper.CreateWebServiceDLL(url, className, methodName, filePath); //以下是使用动态接口的内容 // 读取dll内容 byte[] filedata = File.ReadAllBytes(filePath + className + "_" + methodName + ".dll"); // 加载程序集信息 Assembly asm = Assembly.Load(filedata); Type t = asm.GetType(className); // 创建实例 object o = Activator.CreateInstance(t); MethodInfo method = t.GetMethod(methodName); // 参数 object[] args = {"动态调用WebService" }; // 调用访问,获取方法返回值 string value = method.Invoke(o, args).ToString(); //输出返回值 MessageBox.Show($"返回值:{value}"); }
2.4修改config appSettings中的信息
因为当我们动态生成Webserver中的url是从app.config中的appSettings节点中获取的。所以我们在修改接口时会动态的修改这个配置文件中的获取的url的值。才能让生成的方法调用到我们刚刚修改过的
值。
private void button1_Click(object sender, EventArgs e) { //获取文本值 string weihao= this.textBox1.Text; //这是我的接口 string path = @"http://localhost:1773/services/testService/"; //拼接接口 string newpath = path+weihao; //一下时开始修改配置节点的方法 ConfigurationManager.RefreshSection("appSettings"); //刷新命名节 //获取配置文件中的对象 System.Configuration.Configuration cfa = ConfigurationManager.OpenExeConfiguration(ConfigurationUserLevel.None); //修改配置节点 cfa.AppSettings.Settings["WebServiceAddress"].Value = newpath; //保存 cfa.Save(); //获取刚刚保存的配置文件的值 string url = ConfigurationManager.AppSettings["WebServiceAddress"]; }