Итак, у меня есть список предметов, например, кроссовки. Я получаю эти объекты, анализируя данные 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)
Он возвращает все уникальные имена, но в любом случае я могу получить все уникальные объекты.
Затем базовое равенство на имени
да, но вернет ли он мне объекты? или просто имена снова
@ielyamani не могли бы вы показать мне пример, пожалуйста
Используйте .filter
вместо .map
. Попробуй это
let array : [Obj] = sneakersTemp.sneakers?.brands?.adidas
let filtered = filterUnique(array: array)
где Obj
— ваш класс/структура для массива sneakersTemp.sneakers?.brands?.adidas
(или его суперкласс с имя переменной: строка)
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)
}
собираюсь попробовать это сейчас. если это работает плохо, примите ответ
Как дела?
случай, который вы указали, специфичен для моего примера или сейчас
потому что objsetSet по-прежнему содержит все значения, когда я использую ваш код
каковы ваши условия для фильтрации?
Я хочу, чтобы он не добавлялся в набор, если там есть объект с таким же именем, как у него
есть идеи случайно?
я изменил ответ на ваш случай
Чтобы удалить дубликаты с помощью 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)
попробую и это
Пользователь
.filter
вместо.map