问题:不支持Dictionary;结果:在Web Service中傳送Dictionary
在Web Service中傳送Dictionary
有個需求,想在Web Service中傳遞Dictionary<string, string>參數,例如:
[WebMethod]
public Dictionary<string, string> Process(Dictionary<string, string> dct)
{
//Do something on the Dictionary
//... blah blah blah ....
return dct;
}
天不從人願,以上的寫法會產生Web Service不支援IDictionary的錯誤:
The type System.Collections.Generic.Dictionary`2[[System.String, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089],[System.String, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]] is not supported because it implements IDictionary.
既是IDictionary的原罪,就算是換用Hashtable、ListDictionary應該也是同樣結果,測試之下果然無一倖免。
Google發現討論區有篇來自MS RD的留言,算是證實這是先天限制:
ASMX web services do not support types that implement IDictionary since V1. This is by design and was initially done since key-value constraints could not be appropriately expressed in schema.
來源: http://social.msdn.microsoft.com/Forums/en-US/asmxandxml/thread/d7cb8844-6774-4a98-8aa3-85e445af4867/
既然是By Design,就只有繞道而行。Survey了一下,找到一些建議做法:
- 改用XML
- 使用DataSet
- 另外自訂Class作為參數
- 以DictionaryEntry[]瓜代之
評估了一下,我原本想要借重的就是Dictionary
Key/Value的單純資料結構,XML為開放格式不易限制成Key/Value的形式;小小需求動用到DataSet略嫌笨重;自訂Class在編譯
時期就要確定Key的種類,不符本案例的前題。看來DictionaryEntry[]較合需求,因此我試寫如下:
(剛好Dictionary與DirectionaryEntry的雙向轉換都有示範到)
[WebMethod]
public DictionaryEntry[] Test(System.Collections.DictionaryEntry[] entries)
{
//用ListDictionary主要是為了稍後可以直接CopyTo轉DictionaryEntry[]
//若有效率或其他考量,可改用其他Collection Class
ListDictionary dct = new ListDictionary();
foreach (DictionaryEntry de in entries)
dct.Add(de.Key, de.Value);
//Do something on the Dictionary
//... blah blah blah ....
if (dct.Contains("Kuso"))
dct["Kuso"] = "殺很大";
DictionaryEntry[] result = new DictionaryEntry[dct.Count];
dct.CopyTo(result, 0);
return result;
}
呼叫端範例如下:
protected void Page_Load(object sender, EventArgs e)
{
localhost.AFAWebService aws = new localhost.AFAWebService();
aws.Credentials = CredentialCache.DefaultCredentials;
Dictionary<string, string> dct = new Dictionary<string, string>();
dct.Add("Kuso", "你不要走");
//DictionaryEntry在Web Service傳遞時會被當成自訂類別
//因此要用namespace.DictionaryEntry而非System.Collections.DictionaryEntry
List<localhost.DictionaryEntry> lst = new List<localhost.DictionaryEntry>();
foreach (string key in dct.Keys)
{
localhost.DictionaryEntry de = new localhost.DictionaryEntry();
de.Key = key;
de.Value = dct[key];
lst.Add(de);
}
localhost.DictionaryEntry[] result = aws.Test(lst.ToArray());
Dictionary<string, string> dctRes = new Dictionary<string, string>();
foreach (localhost.DictionaryEntry de in result)
dctRes.Add(de.Key.ToString(), de.Value.ToString());
Response.Write(dct["Kuso"] + "->" + dctRes["Kuso"]);
Response.End();
}
經過這番來回折騰,這方法看來也不怎麼簡潔。
於是,我又嘗試了Paul Welter的SerializableDictionary物件, 做法上要在Web Service與Client端都Reference這個自訂物件,而且使用Visual Studio的Add Web Reference時,自動產生的Proxy Class宣告中SerializableDictionary會被當成DataSet而失敗,因此得改成手動產生Proxy Class後再將DataSet改回SerializableDictionary:
Microsoft (R) Web Services Description Language Utility
[Microsoft (R) .NET Framework, Version 2.0.50727.42]
Copyright (C) Microsoft Corporation. All rights reserved.
Writing file 'AfaWebServiceProxy.cs'.
用了SerializableDictionary後,程式碼簡化許多:
[WebMethod]
public SerializableDictionary<string, string> Test(
SerializableDictionary<string, string> dct)
{
if (dct.ContainsKey("Kuso"))
dct["Kuso"] = "殺很大";
return dct;
}
呼叫端也很單純:
protected void Page_Load(object sender, EventArgs e)
{
localhost.AFAWebService aws = new localhost.AFAWebService();
aws.Credentials = CredentialCache.DefaultCredentials;
SerializableDictionary<string, string> dct =
new SerializableDictionary<string, string>();
dct.Add("Kuso", "你不要走");
SerializableDictionary<string, string> dctRes = aws.Test(dct);
Response.Write(dct["Kuso"] + "->" + dctRes["Kuso"]);
Response.End();
}
但是,這個做法需要在Web Service與Client端加入自訂元件參照、Proxy Class需要手動增加或修改,還是有些許不便。這樣看來,DataSet或XML法雖有其他缺點,但內建支援的特點,在力求簡單的場合裡,倒也值得納入考量吧!
问题:不支持Dictionary;结果:在Web Service中傳送Dictionary的更多相关文章
- 在Web Service中傳送Dictionary
有個需求,想在Web Service中傳遞Dictionary<string, string>參數,例如: 排版顯示純文字 [WebMethod] public Dictionary< ...
- Web Service 中返回DataSet结果大小改进
http://www.cnblogs.com/scottckt/archive/2012/11/10/2764496.html Web Service 中返回DataSet结果方法: 1)直接返回Da ...
- Web Service 中返回DataSet结果的几种方法
Web Service 中返回DataSet结果的几种方法: 1)直接返回DataSet对象 特点:通常组件化的处理机制,不加任何修饰及处理: 优点:代码精减.易于处理,小数据量处理较快: ...
- ASP.NET Web Service中使用Session 及 Session丢失解决方法 续
原文:ASP.NET Web Service中使用Session 及 Session丢失解决方法 续 1.关于Session丢失问题的说明汇总,参考这里 2.在Web Servcie中使用Sessio ...
- 转-Web Service中三种发送接受协议SOAP、http get、http post
原文链接:web服务中三种发送接受协议SOAP/HTTP GET/HTTP POST 一.web服务中三种发送接受协议SOAP/HTTP GET/HTTP POST 在web服务中,有三种可供选择的发 ...
- 企业级SOA之路——在Web Service中使用HTTP和JMS
原文:http://www.tibco.com/resources/solutions/soa/enterprise_class_soa_wp.pdf 概述 IT业界在早期有一种误解,认为 ...
- Web Service中的几个重要术语
WSDL:web service definition language 直译:WebService定义语言 1.对应一种该类型的文件.WSDL 2.定义了Web Service的服务器与客户端应用交 ...
- Web Service中的XFire 传输List 自定义对象.
我把这个创建的步骤和代码的贴出来,. 首先新建一个工程,取名就随便点啦..MyWebService,然后复制jar包到lib目录下, 创建包,建立接口..写一个javaBean的类, 以下是一个简单的 ...
- Dictionary(支持 XML 序列化),注意C#中原生的Dictionary类是无法进行Xml序列化的
/// <summary> /// Dictionary(支持 XML 序列化) /// </summary> /// <typeparam name="TKe ...
随机推荐
- sql中的group by 和 having 用法
sql中的group by 用法:-- Group By语句从英文的字面意义上理解就是“根据(by)一定的规则进行分组(Group)”.--它的作用是通过一定的规则将一个数据集划分成若干个小的区域,然 ...
- node操作mongdb的常用函数示例
node操作mongdb的常用函数示例 链接数据库 var mongoose = require('mongoose'); //引用数据库模块 mongoose.connect('mongodb:// ...
- 去除sql中不可见字符的n种方法
CREATE TABLE [ASCII0127] ( Bin INT, Dec INT, Hex VARCHAR(128), Abbr ...
- java:system根据输入的内容,然后输出(字节流)
把输入的内容输出来:根据system.in的内容System.out.println输出出来 都是字节流,的形式: //限制读取的字符长度 //字节流 InputStream ips = System ...
- MySQL 分区知识点(一 )
前言: 查了下资料,关于 MySQL 分区的博文讲的详细的比较少,也不全,只好在官网去翻译英文文章看了.大体整理了一下记录起来: MySQL 分区类型: 1.RANGE 分区: // 这种类型的分区基 ...
- 求一些数字字符参数的和(Java)
一.思路 输入数字(字符型)参数: 将字符型强制转化为整数型: 求和: 输出: 二.流程图 三.源程序代码及结果
- CodeForces-831A-Unimodal Array (水题)
题目链接 /* Name: Copyright: Author: Date: 2018/5/6 19:34:23 Description: */ #include <iostream> # ...
- hdoj-1013-Digital Roots(九余数定理)
题目链接 #include <iostream> using namespace std; int main() { string a; int b; ") { b = ; ;i ...
- SQL使用指南(2)—— 约束的使用
主键约束 (1)创建表时添加主键约束 primary key<column_name> (2) 修改表时添加主键约束 ALTER TABLE table_name ADD CONSTRAI ...
- LeetCode 293. Flip Game
原题链接在这里:https://leetcode.com/problems/flip-game/description/ 题目: You are playing the following Flip ...