【发布时间】:2018-09-19 18:19:47
【问题描述】:
给定一个文件名,比如 foo.txt 和一个基本目录 URL,比如 ~/bar/directory 找到第一次出现的子目录的最佳方法是什么文件?
- Foundation 或 AppKit 中是否存在现有 API?
- 例如,通过子树进行广度优先搜索是否可行?看起来怎么样?
- 另一种方式?
【问题讨论】:
标签: macos appkit foundation
给定一个文件名,比如 foo.txt 和一个基本目录 URL,比如 ~/bar/directory 找到第一次出现的子目录的最佳方法是什么文件?
【问题讨论】:
标签: macos appkit foundation
一种选择是手动枚举目录:
func manualSearchFile(withName name: String, in path: String) {
func search(url: URL) {
do {
let contents = try FileManager.default.contentsOfDirectory(at: url, includingPropertiesForKeys: [.nameKey, .pathKey, .isDirectoryKey], options: [])
try contents.forEach {
let metadata = try $0.resourceValues(forKeys: [.nameKey, .pathKey, .isDirectoryKey])
if metadata.name == name {
print("Manual Found: \(metadata.path ?? "unknown path")")
}
if metadata.isDirectory == true {
search(url: $0)
}
}
} catch {
print(error)
}
}
search(url: URL(fileURLWithPath: path))
}
manualSearchFile(withName: "foo.txt", in: "/bar/directory")
另一种选择是使用 Spotlight,它速度更快,但仅适用于已编入索引的路径。许多系统目录被排除在外,用户可以在“系统偏好设置”>“聚焦”>“隐私”中排除更多。
var metadataQuery: NSMetadataQuery?
func spotlightSearchFile(withName name: String, in path: String) {
NotificationCenter.default.addObserver(forName: .NSMetadataQueryDidFinishGathering, object: nil, queue: nil) {
guard let query = $0.object as? NSMetadataQuery else { return }
query.enumerateResults { (result, index, cancel) in
let item = result as? NSMetadataItem
let path = item?.value(forAttribute: NSMetadataItemPathKey) as? String
print("Spotlight Found: \(path ?? "unknown path")")
}
}
metadataQuery = NSMetadataQuery()
metadataQuery?.searchScopes = [path]
metadataQuery?.predicate = NSPredicate(format: "%K like[cd] %@", NSMetadataItemDisplayNameKey, name)
metadataQuery?.start()
}
spotlightSearchFile(withName: "foo.txt", in: "/bar/directory")
有关查询语法的更多信息,请参阅Comparison of NSPredicate and Spotlight Query Strings。
在过去,我们使用FSCatalogSearch 和FSGetCatalogInfoBulk 有更多选择。但是这些不再可用,AFAIK。
【讨论】:
NSFileManager.enumerate 函数不也提供什么吗?我对偏好匹配文件而不是目录的 BFS 更感兴趣。