Why Coding like This ? —— Filter 函数揭秘

2.Filter函数揭秘

Topic 2:

请用Filter函数筛选出String数组中后缀是.swift的文件。

Example:

//例一:
var filesNeedToFilter = ["ViewController.swift","HelloWorld.c","Web.java","Person.swift","Main.c"]

//输出["ViewController.swift","Person.swift"]
filesNeedToFilter.filter{
  file in
  file.hasSuffix(".swift")//注意suffix是后缀的意思
}

why coding like this?

命题:假设让你写一个函数来实现筛选后缀是.swift的文件,你该如何实现?
思路:首先声明一个数组用于保存符合的结果值,然后遍历需要筛选的数组,通过hasSufffix函数返回true or false来判断数组元素是否含有指定的后缀。
代码:

//例二:
func getSwiftFiles(files:[String])->[String]{
  var result:[String] = []
  for file in files{
    if file.hasSuffix(".swift"){
      result.append(file)
    }
  }
  return result
}
//不妨来试试//输出["ViewController.swift","Person.swift"]
getSwiftFiles(filesNeedToFilter)

我们注意到只有满足给定的筛选条件,这里是file.hasSuffix(".swift")返回true表示包含后缀,反之不包含。
通过Map章节学习后,立马意识到该函数的局限性—————仅适用筛选后缀是.swift的文件。因此我们需要自己指定筛选条件,而这个筛选条件是一个类型为String->Bool的闭包,String表示传入需要条件判断的元素,满足返回true,不满足返回false,这也是为什么闭包返回Bool的原因了。修改后的代码如下:

//例三:
func getFilesByCondition(files:[String],f:String->Bool)->[String]{
  var result:[String] = []
  for file in files{
    if f(file){
      result.append(file)
    }
  }
  return result
}
//不妨来试试 采用closure的尾包形式
getFilesByCondition(filesNeedToFilter){
  file in
  file.hasSuffix(".c")//返回["HelloWorld.c", "Main.c"]
}

现在可以随心所欲的传入筛选条件了,但是该函数显然仍有不足,缺少什么呢?恩....对!泛型,现在传入的数组不能局
限于String类型,那么作为泛型我们就要使用T来泛指所有类型喽(注意泛型并没有特别指明要用T,U,你可以自由给定,建议使用大写以及驼峰写法)。改写代码如下:

//例四:
func myFilter<T>(xs:[T],check:T->Bool)->[T]{
  var result:[T] = []
  for x in xs{
    //必须经过条件判断才可以
    if check(x){
      result.append(x)
    }
  }
  return result
}

看来又实现了一个自定义的方法,不妨试试好用不好用。

why coding like this 系列,更多是用Why,How看待代码,摒弃那种我只要会写,至于为什么?因为我一直那么写的思想。希望对大家有帮助! 下文带来reduce的揭秘。

推荐阅读更多精彩内容