Как получить несколько объектов по отношению к многозначному столбцу в Dynamic Linq

Сценарий: Мне нужно экспортировать файл excel, который будет содержать список частей. Мы предоставили пользователю возможность выбирать столбцы и получать данные только выбранных столбцов в экспортированном файле. Поскольку это динамический отчет, я не использую какой-либо конкретный класс для сопоставления отчета, так как это приведет к экспорту пустых заголовков столбцов в отчете, что не нужно. Я использую Dynamic Linq для решения этого сценария.

У меня есть список динамических объектов, полученных из динамического linq.

[
    {"CleanPartNo":"Test","Description":"test","AliasPartNo":["258","145","2313","12322"]},
    {"CleanPartNo":"Test1","Description":"test1","AliasPartNo":[]}
]

Как я могу получить 4 строки из этого json, например

Как получить несколько объектов по отношению к многозначному столбцу в Dynamic Linq

Обратите внимание, что я не могу использовать строго типизированный объект для десериализации/сопоставления его с помощью 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

Не могли бы вы предоставить минимальный воспроизводимый пример? В частности, как настроить этот код для запуска в консольном приложении?

Enigmativity 29.01.2019 10:37

Я обновил код, пожалуйста, проверьте

Ali Hasan 29.01.2019 11:09

мы можем знать имена столбцов, которые являются массивами? Это сделает этот 2 лайнера

Krzysztof Skowronek 29.01.2019 15:28
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
1
3
104
2

Ответы 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);

Этот ответ каким-то образом дал мне некоторое представление, но это не то, чего я на самом деле хочу. Я не могу использовать конкретные типы.

Ali Hasan 29.01.2019 12:21

@AliHasan Тогда, может быть, вы могли бы дать больше контекста в вопросе. Это может помочь дать полезный ответ.

Paul Kertscher 29.01.2019 13:22

Из предоставленного вами json вы можете получить значения, сгруппированные по их имени, следующим образом:

var array = JArray.Parse(json);
var lookup = array.SelectMany(x => x.Children<JProperty>()).ToLookup(x => x.Name, x => x.Value);

тогда это просто способ простого цикла поиска для заполнения столбцов excel.

Тем не менее, я бы предложил сделать сглаживание перед JSON. Некоторое время я пытался сделать это, даже не зная имен столбцов, которые являются массивами, но мне это не удалось, и, поскольку это ваша работа, я больше не буду пытаться: P

Я думаю, что лучшим способом здесь было бы реализовать собственный преобразователь, который просто умножал бы объекты на свойства, являющиеся массивами. Если вы сделаете это хорошо, вы получите бесконечные уровни совершенно бесплатно.

Другие вопросы по теме