Swift OOP:如何封装搜索行为

Swift OOP: How to encapsulate Search behavior

目前我有以下代码:

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...)
            [=11=].value1.lowercased().contains(text.lowercased())
        }
    } else {
        data = completeData
    }
    reloadResults()
}

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

如果我还想匹配 value2value3 怎么办?我如何提取搜索逻辑,以便在不触及主要代码的情况下单独更改它。

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

let searchText = text.lowercased()
[=12=].value1.lowercased().contains(searchText) ||
[=12=].value2.lowercased().contains(searchText) ||
[=12=].value3.lowercased().contains(searchText)
...

是否有更优雅的方法来实现相同的结果?

我希望这应该能解决您正在寻找的问题。如果结构是固定的,那么可以实现如下代码,将比较代码封装在结构中。

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

    func compareText(_ searchText: String) -> Bool {
        return value1.lowercased().contains(searchText.lowercased()) || value2.lowercased().contains(searchText.lowercased()) || value3.lowercased().contains(searchText.lowercased())
    }
}

这将比较所有值而不遗漏。

并更新以下代码行。

[=11=].compareText(text)

主要目的是,您可以检查 main objective 方法是否已添加到结构中,因此它可以在任何地方使用而不是比较 filter 方法中的每个值。

希望对您有所帮助。

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

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

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])

方法二:为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 = {
    [=13=].value1.lowercased().contains(searchText) ||
    [=13=].value2.lowercased().contains(searchText)
}

updateSearchResults(for: searchController, using: filterProc)