Fork me on GitHub
代码改变世界

F#注解

2016-02-25 12:17  沉睡的木木夕  阅读(608)  评论(0编辑  收藏  举报

  不要问我为啥要学F#——因为气质摆在那里

  标注:以下内容均来自 anderslly F#系列

  1、类型推演

    let square x = x * x

    //接受一个某类型参数的quare函数返回一个这个参数的平方,因为支持参数*参数的类型有很多;比如int,byte,uint64,double等,而F#会默认为int类型

    //这就类似与js当中var 不同的是,在js当中,函数返回一个数值需要用return关键字 而F#不需要

  2、规定参数的类型(类似于C#的形参 也叫类型注解)

    let concat (x : string) y = x + y;;

    //一个函数接受一个x的参数,这个参数必须为string类型(这里的冒号":"的作用就是起到了约束作用)以及一个参数y

    //这里有同学以为y没有约束类型,那应该是什么类型都可以

    //实际上是由于因为x类型是string,返回的值又是x + y 用了"+"操作,所以类型必须相同,则y也是string(用vs的同学可以按 alt+enter检查)

  3、输出打印

    printfn "N^2 = %A" squares

    //打印信息 这里的%A是匹配任何类型的参数(也就是空格后面的squares) 注:参数的输入 必须要已空格隔开,否则会报错

    这里有几个输出的类型要注明:%d,%f,%s分别是int、float、string的占位符

  4、值,变量的不可变性

    跟C#不同,F#中的变量都是不可变性的,换句话说 就是固定不变的,自然同时就具备了类型安全这个优良的特点,如果你一定要改变这个值,也是可以的!

    用mutable改变这个值(注意:这个改动值,不是寻常我们C#,JS中的改变变量的值,而是重新创建了一个新的变量,这一点上是不是跟我们C#中的string有点类似呢 (^_^))

    然后用左箭头操作符 <- 修改变量的值

    let mutable x = "the original value.";;

    printfn "x's value is '%s'" x;;    //x's value is 'the original value'.

    x <- "the new one.";;

    printfn "x's value is now '%s'" x;;  //x's value is now 'x's value is the original value'

  5、引用值

    引用值是另一种表示可修改数据的方式。但它不是将变量存储在堆栈(stack),引用值其实是一个指向存储在堆(heap)上的变量的指针(pointer)。在F#中使用可修改的值时会有些限制(比如不可以在内部lambda表达式中使用)。而ref对象则可被安全地传递,因为它们是不可变的record值(只是它有一个可修改的字段)。原话引自——【F# 20分钟快速上手(二)

    使用引用值时,用“:=”赋一个新值,使用“!”进行解引用。

    let refCell = ref 42;;     //refCell = 42

    refCell := -1;;  //改变引用值

    !refCell;;  //解析引用值 显示-1

  6、模块(Modules)

    这个就跟C#的using引用程序集是一个道理了 直接看例子  

    module ProgramSettings =
      let version = "1.0.0.0"    
      let debugMode = ref false

    module MyProgram =
          do printfn "Version %s" ProgramSettings.version  
          open ProgramSettings
          debugMode := true 

  7、元组(Tuple)

    表示值的有序集合 也是跟C#中的元组有近似相同;用来传递多个或一组值

    定义一个元组,只要将一组值用逗号分割

    let tuple = (1, false, "text");;

    let getNumberInfo (x : int) = (x, x.ToString(), x * x);;  //定义个接受int的变量x 函数返回三个变量(int,string,int)

    函数可以接受元素为参数,自然C#也是可以的

    let printBlogInfo (owner, title, url) = printfn "%s's blog [%s] is online at '%s'" owner title url;;

    let myBlog = ("Chris", "Completely Unique View", "http://blogs.msdn.com/chrsmith");;

    printBlogInfo myBlog;;  //输出 Chris's blog [Completely Unique View] is online at 'http://blogs.msdn.com/chrsmith'

  8、函数科里化(Function Currying)

    F#中可以只接受某函数的参数 进而将参数传递给另一个新的函数

    let addThree x y z = x + y + z;;  //函数addThree接受三个参数  1

    let addTwo x y = addThree 10 x y;;  //函数addTwo接受两个参数  2

    addTwo 1 1;;  //调用addTwo传1,1两个参数—>由于函数2返回一个新的函数addThree 并接受一个参数10,和前一个函数的两个参数的值

  9、Union类型 

    没有自己的一些见解,直接点击原文

  10、模式匹配

    看起来想swicth case模式

    let listLength aList =
          match aList with
          | [] -> 0
          | a :: [] -> 1
          | a :: b :: [] -> 2
          | a :: b :: c :: [] -> 3
          | _ -> failwith "List is too big!"   //竖线 | 表示各种case ->是lambda表达 表示输出  a::[] 意思是说 a 在 [] 之前的元素(这是空的list集合)

    let isOdd x =
          match x with
          | _ when x % 2 = 0 -> false
          | _ when x % 2 = 1 -> true   //下划线 _ 是匹配任意值

    也可以动态匹配

    let getType (x : obj) =
          match x with
          | :? string -> "x is a string"
          | :? int -> "x is a int"
          | :? System.Exception -> "x is an exception"
          | :? _ -> "invalid type"     // 这里博主没有将:? 符号的意思说明 我就理解为表面意思 :匹配这个值 ?是可以是任何类型的 跟输入的类型是一样的 比如输入"a" 就是string;11 就是int

  11、Forward Pipe operator(|>)

    简单的理解就是一个函数A接受一个参数并返回一个值,然后把这个值又传递给另一个函数B,函数B得到这个参数并返回一个值(类型可能不同),进入接着往下传递

    用C#的写法就是

    Int32 result = Func3(Func2(Fun1(A)))......

    我们来看看F#的写法

    let square x         = x * x
    let toStr (x : int)  = x.ToString()
    let rev   (x : string) = new String(Array.rev (x.ToCharArray()))

    let result = rev (toStr (square 32))  //调用

    上面看起来就跟C#的一样直白,好理解,那我们接着用操作符 |> 写一遍

    let result = 32 |> square |> toStr |> rev

    可以看出这样极大的简化了我们代码,并且更说明了Forward Pipe Operator的含义

  12、集合(Collection:Seq,List,Array)

    F#表示集合的有三种,Seq;List; Array

    这里每个集合类型都有很多方法,大家可以用VS的智能提示了解(吐槽一下:不知道我的vs是怎么滴,一定得手动按ctrl+j才行 不能想C#中实时出现提示的)这里我记录几个常用的

    iter。“iter”函数遍历集合的每一项。 跟foreach遍历一样

    List.iter (fun i -> printfn "Has element %d" i) [1 .. 10]  从集合[1..10](这个是集合的元素从1到10)循环遍历输出Has element 1 2 3...

    map map函数基于一个指定的函数对集合的值进行转换

    Array.map (fun (i : int) -> i.ToString()) [| 1 .. 10 |]  //([|1..10 |] 前后加竖线是说明此几何为整数数组)    次数是由原来的整数集合变为一个字符串集合

    fold。“fold”函数接受一个集合,并将集合的值折叠为单个的值。像iter和map一样,它接受一个函数,将其应用于集合的每个元素,但它还接受另一个“accumulator”参数。fold函数基于上一次运算不断地累积  accumulator参数的值  注:是不是跟C#的ForEach(Func)很想呢

    Seq.fold (fun acc i -> i + acc) 10 { 1 .. 10 }

  13、可选值(Option Values)

    F#中的“可选类型(option type)”有两种状态:“Some”和“None”。在下面的记录类型Person中,中间的字段可能有值,也可能没有值

    type Person = { First : string; MI : string option; Last : string }
    let billg    = {First = "Bill";  MI = Some("H"); Last = "Gates" }
    let chrsmith = {First = "Chris"; MI = None;      Last = "Smith" } 

  14、延迟求值 (LazyValue)

    延迟初始化表示一些值,它们在需要时才进行计算。F#拥有延迟求值特性。看下面的例子,“x”是一个整数,当对其进行求值时会打印“Computed”。

    let x = lazy (printfn "Computed."; 42);;  //初始化值42 但是被延迟了

    let listOfX = [x; x; x];;    

    x.Force();;  //直到使用Force 才会初始化x的值