Сценарий: Мне нужно экспортировать файл excel, который будет содержать список частей. Мы предоставили пользователю возможность выбирать столбцы и получать данные только выбранных столбцов в экспортированном файле. Поскольку это динамический отчет, я не использую какой-либо конкретный класс для сопоставления отчета, так как это приведет к экспорту пустых заголовков столбцов в отчете, что не нужно. Я использую Dynamic Linq для решения этого сценария.
У меня есть список динамических объектов, полученных из динамического linq.
[
{"CleanPartNo":"Test","Description":"test","AliasPartNo":["258","145","2313","12322"]},
{"CleanPartNo":"Test1","Description":"test1","AliasPartNo":[]}
]
Как я могу получить 4 строки из этого json, например
Обратите внимание, что я не могу использовать строго типизированный объект для десериализации/сопоставления его с помощью JSON.Net.
Обновлять Ниже приведен код:
public class Part
{
public int Id { get; set; }
public string CleanPartNo { get; set; }
public string Description { get; set; }
public List<PartAlias> AliasPartNo { get; set; }
}
public class PartAlias
{
public int PartId { get; set; }
public int PartAliasId { get; set; }
public string AliasPartNo { get; set; }
}
var aliases = new List<PartAlias> {
new PartAlias{AliasPartNo = "258" },
new PartAlias{AliasPartNo = "145" },
new PartAlias{AliasPartNo = "2313" },
new PartAlias{AliasPartNo = "12322" }
};
List<Part> results = new List<Part> {
new Part{CleanPartNo = "Test", Description= "test", PartAlias=aliases },
new Part{CleanPartNo = "Test1", Description= "test1" }
};
var filters = "CleanPartNo,Description, PartAlias.Select(AliasPartNo) as AliasPartNo";
var dynamicObject = JsonConvert.SerializeObject(results.AsQueryable().Select($"new ({filters})"));
в переменной dynamicObject я получаю упомянутый выше json
Я обновил код, пожалуйста, проверьте
мы можем знать имена столбцов, которые являются массивами? Это сделает этот 2 лайнера





Отказ от ответственности: Следующее опирается на анонимные классы, что не совсем то же самое, что и динамический LINQ (совсем нет), но я решил, что это все равно может помочь, в зависимости от ваших потребностей, поэтому я решил опубликовать это.
Чтобы сгладить список, вы можете использовать вложенный Select, за которым следует SelectMany (Отказ от ответственности: Предполагается, что каждая часть имеет хотя бы один псевдоним, см. ниже полный код)
var flattenedResult = result.Select(part => part.AliasPartNumber.Select(alias => new
{
CleanPartNo = part.CleanPartNo,
Description = part.Description,
AliasPartNo = alias.AliasPartNo
})
.SelectMany(part => part);
Сначала вы проецируете свои элементы из result (внешнего Select). Проекция проецирует каждый элемент на IEnumerable анонимного типа, в котором каждый элемент соответствует номеру псевдонима. Поскольку внешний Select даст IEnumerable<IEnumerable> (или что-то подобное), мы используем SelectMany, чтобы получить один IEnumerable всех элементов из вашего вложенного IEnumerables. Теперь вы можете сериализовать этот IEnumerable экземпляров анонимного класса с помощью JsonConvert.
var json = sonConvert.SerializeObject(flatResults);
Если псевдонимов нет, внутренний выбор даст пустой IEnumerable, поэтому нам придется ввести особый случай.
var selector = (Part part) => part.AliasPartNumber?.Any() == true
? part.AliasPartNumber.Select(alias => new
{
CleanPartNo = part.CleanPartNo,
Description = part.Description,
AliasPartNo = alias.AliasPartNo
})
: new[]
{
new
{
CleanPartNo = part.CleanPartNo,
Description = part.Description,
AliasPartNo = alias.AliasPartNo
}
};
var flattenedResult = result.Select(selector).SelectMany(item => item);
Этот ответ каким-то образом дал мне некоторое представление, но это не то, чего я на самом деле хочу. Я не могу использовать конкретные типы.
@AliHasan Тогда, может быть, вы могли бы дать больше контекста в вопросе. Это может помочь дать полезный ответ.
Из предоставленного вами json вы можете получить значения, сгруппированные по их имени, следующим образом:
var array = JArray.Parse(json);
var lookup = array.SelectMany(x => x.Children<JProperty>()).ToLookup(x => x.Name, x => x.Value);
тогда это просто способ простого цикла поиска для заполнения столбцов excel.
Тем не менее, я бы предложил сделать сглаживание перед JSON. Некоторое время я пытался сделать это, даже не зная имен столбцов, которые являются массивами, но мне это не удалось, и, поскольку это ваша работа, я больше не буду пытаться: P
Я думаю, что лучшим способом здесь было бы реализовать собственный преобразователь, который просто умножал бы объекты на свойства, являющиеся массивами. Если вы сделаете это хорошо, вы получите бесконечные уровни совершенно бесплатно.
Не могли бы вы предоставить минимальный воспроизводимый пример? В частности, как настроить этот код для запуска в консольном приложении?