题目有些大,但文中谈到的问题很小;看似表扬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),才吸引了如些多的开发者。如果互联网应用是未来,即使刀柄再舒服,用牛刀杀鸡的感觉毕竟不好,牛刀自身的改变才是解决之道。

点赞(0)

评论列表 共有 0 条评论

暂无评论
立即
投稿
发表
评论
返回
顶部