title: “Why coding like This —— Reduce 函数揭秘”
date: 2015-08-03 19:24:16
categories: “why coding like this”
tags: [swift进阶]

3.Reduce 函数揭秘

Topic 3:

请用Reduce函数对Int类型数组内所有元素求和,例如数组[1,2,3,4]的4个元素和为10.

Example:

//例一:
let intArray = [1,2,3,4]

var sum = intArray.reduce(0){
  result,x in
  result + x
}

why coding like this?

1.开篇

命题一:假设让你写一个函数来实现对Int类型数组内所有元素求和,例如数组[1,4]的求和结果为10。
思路:与类map,filter函数思路不同,前者是对数组内每一个元素单独做处理,而reduce恰恰相反,是将所有数组元素整合到一起。因此关于求和函数思路很明确,遍历整个数组元素,逐一求和后作为结果值返回,大致为 0 + 1 + 2 + 3 +4。其中0是初始值。
代码:

//例二:
func sum(xs:[Int])->Int{
  var result:Int = 0
  for x in xs{
    result += x
  }
  return result
} //不妨测试下
sum(intArray)//print 10

可以看到对于sum函数传入Int类型数组,首先我们声明一个Int类型变量用于保存最后的求和结果值,初始值为0;接着使用for-in语句遍历传入的数组,使用result += x逐一求和累加;最后返回结果值result。

2.过渡

命题二:基础入门之后,按照惯例对条件进行改变,假如把所有元素求和更改为对所有元素求乘,如[1,4] 返回1 * 2 * 3 * 4 = 24。
思路:显然对于前面的代码稍许改动即可,主要是替换result += x(累加运算) 更为result = result * x(阶乘运算)。当然有一点切记不要遗忘,就是初始result = 1
代码:

//例三:
func multiplicator(xs:[Int])->Int{
    var result:Int = 1
    for x in xs{
        result = x * result
    }
    return result
}
multiplicator(intArray)//输出24

命题三: 说完Int类型数组,再来说说String类型,将一个文字片段数组组成一个完整的句子,例如["hello"," ","world","!","say","by","optionalswift"],最后整合成“Hello World!say by optionalswift”
思路: 思路大概是遍历数组,然后将所有字符串元素拼接在一起,关键这个拼接用什么?例如有str1str2,swift中其实只需要简单的newStr = str1 + str2 即可。
代码:

//例四:
func jointStringArray(xs:[String])->String{
  var result :String = ""

  for x in xs{
    result += x
  }
  return result
}
let helloworld = ["hello"," ","world","!","say","by","optionalswift"]
jointStringArray(helloworld)//输出"Hello World!say by optionalswift" 

你已经急着上手想写泛型函数了吗?等等,我们貌似还忽略了一些复杂的链接情况。比如[“hello”,”world”,”say”,”By”,”optional”],我们需要在每一个单词之间插入”-“连字符,别问我为什么?因为是我出题!

//例五:hyphen 意思为连字符`-`
func jointStringArrayByHyphen(xs:[String])->String{
  var result : String = "整合后的字符串为:"
  for x in xs{
    result = result + "-" + x  //注意右侧是一个类似combine(result,x)函数进行处理 得到整合后的结果给result
  }
  return result
}
let newArray = ["hello","By","optional"]
jointStringArrayByHyphen(newArray)//输出"整合后的字符串为:-hello-world-say-By-optional"

3.高潮

在开始写我们的泛型reduce函数之前,分析函数的输入输出以及如何实现:首先该函数将传入一个泛型类数组,暂且定为[T];此外函数可自己设定初始值给result,暂且定为U;最后是重中之重:传入一个闭包作为连接result和数组元素的处理函数,combine(result,x),不难分析result的类型为T,x是数组xs中的元素,类型为U,返回嘛自然是和result一致的类型喽。因此combine闭包类型为(U,T)->U。因此代码这么写:

//例六:
func myReduce<T,U>(arr:[T],initialValue:U,combine:(U,T)->U)->U{
  var result = initialValue //赋值初始值 类型为U 并且是作为结果值返回的
  for elem in arr{
    result = combine(result,elem)   //注意右侧是传入的闭包 该闭包类型为(U,T)->U,即把上一次的结果值依次和数组元素做拼接处理,该处理可在闭包中实现,取决于你
  }
  return result
}
myReduce(newArray,"整合后的字符串为:"){
  result,elem in //注意result,x 于combine(result,elem)相对应
  return result + "-" + x //注意这里return 其实是可以省略的!
}

不得不说,这个函数还是有点料的,需要细细品味一番。你以为这就结束了吗,还有落幕呢?

4.落幕

