从 FengNiao 中学习 Swift (四)

从 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 []
        }
        
    }

参考

  1. https://www.ibm.com/support/knowledgecenter/zh/ssw_aix_72/com.ibm.aix.cmds2/find.htm
  2. http://swifter.tips/init-keywords/
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。