在从 FengNiao 中学习 Swift (一) 中介绍了 Swift Package Manager 的基本用法,然后创建了FengNiaoCopy 项目。
在从 FengNiao 中学习 Swift (二) 中主要介绍如何处理命令行的输入参数。
在从 FengNiao 中学习 Swift (三) 介绍 FengNiao 的字符串搜索规则和 swift 的 protocol 和 extension。
这篇文章主要是介绍 FengNiao 的 图片资源搜索规则。
find 命令
对于命令行程序来说,要进行搜索功能的开发,一个选择是完全自行开发搜索功能,一个是基于系统功能。我们这里的选择是使用系统自带的 find 命令进行搜索,基于 find 命令来做功能开发。关于 find 命令的具体细节大家可以自行搜索了解或者查看参考链接中的资料,这里就不再累赘。我们新建一个 ExtensionFindProcess 类,该类继承自 NSObject ,我们在这个类中做图片资源搜索的相关操作。
class ExtensionFindProcess : NSObject {
}
我们在 ExtensionFindProcess 类的 init 方法里面完成 find 的命令拼接。因为 init 可能返回 nil,所以方法使用 init? 。在 init 里我们可以对 let 的实例常量进行赋值,这是初始化方法的重要特点。在 Swift 中不加修饰的 init 方法都需要在方法中保证所有非 Optional 的实例变量被赋值初始化,这个叫做 init 方法叫做 designated 初始化方法。
// 系统 find 命令处理工具
class ExtensionFindProcess : NSObject {
let p : Process
init?(path:Path,extensions: [String],excluded:[Path]){
// find 命令
p = Process()
// find 命令路径
p.launchPath = "/usr/bin/find"
//
guard !extensions.isEmpty else {
return nil
}
// 参数拼接
var args = [String]()
args.append(path.string)
// 搜索文件的后缀
// 快速枚举某个数组的 EnumerateGenerator,它的元素是同时包含了元素下标索引以及元素本身的多元组
for(i,ext) in extensions.enumerated(){
if i == 0 {
args.append("(")
} else {
args.append("-or")
}
args.append("-name")
args.append("*.\(ext)")
if i == extensions.count - 1 {
args.append(")")
}
}
// 排除路径
for excludedPath in excluded {
args.append("-not")
args.append("-path")
// 文件路径判断
let filePath = path + excludedPath
guard filePath.exists else{
continue
}
// 文件夹处理
if filePath.isDirectory {
args.append("\(filePath.string)/*)")
}else{
args.append(filePath.string)
}
}
p.arguments = args
}
}
写完 init 方法之后,我们为了使用方便增加一个 convenience init 方法,这类方法是 Swift 初始化方法中的 “二等公民”,只作为补充和提供使用上的方便。所有的 convenience 初始化方法都必须调用同一个类中的 designated 初始化完成设置。
// 系统 find 命令处理工具
class ExtensionFindProcess : NSObject {
// ...
convenience init(path: String,extensions: [String], excluded: [String]) {
self.init(path: path, extensions: extensions, excluded: excluded)
}
}
初始化方法完成之后,我们增加一个 execute 方法用于处理 find 的搜索结果,该方法拿到 find 命令的处理结果之后,将结果按 “\n” 分割,然后将最后一个空元素去掉,得到一个图片资源文件路径的 Set 集合。
// 执行结果处理,返回资源文件路径的 Set 集合
func execute() -> Set<String> {
let pipe = Pipe()
p.standardOutput = pipe
let fileHandler = pipe.fileHandleForReading
p.launch()
let data = fileHandler.readDataToEndOfFile()
// 拿到搜索结果之后按 “\n” 分割,然后去掉最后一个空元素
if let string = String(data:data,encoding:.utf8){
return Set(string.components(separatedBy: "\n").dropLast())
}else{
return []
}
}