[英]Need help converting my RestSharp code to use HttpClient instead
由于我需要将此C#dll转换为要从Visual Basic 6调用的tlp文件,因此我需要避免使用外部依赖项。 我使用RestSharp通过执行以下操作来使用WebAPI:
using RestSharp;
using Newtonsoft.Json;
..
public string GetToken (string Key, string Password) {
var client = new RestClient (BaseUrl + "auth/GetToken");
var request = new RestRequest (Method.POST);
request.AddHeader ("cache-control", "no-cache");
request.AddHeader ("Content-Type", "application/json");
Dictionary<string, string> data = new Dictionary<string, string> {
{ "APIKey", Key },
{ "APIPassword", Password }
};
var dataJSON = JsonConvert.SerializeObject (data);
request.AddParameter ("undefined", dataJSON, ParameterType.RequestBody);
IRestResponse response = client.Execute (request);
GetTokenResults g = JsonConvert.DeserializeObject<GetTokenResults> (response.Content);
return g.Token;
}
其中,GetTokenResults是一个结构,其中包含字符串Token的声明。 我想在不使用RestSharp的情况下实现相同的功能。 这是我失败的尝试:
using System.Net.Http;
using System.Net.Http.Headers;
..
public async void GetToken (string Key, string Password) {
var client = new HttpClient ( );
client.DefaultRequestHeaders.Accept.Clear ( );
client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue ("application/json"));
client.BaseAddress = new Uri (BaseUrl + "auth/GetToken");
Dictionary<string, string> data = new Dictionary<string, string> {
{ "APIKey", Key },
{ "APIPassword", Password }
};
var dataJSON = JsonConvert.SerializeObject (data);
var content = new StringContent (dataJSON, Encoding.UTF8, "application/json");
var response = await client.PostAsync ("", content);
}
我不清楚如何使用HttpClient达到与之前使用RestSharp相同的结果(发送API密钥和密码,以字符串形式返回令牌)。 任何能指出我正确方向的东西都将不胜感激!
我认为您在方法PostAsync中缺少第一个参数,即requestUri = Client.BaseAddress(请参阅下面的实现)。
请先尝试一下,如果没有用,请阅读以下内容。 我有一个不同的实现,我将client.BaseAddress
作为第一个参数传递,并将我的内容作为ByteArrayContent传递。 就我而言,我必须将我的内容作为我的代码的“ application / x-www-form-urlencoded”摘录传递:
var buffer = Encoding.UTF8.GetBytes(content);
var byteContent = new ByteArrayContent(buffer);
//as I can't send JSON, probably, you can skip as it's already JSON
byteContent.Headers.ContentType = new System.Net.Http.Headers.MediaTypeHeaderValue("application/x-www-form-urlencoded");
//requestUri=client.BaseAddress
await client.PostAsync(requestUri, byteContent).ConfigureAwait(false);
我们的需求有所不同,但我认为您非常接近。 如果没有帮助,请写信给我,我将分享我的代码。 阅读评论后,我想分享我如何制作HttpClient。 代码是这样的:
using (var client = CreateMailClientForPOST($"{BaseUrl}/"))
{
//removed code, you can call above code as method like
var response= await client.DoThingAsAsync($"{client.BaseAddress}, content").ConfigureAwait(false);
}
protected HttpClient CreateMailClientForPOST(string resource)
{
var handler = new HttpClientHandler();
if (handler.SupportsAutomaticDecompression)
{
handler.AutomaticDecompression = System.Net.DecompressionMethods.GZip | System.Net.DecompressionMethods.Deflate;
}
var client = new HttpClient(handler)
{
BaseAddress = new Uri($"https://api.address.com/rest/{resource}")
};
return client;
}
我认为您对此问题感到ung恼。 简而言之, client.BaseAddress
的URI在其client.BaseAddress
需要一个斜杠。
但是,我不会简单地添加它,我会考虑做一些不同的事情。 假设您要向其附加"auth/GetToken"
,则您的BaseUrl
可能已经有一个斜杠。 我会这样:
client.BaseAddress = new Uri(BaseUrl);
...
var response = await client.PostAsync("auth/GetToken", content);
如您所见, HttpClient
非常适合您的代码设置方式,即您有一个带有尾部斜杠的“基本”地址,并且您想为特定的调用附加到该地址。
这应该使您摆脱困境。 您需要解决的下一件事是反序列化JSON响应,以便您可以从中获取令牌。 除了在response.Content
不是HttpClient
世界中的字符串之外,它与在RestSharp中的操作类似。因此您还需要执行以下步骤:
var json = await response.Content.ReadAsStringAsync();
GetTokenResults g = JsonConvert.DeserializeObject<GetTokenResults>(json);
return g.Token;
您需要做的最后一件事就是将方法签名更改为:
public async Task<string> GetTokenAsync
最后一点:您现在处于异步世界,这是一件好事,但是您需要知道如何正确使用它,否则可能会遇到死锁和其他神秘的错误。 简而言之, 不要在调用堆栈的任何地方调用.Result
或.Wait()
阻塞异步代码 。 这是迄今为止人们最常见的错误。 使用async/await
一路下跌 。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.