博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
给swift程序猿留下深刻印象的10个Swift代码
阅读量:5906 次
发布时间:2019-06-19

本文共 4991 字,大约阅读时间需要 16 分钟。

通过使用单行代码完成同样的 10 个练习,我们来看看 Swift 和其他语言之间的较量。

  

  将数组中每个元素的值乘以 2

使用map来实现

var arr = [1,2,3,4];        var newArr = arr.map{$0*2}        for item in newArr        {            print(item)        }

代码简单明了地完成了数组元素乘2

 

  求一组数字的和

  这个问题可以通过使用 reduce 方法和加号运算符解决,这是因为加号运算符实际上也是一个函数。不过这个解法是非常显而易见的,待会儿我们会看到 reduce 方法更具有创造力的使用。

var arr = [1,2,3,4];                let sum = arr.reduce(0, combine: +)                print(sum)

 

检索字符串中含有某个单词

我们使用 contains方法判断一个字符串中是否至少含有一个被选中的关键字:

let arr = ["ForrestWoo","Swift1"]        let str = "My name is ForrestWoo,I am learning Swift"        let query = arr.contains(str.containsString)        print(query)

 

 

  读取一个文件

 

        let path = NSBundle.mainBundle().pathForResource("filter", ofType: "rtf")

        let lines = try? String(contentsOfFile: path!).characters.split{$0 == "\n"}.map(String.init)

        for item in lines! {

            print(item)

        }

 

 

 

 

  祝你生日快乐

let name = "Forrest"        (1...4).forEach{print("Happy Birthday " + (($0 == 3) ? "dear \(name)":"to You"))}

 

这段代码会将“祝你生日快乐”这首歌的歌词输出到控制台中,它在一段区间内简单的使用了 map 函数,同时也用到了三元运算符。

 

  数组过滤

假设我们需要使用一个给定的过滤函数将一个序列(sequence)分割为两部分。很多语言除了有常规的mapflatMapreducefilter 等函数外,还有一个 partitionBy 函数恰好可以完成这个需求。正如你所知,Swift 没有类似的函数(我们不想在这里使用 NSArray 中的函数,并通过NSPredicate 实现过滤功能)。

所以,我们可以通过拓展 SequenceType,并为它添加 partitionBy 函数来解决这个问题。我们使用这个函数将整数数组分割为两部分:

extension SequenceType{    typealias Element = Self.Generator.Element         func partitionBy(fu: (Element)->Bool)->([Element],[Element]){        var first=[Element]()        var second=[Element]()        for el in self {            if fu(el) {                first.append(el)            }else{                second.append(el)            }        }        return (first,second)    }} let part = [82, 58, 76, 49, 88, 90].partitionBy{$0 < 60}part // ([58, 49], [82, 76, 88, 90])

 

实际上,这不是单行代码,而且使用了命令式的解法。能不能使用 filter 对它略作改进呢?

extension SequenceType{         func anotherPartitionBy(fu: (Self.Generator.Element)->Bool)->([Self.Generator.Element],[Self.Generator.Element]){        return (self.filter(fu),self.filter({!fu($0)}))    }} let part2 = [82, 58, 76, 49, 88, 90].anotherPartitionBy{$0 < 60}part2 // ([58, 49], [82, 76, 88, 90])

 

这种解法略好一些,但是他遍历了序列两次。而且为了用单行代码实现,我们删除了闭合函数,这会导致很多重复的内容(过滤函数和数组会在两处被用到)。

能不能只用单个数据流就对原来的序列进行转换,把两个部分分别存入一个元组中呢?答案是是可以的,使用 reduce 方法:

var part3 = [82, 58, 76, 49, 88, 90].reduce( ([],[]), combine: {    (a:([Int],[Int]),n:Int) -> ([Int],[Int]) in    (n<60) ? (a.0+[n],a.1) : (a.0,a.1+[n]) })part3 // ([58, 49], [82, 76, 88, 90])

 

这里我们创建了一个用于保存结果的元组,它包含两个部分。然后依次取出原来序列中的元素,根据过滤结果将它放到第一个或第二个部分中。

我们终于用真正的单行代码解决了这个问题。不过有一点需要注意,我们使用 append 方法来构造两个部分的数组,所以这实际上比前两种实现慢一些。

 

  获取并解析 XML 格式的网络服务

上述的某些语言不需要依赖外部的库,而且默认有不止一种方案可以处理 XML 格式的数据(比如 Scala 自身就可以将 XML 解析成对象,尽管实现方法比较笨拙),但是 (Swift 的)Foundation 库仅提供了 SAX 解析器,叫做 NSXMLParser。你也许已经猜到了:我们不打算使用这个。

在这种情况下,我们可以选择一些开源的库。这些库有的用 C 实现,有的用 Objective-C 实现,还有的是纯 Swift 实现。

