题目有些大,但文中谈到的问题很小;看似表扬C#,实际不是。
这个小问题来自这样的应用场景——以HTTP POST的方式调用第三方API,第三方API不支持JSON传参,只能通过URL query string方式传参(a=1&b=2)。
假设API的地址是http://www.cnblogs.com/api/say,需要传递的参数是username与words,只支持HTTP POST调用。
另外,加一个约束条件——不允许用字符串拼接,比如:"username="+username+"&words="+words;
jQuery中的调用示例代码
var postData = { username: 'test', words: 'hello world' }; $.ajax({ url: 'http://www.cnblogs.com/api/say', data: postData, type: 'post', });
在上面的Javascript代码执行时,jQuery会自动将js对象postData转换为Url query string的形式(username=test&words=hello+world),并自动进行Url encode。
PHP中的调用示例代码
$url = 'http://www.cnblogs.com/api/say'; $data = array('username' => 'test', 'words' => 'hello world'); $data = http_build_query($data); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_POSTFIELDS, $data); $response = curl_exec($ch); curl_close($ch);
PHP内置的http_build_query()函数能自动将数组转换为Url query string的形式(username=test&words=hello+world),并自动进行Url encode。
C#中的调用示例代码
1、在.NET Framework 4.0中的实现(HttpWebRequest+匿名类型+反射+LINQ)
参数通过匿名类型(Anonymous Type)进行定义:
var postData = new { username = "test", words = "hello world" };
.NET Framework 4.0的类库中没有提供直接将“匿名类型实例”转换为“Url查询参数”的API,只能借助“反射+LINQ”自己实现。实现代码如下:
static void Main(string[] args) { var url = "http://www.cnblogs.com/api/say"; var postData = new { username = "test", words = "hello world" }; var webRequest = WebRequest.Create(url) as HttpWebRequest; webRequest.Method = "post"; webRequest.ContentType = "application/x-www-form-urlencoded"; var queryString = string.Join("&", from p in postData.GetType().GetProperties() select p.Name + "=" + HttpUtility.UrlEncode(p.GetValue(postData, null).ToString())); using (var sw = new StreamWriter(webRequest.GetRequestStream())) { sw.Write(queryString); } using (var response = webRequest.GetResponse()) { using (var sr = new StreamReader(response.GetResponseStream())) { Console.WriteLine(sr.ReadToEnd()); } } }
2、在.NET Framework 4.5中的实现(HttpClient+FormUrlEncodedContent)
.NET Framework 4.5考虑到了这个应用场景,提供了FormUrlEncodedContent,但它不支持匿名类型(Anonymous Type),只支持字典(Dictionary)。参数需要这样定义:
var postData = new Dictionary<string, string> { { "username", "test" }, { "words", "hello world" } };
完整实现代码如下(需要引用System.Net.Http):
static void Main(string[] args) { var url = "http://www.cnblogs.com/api/say"; var postData = new Dictionary<string, string> { { "username", "test" }, { "words", "hello world" } }; var urlEncodedContent = new FormUrlEncodedContent(postData); var httpClient = new HttpClient(); var result = httpClient.PostAsync(url, urlEncodedContent).Result.Content.ReadAsStringAsync().Result; Console.WriteLine(result); }
.NET Framework 4.5中的实现还算简单,但是FormUrlEncodedContent只支持Dictionary,考虑还是不周到。
感想
.NET因互联网而生,而通过URL query string传参的需求在互联网应用中普通存在,但.NET从4.5才开始考虑这个应用场景,实在有点说不过去。
多数开发互联网应用多年的.NET开发者都有多年拼接字符串的经历,但是.NET也没考虑到这个场景,比如双引号问题(字符串不支持单引号内直接包含双引号)。即使拼接字符串,也没有Javascript与PHP中操作方便。
.NET功能强大、设计领先,但是对互联网应用场景缺少细致入微的考虑,在用.NET开发互联网应用时经常有杀鸡用牛刀的感觉。幸亏牛刀上有非常舒服的刀柄(Visual Studio),才吸引了如些多的开发者。如果互联网应用是未来,即使刀柄再舒服,用牛刀杀鸡的感觉毕竟不好,牛刀自身的改变才是解决之道。