首页 > 解决方案 > Swift OOP:如何封装搜索行为

问题描述

目前我有以下代码:

import UIKit

struct ToBeSearched {
    var value1 = "1"
    var value2 = "3"
    var value3 = "3"
    var boolean = true
}

var data = [ToBeSearched]()
var completeData = [ToBeSearched]()

public func updateSearchResults(for searchController: UISearchController) {
    if let text = searchController.searchBar.text,
        !text.isEmpty {
        data = completeData.filter{
            // How to encapsulate this behavior, i.e. to extend it to use new values (value2, value2...)
            $0.value1.lowercased().contains(text.lowercased())
        }
    } else {
        data = completeData
    }
    reloadResults()
}

这是一个简单的搜索代码,可以找到value1包含搜索文本的所有值。

如果我也想匹配value2andvalue3怎么办?我怎样才能提取搜索逻辑,以便可以单独更改它,而无需触及主代码。

目前,我必须使用binary OR运算符来处理所有案例:

let searchText = text.lowercased()
$0.value1.lowercased().contains(searchText) ||
$0.value2.lowercased().contains(searchText) ||
$0.value3.lowercased().contains(searchText)
...

有没有更优雅的方式来实现相同的结果?

标签: swiftoopsearchencapsulationobject-oriented-analysis

解决方案


方法 1:指定要搜索的属性[KeyPath]

如果您只是想灵活地指定要搜索结构的哪些字段,您可以传入一个要搜索的属性ToBeSearched数组,并使用其中的闭包来检查 keyPaths 标识的任何属性是否包含您要搜索的文本搜索:[KeyPath]containsfilter

public func updateSearchResults(for searchController: UISearchController, using keyPaths: [KeyPath<ToBeSearched, String>]) {
    if let text = searchController.searchBar.text,
        !text.isEmpty {
        data = completeData.filter { element in
            keyPaths.contains { keyPath in element[keyPath: keyPath].lowercased().contains(text.lowercased()) }
        }
    } else {
        data = completeData
    }
    reloadResults()
}

例子:

搜索value1value2

updateSearchResults(for: searchController, using: [\.value1, \.value2])

方法2:为方法传入一个闭包filter

public func updateSearchResults(for searchController: UISearchController, using filterProc: (ToBeSearched) -> Bool)  {
    if let text = searchController.searchBar.text,
        !text.isEmpty {
        data = completeData.filter(filterProc)
        }
    } else {
        data = completeData
    }
    reloadResults()
}

例子:

let filterProc: (ToBeSearched) -> Bool = {
    $0.value1.lowercased().contains(searchText) ||
    $0.value2.lowercased().contains(searchText)
}

updateSearchResults(for: searchController, using: filterProc)

推荐阅读