这次,我们打算使用纯 Swift 实现的库:AEXML:

let xmlDoc = try? AEXMLDocument(xmlData: NSData(contentsOfURL: NSURL(string:"https://www.ibiblio.org/xml/examples/shakespeare/hen_v.xml")!)!) if let xmlDoc=xmlDoc {    var prologue = xmlDoc.root.children[6]["PROLOGUE"]["SPEECH"]    prologue.children[1].stringValue // Now all the youth of England are on fire,    prologue.children[2].stringValue // And silken dalliance in the wardrobe lies:    prologue.children[3].stringValue // Now thrive the armourers, and honour's thought    prologue.children[4].stringValue // Reigns solely in the breast of every man:    prologue.children[5].stringValue // They sell the pasture now to buy the horse,}

 

 

  找到数组中最小(或最大)的元素

我们有多种方式求出 sequence 中的最大和最小值,其中一种方式是使用 minElement 和maxElement 函数:

//Find the minimum of an array of Ints[10,-22,753,55,137,-1,-279,1034,77].sort().first[10,-22,753,55,137,-1,-279,1034,77].reduce(Int.max, combine: min)[10,-22,753,55,137,-1,-279,1034,77].minElement() //Find the maximum of an array of Ints[10,-22,753,55,137,-1,-279,1034,77].sort().last[10,-22,753,55,137,-1,-279,1034,77].reduce(Int.min, combine: max)[10,-22,753,55,137,-1,-279,1034,77].maxElement()

 

 

  

   并行处理

某些语言支持用简单透明的方式允许对序列的并行处理,比如使用 map 和 flatMap 这样的函数。这使用了底层的线程池,可以加速多个依次执行但又彼此独立的操作。

Swift 还不具备这样的特性,但我们可以用 GCD 实现:

 

  

  埃拉托色尼选筛法

古老而优秀的埃拉托色尼选筛法被用于找到所有小于给定的上限 n 的质数。

首先将所有小于 n 的整数都放入一个序列(sequence)中,这个算法会移除每个数字的倍数,直到剩下的所有数字都是质数。为了加快执行速度,我们其实不必检查每一个数字的倍数,当检查到 n 的平方根时就可以停止。

基于以上定义,最初的实现可能是这样的:

var n = 50var primes = Set(2...n) (2...Int(sqrt(Double(n)))).forEach{primes.subtractInPlace((2*$0).stride(through:n, by:$0))}primes.sort()

 

在外层的区间里,我们遍历每一个需要检查的数字。对于每一个数字,我们使用stride(through:Int by:Int) 函数计算出由它的倍数构成的序列。最初,我们用所有 2 到 n 的整数构造了一个集合(Set),然后从集合中减掉每一个生成的序列中的元素。

不过正如你所见,为了真正的删除掉这些倍数,我们使用了一个外部的可变集合,这会带来副作用。

我们总是应该尝试消除副作用,所以我们先计算所有的子序列,然后调用 flatMap 方法将其中所有的元素展开,存放到单个数组中,最后再从原始的集合中删除这些整数。

var sameprimes = Set(2...n) sameprimes.subtractInPlace((2...Int(sqrt(Double(n))))                           .flatMap{ (2*$0).stride(through:n, by:$0)})sameprimes.sort()

 

 

  使用析构交换元组中的值

既然是福利,自然并非每个人都知道这一点。和其他具有元组类型的语言一样,Swift 的元组可以被用来交换两个变量的值,代码很简洁:

var a=1,b=2 (a,b) = (b,a)a //2b //1

以上就是全部内容,正如我们预料的那样,Swift 和其他语言一样富有表现力。

 

转载地址:http://kucpx.baihongyu.com/

你可能感兴趣的文章
企业shell常见面试题及企业实战案例深入浅出讲解
查看>>
Load Test
查看>>
美文共赏
查看>>
RHEL6入门系列之十七,打包与压缩
查看>>
SQLite 3.7.13的加密解密(二)—— 开放宏定义
查看>>
禁止server 2008域端口的脚本
查看>>
数据结构图之二(最小生成树--普里姆算法)
查看>>
HTML输出 一 控制列背景颜色
查看>>
Redis for Windows(C#缓存)配置文件详解
查看>>
回忆2013年的点点滴滴(各个方面)
查看>>
ASP.NET MVC 4使用PagedList.Mvc分页
查看>>
HDOJ 2066 floyed优化算法
查看>>
window.onscroll
查看>>
开发常用动画收集
查看>>
nginx js、css多个请求合并为一个请求(concat模块)
查看>>
mybatis实战教程(mybatis in action)之五:与spring3集成
查看>>
解决浏览器Adobe Flash Player不是最新版本问题
查看>>
SQLite 约束
查看>>
Python爬虫学习——使用Cookie登录新浪微博
查看>>
linux配置网络
查看>>