美文网首页
从 FengNiao 中学习 Swift (四)

从 FengNiao 中学习 Swift (四)

作者: 要上班的斌哥 | 来源:发表于2017-12-14 21:26 被阅读52次

    从 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/

    相关文章

      网友评论

          本文标题:从 FengNiao 中学习 Swift (四)

          本文链接:https://www.haomeiwen.com/subject/dxhoixtx.html