[Learning Scala] Functions

object SparkTest004 {
  def main(args: Array[String]): Unit = {
    /**
     * Chapter 5 functions
     */

    def jump(title: String = ""): Unit = print(s"\n ---------${title}----------- \n")

    jump("Function Types and Values")
    def doubleNumbers (x: Int): Int = x * 2

    // here the value "myDoubleNumbers is a type of function. Its implementation is "function: doubleNumbers"
    val myDoubleNumbers: (Int) => Int = doubleNumbers // Function types with a single parameter can leave off the parentheses.
    val copyMyDoubleNumbers = myDoubleNumbers  // function copy
    print(s"double of 5 is: ${myDoubleNumbers(5)}, copyMyDoubleNumbers as new function: ${copyMyDoubleNumbers(5)}\n")

    def max(x: Int, y: Int, z: Int): Int = {
      def max(x: Int, y: Int): Int = if (x < y) y else x
      max(x, max(y, z))
    }

    val maximize: (Int, Int, Int) => Int = max
    val maximize2 = max _
    print(s"maximum number of three is: ${maximize2(1,5,3)}")

    jump("Assigning a Function with the Wildcard Operator")
    val myDoubleNumbersWithNoFunctionTypeDeclare = doubleNumbers _ // comparing with the above declare method
    print(s"double of 5 is: ${myDoubleNumbersWithNoFunctionTypeDeclare(5)}")

    jump("Higher-Order Functions")
    def safeStringOp(s: String, f: String => String): String = { // using function to be a parameter
      if (s != null) f(s) else s
    }

    def trimString(s: String): String = s.trim()

    print(s"test1: ${safeStringOp("hello world        ", trimString)}, length is: ${safeStringOp("hello world        ", trimString).length}\n")
    print(s"test2: ${safeStringOp("hello world        !", trimString)}, length is: ${safeStringOp("hello world        !", trimString).length}\n")
    print(s"test3: ${safeStringOp("", trimString)}\n")

    jump("Function Literals")
    // method 1
    def greeter(name: String): String = s"hello ${name}\n"
    val greeter1 = greeter _
    // method 2 function literals
    val greeter2 = (name: String) => s"hello ${name}\n" // function literals
    print(greeter2("gavin"))

    val maximize3 = (x: Int, y: Int) => if (x > y) x else y // compare to maximize and maximize2
    print(s"function literals maximum: ${maximize3(23, 97)}\n")

    val moveItOn = () => "=" * 50 + "getting started" + "=" * 50
    print(s"invocation the function: ${moveItOn()}\n")
    print(s"the function type is: ${moveItOn}\n")

    // instead of using a function value, we can also use function literals as a parameter
    print(s"test4: ${safeStringOp("hello world        ", (s: String) => s.trim())}, " +
      s"length is: ${safeStringOp("hello world        ", (s: String) => s.trim()).length}\n")

    print(s"test5: ${safeStringOp("hello world    ", s => s.trim())}") // elegant

    jump("Placeholder Syntax")
    print(s"test6: ${safeStringOp("hello world    ", _.trim())}\n") // placeholder

    def combination(x: Int, y: Int, f: (Int, Int) => Int) = f(x, y)
    print(s"the combination result is: ${combination(3, 5, _ - _)}\n")
    print(s"the combination result is: ${combination(5, 3, _ - _)}\n")

    def triple(x: Int, y: Int, z: Int, f: (Int, Int, Int) => Int) = f(x, y, z)
    print(s"triple test 1: ${triple(1, 2, 3, _ + _ + _)}\n")
    def triple2[A, B](x: A, y: A, z: A, f: (A, A, A) => B) = f(x, y, z)
    print(s"triple test 2: ${triple2[Int, Double](1, 2, 3, _ + _ + _)}\n")

    jump("Partially Applied Functions and Currying")
    def factorOf(x: Int, y: Int) = y % x == 0
    val f = factorOf _
    val factorOf3 = f(3, _: Int) // the type "Int" must be specified
    print(factorOf3(9)) // partially apply function

    //curring the function: breaking up the parameters into two list
    def factorof2(x: Int)(y: Int) = y % x == 0
    def isEven = factorof2(2) _
    print(s"\nIs 4 even: ${isEven(4)}")

    jump("By-Name Parameters")
    def doubles(x: Int) = {
      println("invoking the function, value is: " + x)
      x * 2
    }

    def greeter3(x: Int) = {println("getting into greeter3"); x}
    doubles(5)
    doubles(greeter3(5)) // didn't get into the greeter3 twice! the book says it will get into it twice ~ ~

    jump("Partial Functions")
    // Such functions are called partial functions because they can only partially apply to their input data.
    val statusHandler: Int => String = {
      case 100 => "Okay"
      case 400 => "error 400"
      case 500 => "error 500"
      case _ => "can't handle this case"
    }
    println(statusHandler(9))
    println(statusHandler(100))
    println(statusHandler(400))
    println(statusHandler(500))

    jump("Invoking Higher-Order Functions with Function Literal Blocks")
    val uuid = java.util.UUID.randomUUID.toString
    val timedUUID = safeStringOp(uuid, { s =>
      val now = System.currentTimeMillis
      val timed = s.take(24) + now
      timed.toUpperCase()
    })
    println(uuid)
    println(timedUUID)

  }
}
相關文章
相關標籤/搜索