Хобрук: Ваш путь к мастерству в программировании

Фильтрация массива внутри словаря — Swift

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

var clients = Client.loadAllClients()     //Returns client array
var contacts = [String: [Client]]()      //Indexed clients in a dictionary 
var letters: [String] = [] 

var filteredClient = [Client]()
var shouldShowSearchResults = false
var searchController : UISearchController!   

Когда я делаю индексацию, словарь контактов возвращает:

{A: [Client("Andrew")]}

Массив Letters возвращает:

[A]

Я использую UISearchController для отображения отфильтрованного массива клиентов.

func updateSearchResults(for searchController: UISearchController) {
    // how to filter the dictionary 
    self.tableView.reloadData()
}

Однако я понятия не имею, как отфильтровать словарь, чтобы вернуть правильный список клиентов. я пытался использовать

contacts.filter(isIncluded: ((key: String, value: [Client])) throws -> Bool((key: String, value: [Client])) throws -> Bool)

Но я был очень смущен реализацией. Я использую Xcode 8.0 и Swift 3.0.

Если бы кто-нибудь мог указать мне в правильном направлении, это было бы очень признательно. Пожалуйста, дайте мне знать, если мне нужно что-то уточнить. Заранее спасибо. Полный код можно найти на моем Github.


Ответы:


1

Основная проблема заключается в том, что вы используете словарь в качестве источника данных массив.

Мое предложение - использовать пользовательскую структуру в качестве модели.

struct Contact {
   let letter : String
   var clients : [Client]

   init(letter: String, clients : [Client] = [Client]()) {
      self.letter = letter
      self.clients = clients
   } 

   mutating func add(client : Client) {
        clients.append(client)
   }
}

Затем создайте массив источников данных

var contacts = [Contact]()  

и массив букв как вычисляемое свойство

var letters : [String] = {
   return contacts.map{ $0.letter }
}

Легко отсортировать массив по буквам

contacts.sort{ $0.letter < $1.letter }

Теперь вы можете искать / фильтровать таким образом (text - это текст для поиска)

 filteredClient.removeAll()
 for contact in contacts {
     let filteredContent = contact.clients.filter {$0.name.range(of: text, options: [.anchored, .caseInsensitive, .diacriticInsensitive]) != nil }
     if !filteredContent.isEmpty {
         filteredClient.append(filteredContent)
     }
 }

Вы даже можете сохранить разделы (буквы), если объявите filteredClient также как [Contact] и создадите временные экземпляры Contact с отфильтрованными элементами.

Конечно, вам нужно изменить все методы источника/делегата данных табличного представления, чтобы они соответствовали массиву Contact, но оно того стоит. Массив как источник данных более эффективен, чем словарь.

27.03.2017
  • Этот подход хорош, но я думаю, что есть проблема из-за Struct, когда я пытаюсь вызвать метод добавления и добавить новый экземпляр клиента, массив клиентов не получает этот экземпляр. Я проверяю адреса памяти до и после добавления элемента, и они разные. Так вы знаете, в чем здесь проблема, или я делаю что-то не так? 05.12.2018
  • структуры являются типами значений. Во многих случаях экземпляры копируются при назначении новой переменной. 05.12.2018
  • Да, я знаю, что структура является типом значения, и я думаю, что именно поэтому в моем случае я не могу использовать структуру в качестве источника данных для представления секционированной таблицы. Я должен использовать словарь. о чем ты думаешь? 05.12.2018
  • Swift Dictionary также является типом значения. Адрес памяти не имеет значения, важно содержимое. Если вам действительно нужна эталонная семантика, используйте class. 05.12.2018
  • Новые материалы

    ВЫ РЕГРЕСС ЭТО?
    Чтобы понять, когда использовать регрессионный анализ, мы должны сначала понять, что именно он делает. Вот простой ответ, который появляется, когда вы используете Google: Регрессионный..

    Не зря же это называют интеллектом
    Стек — C#, Oracle Опыт — 4 года Работа — Разведывательный корпус Мне пора служить Может быть, я немного приукрашиваю себя, но там, где я живу, есть обязательная военная служба на 3..

    LeetCode Проблема 41. Первый пропущенный положительный результат
    LeetCode Проблема 41. Первый пропущенный положительный результат Учитывая несортированный массив целых чисел, найдите наименьшее пропущенное положительное целое число. Пример 1: Input:..

    Расистский и сексистский робот, обученный в Интернете
    Его ИИ основан на предвзятых данных, которые создают предрассудки. Он словно переходит из одного эпизода в другой из серии Черное зеркало , а вместо этого представляет собой хронику..

    Управление состоянием в микрофронтендах
    Стратегии бесперебойного сотрудничества Микро-фронтенды — это быстро растущая тенденция в сфере фронтенда, гарантирующая, что удовольствие не ограничивается исключительно бэкэнд-системами..

    Декларативное и функциональное программирование в стиле LINQ с использованием JavaScript с использованием каррирования и генератора ...
    LINQ - одна из лучших функций C #, которая обеспечивает элегантный способ написания кода декларативного и функционального стиля, который легко читать и понимать. Благодаря таким функциям ES6,..

    Структуры данных в C ++ - Часть 1
    Реализация общих структур данных в C ++ C ++ - это расширение языка программирования C, которое поддерживает создание классов, поэтому оно известно как C с классами . Он используется для..