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包含搜索文本的所有值。

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

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

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

有没有更优雅的方式来达到相同的结果?

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

如果您只想灵活地指定要搜索ToBeSearched结构的字段,则可以传入要搜索的属性[KeyPath]数组,并在 filter 中使用带有闭包的 contains 来检查 keyPath 标识的任何属性是否包含您要搜索的文本:

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)

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

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())
    }
}

这将比较所有值而不会丢失。

并更新以下代码行。

$0.compareText(text)
主要

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

我希望这对你有帮助。

最新更新