对sum函数进行改写,使用前面自定义的myReduce函数封装

func sumUsingReduce(xs:[Int])->Int{
    return myReduce(xs,0){result,x in result + x}
}

注意到省略了return 因为swift会帮你推算要返回什么。简化的感觉不够彻底。

func sumUsingReduce(xs:[Int])->Int{
    return myReduce(xs,0,+)
}

闭包仅仅传入了一个+号,swift推算过程是首先combine闭包有两个传入参数result和elem,除此之外别无其他,因此+只能对这两个参数求和,得到一个结果值x,由于combine函数还需要返回一个结果值,但是思来想去貌似除了x没有其他可用,因此把x作为闭包结果值返回和result相加。

类似的你可是使用return myReduce(xs,1,*)。

你以为这就结束了吗? 现在用reduce来改写map函数 以及filter函数

func mapUsingReduce<T,U>(xs:[T],f:T->U)->[U]{
    return xs.reduce([]){result,x in result + [f(x)]}//使用了系统API 尝试用自定义的
}

首先注意到函数传入的两个参数以及返回结果值和早前map函数是一模一样的,关键是在主体的实现上!设定初始值为[]等价于var result = [],而闭包的内容result,x in result + [f(x)] 又等价于遍历数组时执行的语句result.apped(f(x))。请仔细思考下!小技巧 reduce的闭包内容就是等式result = combine(result,x) 的combine函数,意味着每次调用闭包返回一个处理过后的值给result即可,而result又是函数内私有变量,只有全部调用后才返回。

再来看filter的实现:

filterUsingReduce<T>(xs:[T],check:T->Bool)->[T]{
    return xs.reduce([]){
        return check(x) ? result + [x] : result //使用了系统API 尝试用自定义的
    }
}

map,filter,reduce 小节至此结束! 希望对大家有收获!

Why Coding Like this -------Reduce揭秘的更多相关文章

  1. html5使用canvas实现弹幕功能示例

    这篇文章主要介绍了html5使用canvas实现弹幕功能示例的相关资料,需要的朋友可以参考下

  2. HTML5 WebSocket实现点对点聊天的示例代码

    这篇文章主要介绍了HTML5 WebSocket实现点对点聊天的示例代码的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  3. 前端实现弹幕效果的方法总结(包含css3和canvas的实现方式)

    这篇文章主要介绍了前端实现弹幕效果的方法总结(包含css3和canvas的实现方式)的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  4. H5 canvas实现贪吃蛇小游戏

    本篇文章主要介绍了H5 canvas实现贪吃蛇小游戏,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  5. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

  6. ios – parse.com用于键,预期字符串的无效类型,但是得到了数组

    我尝试将我的数据保存到parse.com.我已经预先在parse.com上创建了一个名为’SomeClass’的类.它有一个名为’mySpecialColumn’的列,其数据类型为String.这是我尝试使用以下代码保存数据的代码:如果我运行这个我得到:错误:密钥mySpecialColumn的无效类型,预期字符串,但得到数组这就是我在parse.com上的核心外观:有谁知道我为什么会收到这个错误?

  7. ios – 在Swift中将输入字段字符串转换为Int

    所以我非常擅长制作APP广告Swift,我试图在文本字段中做一些非常简单的输入,取值,然后将它们用作Int进行某些计算.但是’vardistance’有些东西不正确它是导致错误的最后一行代码.它说致命错误:无法解开Optional.None解决方法在你的例子中,距离是一个Int?否则称为可选的Int..toInt()返回Int?因为从String到Int的转换可能失败.请参阅以下示例:

  8. 如何在iOS中检测文本(字符串)语言?

    例如,给定以下字符串:我想检测每个声明的字符串中使用的语言.让我们假设已实现函数的签名是:如果没有检测到语言,则返回可选字符串.因此,适当的结果将是:有一个简单的方法来实现它吗?

  9. ios – 上下文类型’NSFastEnumeration’不能与数组文字一起使用

    斯威夫特3,你会这样做吗?解决方法正如您所发现的,您不能使用as-casting将数组文字的类型指定为NSFastEnumeration.您需要找到一个符合NSFastEnumeration的正确类,在您的情况下它是NSArray.通常写这样的东西:

  10. xamarin – 崩溃在AccountStore.Create().保存(e.Account,“);

    在Xamarin.Forms示例TodoAwsAuth中https://developer.xamarin.com/guides/xamarin-forms/web-services/authentication/oauth/成功登录后,在aOnAuthenticationCompleted事件中,应用程序在尝试保存到Xamarin.Auth时崩溃错误说不能对钥匙串说期待着寻求帮助.解决方法看看你

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部