RestSharp解决Encoding乱码问题
RestSharp
是一个第三方开源的Http模拟请求辅助类,其底层实现基于System.Net.HttpWebRequest
,且不依赖于任何第三方控件。其github地址为:https://github.com/restsharp/RestSharp,start数可以说明该类库的知名度,当然侧面也可以证明它的确是一个比较好用的HTTP请求辅助类。
一般情况下,RestSharp
都工作的很好,只是当服务端地址指定了非Utf-8
编码方式时,RestSharp
会开始出现各种乱码问题(请求内容乱码、响应内容乱码)。
对于请求部分的Encoding
问题,通过RestClient.ConfigureHttp
以及Http.CalculateContentLength
这两个方法的源代码,发现我们可以通过设置RestClient.Encoding
来解决请求部分的编码问题,通过RestClient
的构造函数,我们也可以发现其初始值为Encoding.UTF8
,当然别忘了设置请求相应的ContentType
。
var client = new RestClient(url);
client.Encoding = Encoding.GetEncoding("gb2312");
client.AddDefaultHeader("Content-Type", "text/html;charset=gb2312");
而对于IRestResponse
部分,RestSharp
的处理方式就不是那么友好了,通过RestResponseBase.Content
以及MiscExtensions.AsString
,我们可以看到其赤果果的简单粗暴,就是只认Encoding.UTF8
,也就是对于IRestResponse
部分,默认情况下,RestSharp
根本就没考虑过Encoding
问题,IRestResponse.ContentEncoding
和IRestResponse.ContentType
在这里就是摆设!但幸好,IRestResponse.Content
是可读可写的,而且我们也可以通过IRestResponse.RawBytes
来得到服务器原始的响应内容,所以我们可以通过自定义一个扩展来自己完成这个编码的转换过程。
/// <summary>
/// <see cref="RestSharp"/>辅助类
/// </summary>
public static class RestSharpHelper
{
/// <summary>
/// 根据<see cref="IRestResponse.ContentEncoding"/>或<see cref="IRestResponse.ContentType"/>设置<see cref="IRestResponse.Content"/>
/// </summary>
/// <param name="response">Rest响应实体</param>
public static void SetResponseEncoding(this IRestResponse response)
{
var encoding = response.ContentEncoding;
if (string.IsNullOrWhiteSpace(encoding) && !string.IsNullOrWhiteSpace(response.ContentType))
{
var tmp = response.ContentType.Split(';').Select(s => s.Split('='));
var arr = tmp.LastOrDefault(t => t.Length == 2 && t[0].Trim().ToLower() == "charset");
if (arr != null)
{
encoding = arr[1].Trim();
}
}
if (!string.IsNullOrWhiteSpace(encoding))
{
response.SetResponseEncoding(encoding);
}
}
/// <summary>
/// 根据Encoding设置<see cref="IRestResponse.Content"/>
/// </summary>
/// <param name="response">Rest响应实体</param>
/// <param name="encoding">响应内容编码方式</param>
public static void SetResponseEncoding(this IRestResponse response, string encoding)
{
if (!string.IsNullOrWhiteSpace(encoding))
{
response.SetResponseEncoding(Encoding.GetEncoding(encoding));
}
}
/// <summary>
/// 根据Encoding设置<see cref="IRestResponse.Content"/>
/// </summary>
/// <param name="response">Rest响应实体</param>
/// <param name="encoding">响应内容编码方式</param>
public static void SetResponseEncoding(this IRestResponse response, Encoding encoding)
{
if (encoding != null)
{
response.ContentEncoding = encoding.WebName;
response.Content = encoding.GetString(response.RawBytes);
}
}
}
接下来就是在执行请求后简单的调用一次
var request = new RestRequest(Method.POST);
//设置反序列化时预先处理乱码问题,如果调用的是Execute<T>方法,那么后面无需再次调用该方法
request.OnBeforeDeserialization = res => RestSharpHelper.SetResponseEncoding(res, "gb2312");
var response = client.Execute(request);
//直接读取Content还是需要调用该方法
RestSharpHelper.SetResponseEncoding(response, "gb2312");
注意在这里我们对IRestRequest.OnBeforeDeserialization
进行了赋值,该委托主要用于反序列化时解决Encoding
导致的乱码问题,所以说其实RestSharp
是知道有这个问题,但没有对该问题进行处理……
更多推荐
所有评论(0)