Возвращаемся к проекту на C#.
Мы остановились на списке материалов.
Реализуем получение списка материалов выбранного продукта с помощью HTTP-запроса к АПИ, которое мы реализовали в прошлой теме.
Напоминаю, GET-запрос на локальный компьютер:
GET http://localhost:8080/Material?product_id=1
Authorization: Basic esmirnov 111103
На C# нам надо решить две задачи:
Для HTTP-запросов будем использовать встроенный класс HttpClient
var client = new HttpClient();
В АПИ мы использовали "базовую" авторизацию. В C# я не нашёл встроенных библиотек для облегчения формирования заголовка для "базовой" авторизации
Формируем base64-кодированную строку:
var basic = Convert.ToBase64String(
ASCIIEncoding.ASCII.GetBytes("esmirnov:111103"));
И добавляем заголовок нашему http-клиенту:
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", basic);
Теперь можно запрашивать данные.
У HttpClient все методы асинхронные, нужно понимать как работают async/await. Я для облегчения вашей работы нарисовал синхронную реализацию:
// в параметрах URL
private string GetString(string url){
var basic = Convert.ToBase64String(
ASCIIEncoding.ASCII.GetBytes("esmirnov:111103"));
var client = new HttpClient();
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", basic);
return client.GetStringAsync(url).Result;
}
Сначала допишем интерфейс нашего поставщика данных:
interface IDataProvider
{
...
IEnumerable<MaterialTC> GetMaterials(int ProductId);
}
Здесь я класс назвал не Material, а MaterialTC, потому что АПИ возвращает не весь объект Material, а только название и количество (TitleCount).
И реализуем его:
[DataContract]
internal class MaterialTC
{
[DataMember]
public string Title { get; set; }
[DataMember]
public int Count { get; set; }
}
// В ответе у нас не сразу массив материалов, а два вложенных объекта, поэтому надо их тоже описать
[DataContract]
internal class Notice
{
[DataMember]
public Material[] data { get; set; }
}
[DataContract]
internal class Answer
{
[DataMember]
public Notice notice { get; set; }
}
public IEnumerable<MaterialTC> GetMaterials(int ProductId) {
var result = new List<MaterialTC>();
var resp = GetString($"http://localhost:8080/Material?product_id={ProductId}");
var Serializer = new DataContractJsonSerializer(typeof(Answer));
using (var sr = new StreamReader(new MemoryStream(Encoding.UTF8.GetBytes(resp))))
{
var answer = (Answer)Serializer.ReadObject(sr.BaseStream);
foreach (MaterialTC material in answer.notice.data)
{
result.Add(material);
}
}
return result;
}
Стандартный вариант слишком монстрообразный, на мой взгляд. Можно тоже самое реализовать через регулярки:
Класса нам достаточно одного:
internal class MaterialTC
{
public string Title { get; set; }
public int Count { get; set; }
}
Реализация метода GetMaterials:
public IEnumerable<MaterialTC> GetMaterials(int ProductId) {
var result = new List<MaterialTC>();
var resp = GetString($"http://localhost:8080/Material?product_id={ProductId}");
Regex regex = new Regex(@"\{""Title"":""(.*?)"",""Count"":(.*?)\}", RegexOptions.Singleline);
MatchCollection matches = regex.Matches(resp);
if (matches.Count > 0)
{
foreach (Match match in matches)
res.Add(new MaterialTC {
Title = match.Groups[1].ToString(),
Count=Convert.ToInt32(match.Groups[2].ToString())
});
}
return result;
}
Надо, конечно, ещё проверить валидность ответа (notice->data|notice->answer), но с регулярками это тоже проще - вообще не нужно рисовать новый класс.
Оказывается на WorldSkills можно использовать не только "голый" .NET Framework, но и библиотеки из других компонентов Visual Studio.
В пространстве имён System.Web.Script.Serialization есть класс JavaScriptSerializer, который выглядит попроще чем классическая реализация:
// целевые классы нам по прежнему нужны, но уже без всяких аннотаций
internal class MaterialTC
{
public string Title { get; set; }
public int Count { get; set; }
}
internal class Notice
{
public Material[] data;
}
internal class Answer
{
public Notice notice;
}
// в месте, где нам нужно распарсить JSON создаем сериализатор и разбираем строку
var serializer = new JavaScriptSerializer();
var answer = serializer.Deserialize<Answer>("тут ваша JSON-строка");
// и ВСЁ
Возможно понадобится что-то послать в АПИ, разберём как это делается:
// сначала запихиваем объект в JSON-строку.
// тут можно завести отдельный класс, а можно создать анонимный объект
var obj = new {username="qq", password="ww"};
// и тем же сериализатором превращаем в JSON-строку
var jsonString = serializer.Serialize(obj);
// создаём контент для запроса
var json = new StringContent(jsonString, Encoding.UTF8, "application/json");
// и вызываем POST-запрос
var client = new HttpClient();
var result = client.PostAsync("http://localhost:8080/echo", json).Result;
Console.WriteLine(result.Content.ReadAsStringAsync().Result);