Фильтрация объектов по определенному объекту в этом значении - PullRequest
0 голосов
/ 08 апреля 2019

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

@objcMembers class SneakerInfoTemp: Object, Decodable {
//    dynamic var id = UUID().uuidString

    //list of dynamic properties that will be deserialized from the json and then passed into realm
    dynamic var brand: String?
    dynamic var category: String?
    dynamic var colorway: String?
    dynamic var currentdescription: String?
    dynamic var designer: String?
    dynamic var imagesrc: String?
    dynamic var maincolor: String?
    dynamic var name: String?
    dynamic var nickname: String?
    dynamic var price: String?
    dynamic var productlink: String?
    dynamic var productlinkhref: String?
    dynamic var releasedate: String?
    dynamic var silhouette: String?
    dynamic var technology: String?
    dynamic var webscraperorder: String?
    dynamic var webscraperstarturl: String?

    enum CodingKeys: String, CodingKey {
        case brand, category, colorway, designer, imagesrc, maincolor, name, nickname, price, productlink, productlinkhref, releasedate, silhouette, technology,webscraperorder,webscraperstarturl,currentdescription
    }

    required init(from decoder: Decoder) throws
    {
        let container = try decoder.container(keyedBy: CodingKeys.self)

        //sneaker information that will be decoded
        //made optional due to the fact that some fields don't exist with every shoe
        brand = try? container.decode(String.self, forKey: .brand)
        category = try? container.decode(String.self, forKey: .category)
        colorway = try? container.decode(String.self, forKey: .colorway)
        currentdescription = try? container.decode(String.self, forKey: .currentdescription)
        designer = try? container.decode(String.self, forKey: .designer)
        imagesrc = try? container.decode(String.self, forKey: .imagesrc)
        maincolor = try? container.decode(String.self, forKey: .maincolor)
        name = try? container.decode(String.self, forKey: .name)
        nickname = try? container.decode(String.self, forKey: .nickname)

        price = try? container.decode(String.self, forKey: .price)
        productlink = try? container.decode(String.self, forKey: .productlink)
        productlinkhref = try? container.decode(String.self, forKey: .productlinkhref)
        releasedate = try? container.decode(String.self, forKey: .releasedate)
        silhouette = try? container.decode(String.self, forKey: .silhouette)
        technology = try? container.decode(String.self, forKey: .technology)
        webscraperorder = try? container.decode(String.self, forKey: .webscraperorder)
        webscraperstarturl = try? container.decode(String.self, forKey: .webscraperstarturl)
        super.init()
    }


    required init()
    {
        super.init()
    }

    override class func primaryKey() -> String? {
        return "name"
    }

    required init(value: Any, schema: RLMSchema)
    {
        super.init(value: value, schema: schema)
    }

    required init(realm: RLMRealm, schema: RLMObjectSchema)
    {
        super.init(realm: realm, schema: schema)
    }


}

Когда все сказано и сделано, я получаю список объектов. Проблема в том, что некоторые объекты являются дубликатами. Я пытался использовать набор, чтобы удалить их, но набор не может определить разницу между объектами. Я хочу взять этот список объектов и добавить их в набор на основе какого-либо свойства, скажем, имя

Я хочу сделать что-то подобное

   let objectSet = Set((sneakersTemp.sneakers?.brands?.adidas.map { $0.name })!)
        print(objectSet.count)

Возвращает все уникальные имена, но я все равно могу получить все уникальные объекты.

Ответы [ 2 ]

0 голосов
/ 08 апреля 2019

Чтобы удалить дубликаты с помощью Set, тип должен соответствовать Hashable.

. Для упрощения предположим, что класс SneakerInfoTemp определен так:

class SneakerInfoTemp {
    let name: String
    let price: Double

    init(name: String, price: Double) {
        self.name = name
        self.price = price
    }
}

Давайте согласуемся с необходимыми параметрами:

extension SneakerInfoTemp: Equatable {
    static func == (lhs: SneakerInfoTemp, rhs: SneakerInfoTemp) -> Bool {
        return lhs.name == rhs.name
    }
}

extension SneakerInfoTemp: Hashable {
    func hash(into hasher: inout Hasher) {
        hasher.combine(name)
    }
}

Итак, с учетом массива кроссовок:

var array = [SneakerInfoTemp(name: "a", price: 1),
             SneakerInfoTemp(name: "b", price: 2),
             SneakerInfoTemp(name: "a", price: 3),
]

Вы можете создать набор уникальных элементов, используя Set:

let set = Set(array)
0 голосов
/ 08 апреля 2019

Используйте .filter вместо .map. Попробуйте это

let array : [Obj] = sneakersTemp.sneakers?.brands?.adidas
let filtered = filterUnique(array: array)

где Obj - ваш класс / структура для массива sneakersTemp.sneakers?.brands?.adidas (или его суперкласс с именем переменной: String )

func filterUnique(array : [Obj]) -> [Obj] {
    var filtered : [String] = []
    let objectSet = array.filter({ obj -> Bool in
        if filtered.contains(obj.name) {
            return false
        }
        filtered.append(obj.name)
        return true
    })
    print(objectSet.count)
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...