Scala学习-变量常量运算符流程控制和函数

Posted youngchaolin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Scala学习-变量常量运算符流程控制和函数相关的知识,希望对你有一定的参考价值。

scala是马丁.奥德斯克设计的,专门为程序员设计,广泛应用于大数据的语言。它同时支持面向对象和面向函数编程,运行scala需基于JVM,使用它需要提前安装好JDK和scala SDK。scala的的代码一行可以顶多行java代码,开发效率高,并且兼容java类库,scala编译完也是.class文件。另外大数据框架kafka和spark是基于scala开发的,因此想做流处理需要学习scala。

基本数据类型

scala中没有基本类型的说法,绝大多数类型(类似java类型)都封装成了类,以下是它的简图。

(1)Any是scala顶级父类。

(2)AnyVal是所有数值类型的父类,注意有一个Unit,它只有一个实例()。

(3)AnyRef是所有对象类型的父类,注意Null是它的子类,它的实例对象是null,可以赋值给任意对象类型。

(3)Nothing可以是任何类型的子类,可以表示不正常的返回值类型,如异常。

# RPEL交互方式
scala> def test():Nothing={
     | throw new IllegalArgumentException()
     | }
test: ()Nothing

技术图片

变量和常量

变量声明的语法:‘var 变量名:数据类型=数据值‘ 或者 ‘var 变量名=数据值‘,后者会进行自动类型转换。常量声明类似,使用val代表常量。

使用变量或常量需要标识符的支撑,scala中也有一套标识符规则。

标识符规则

(1)使用数字、字母、下划线和$符号;

scala> var name:String="clyang"
name: String = clyang

scala> var age:Int=28
age: Int = 28

(2)不能以数字开头;

(3)如果有特殊字符,标识符全部都需要是特殊字符;

scala> var +-*/ :Double=3.14
+-*/: Double = 3.14

(4)可以使用``来标记任何字符;

# 可以用scala的关键字,任何字符均可
scala> var `private`:String="test"
private: String = test

(5)见名知意;

# year代表年,365天
scala> val year:Int=365
year: Int = 365

# 如果是常量,声明后不可以修改
scala> year=3650
<console>:13: error: reassignment to val
       year=3650
           ^

类型转换

(1)自动类型推导

# 自动识别类型
scala> var name="clyang"
name: String = clyang

scala> var age=28
age: Int = 28

(2)手动指定向上类型,否则按自动类型推导出数据类型

# 默认是Double类型,手动指定Any类型
scala> var a:Any=3.14
a: Any = 3.14

(3)强制转换,从大往小转

# 强制Double转Int
scala> var d:Double=3.14
d: Double = 3.14

scala> var i:Int=d.toInt
i: Int = 3

(4)自定义类型转换,需自定义函数,可以显示调用和隐式调用。

显示调用

# 自定义类型转换函数
scala> def string2Int(str:String):Int={
     | return Integer.parseInt(str)
     | }
string2Int: (str: String)Int
# 给一个字符串
scala> var str="268"
str: String = 268
# 调用转换
scala> var number=string2Int(str)
number: Int = 268

隐式调用

# 需使用implicit关键字
scala> implicit def string2Int(str:String):Int={
     | return Integer.parseInt(str)
     | }
warning: there was one feature warning; re-run with -feature for details
string2Int: (str: String)Int

scala> str
res1: String = 268

scala> var number:Int=str
number: Int = 268

(5)向上造型创建的对象,可以使用asInstanceOf来向下转。

# 定义Person类
scala> class Person{}
defined class Person

# 创建Any类型的对象,向上造型
scala> var p:Any=new Person()
p: Any = Person@6a370f4

# 向下转
scala> var s:Person=p.asInstanceOf[Person]
s: Person = Person@6a370f4

运算符

scala中运算符包括算术、赋值、关系、逻辑和位运算,注意没有三元运算,并且所有的运算符都封装成了方法。

参考菜鸟教程:https://www.runoob.com/scala/scala-operators.html

算术

注意一下,scala中的运算符是封装成了方法,因此a+b,其实就是a.+(b)的简写,其他依次类推。

scala> var a=500
a: Int = 500

scala> var b=20
b: Int = 20

# 加
scala> a+b
res2: Int = 520

# 完整写法
scala> a.+(b)
res3: Int = 520

# 以下三种方式等效,都可以加1
scala> a=a+1
a: Int = 501

scala> a+=1

scala> a
res5: Int = 502

scala> a.+=(1)

scala> a
res7: Int = 503

如果有点,则计算点。

//运算符运用,有点先算点
println(3+5*7)//38
println(3+(5)*7)//38
println(7*3.+(5)) //56
println(7*3+5) //26
println(7*(3).+(5)) //56
println(3.+(5).*(7))//56

赋值

scala> var a=520
a: Int = 520

scala> var b=1314
b: Int = 1314

# 赋值运算的返回值类型为Unit,对应实例对象为()
scala> var z:Unit=a=b
z: Unit = ()

# a被赋值为1314
scala> a
res8: Int = 1314

关系

就是类似java中的大于小于等于之类的。

scala> var r=1314>520
r: Boolean = true
scala> var r="婚姻"=="车子房子票子"
r: Boolean = false

逻辑

逻辑与、逻辑或和逻辑非。

scala> var r=true&&true
r: Boolean = true

scala> var r=true&&false
r: Boolean = false

位运算

类似java中的位运算符,按位与、按位或、按位异或、取反、左移、右移、无符号右移。

//判断一个数是否是2的幂
println(8&7)//0
//左移比直接相乘快
println(4<<2)//16
println(~4)//-5 取反减1

流程控制

scala中没有switch-case,有条件判断和循环。

if条件判断

类似java,scala中if-else是有返回值结果的,根据if-else的{}最后一行来确认。

package day01.clyang.control

import scala.io.StdIn

/**
  * 流程控制
  */
object IfDemo2 {

  def main(args: Array[String]): Unit = {
    //if-else是有返回值结果的,根据if-else的{}最后一行来确认
    var i=StdIn.readInt()
    var result:String=if(i%2!=0){
      println("~~~")
      "我是谁我在哪里"
      "奇数"//最后一行
    }else{
      "偶数"//最后一行
    }
    //打印result
    println(result)
  }

}

控制台输入奇数后,只返回最后一行‘奇数‘。

99
~~~
奇数

scala中也支持if-else if。

package day01.clyang.control

import scala.io.StdIn

/**
  * 多级if-else
  */
object IfDemo3 {
  def main(args: Array[String]): Unit = {
    var score=StdIn.readInt()
    var result:String=if(score>90) "优秀"
    else if(score>80) "良好"
    else if(score>=70) "一般"
    else "及格"
    //打印结果
    println(result)
  }
}

控制台输入测试,ok。

98
优秀

循环

可以使用while循环,和for循环。

(1)while循环

# while循环,返回值类型为Unit
scala> var count=0
count: Int = 0

scala> var result=while(count<3){
     | println("while")
     | count=count+1
     | }
while
while
while
result: Unit = ()

(2)do-while循环,这个用的少

# do-while返回值类型也是Unit
scala> var count=0
count: Int = 0

# scala中,也类似python中,字符串可以和数字相乘
scala> var result=do{
     | println("*"*5)
     | count=count+1
     | }while(count<3)
*****
*****
*****
result: Unit = ()

(3)for循环

注意to和until的使用,两者均是函数。

# 准备一个数组
scala> var arr:Array[Int]=Array[Int](1,2,3,4,5)
arr: Array[Int] = Array(1, 2, 3, 4, 5)
# to为后包括
scala> var result=for(i <- 0 to arr.length-1){
     | println(arr(i))
     | }
1
2
3
4
5
result: Unit = ()
# until为后不包括
scala> var result=for(i <- 0 until arr.length){
     | println(arr(i))
     | }
1
2
3
4
5
result: Unit = ()

to和until都是函数,如果要指定步长,就不能省略.和( )。

# 指定步长,打印0到10范围的偶数
scala> for(i <- 0.to(10,2)){
     | println(i)
     | }
0
2
4
6
8
10
# 倒着来
scala> for(i <- 10.to(0,-2)){
     | println(i)
     | }
10
8
6
4
2
0
# reverse倒着来
scala> for(i <- 0 to 3 reverse){
     | println(i)
     | }
warning: there was one feature warning; re-run with -feature for details
3
2
1
0

for循环最后一行的结果,可以通过yield关键字把它放到一个集合,集合类型是Vector。

scala> import scala.io.StdIn
import scala.io.StdIn
# 输入5
scala> var n=StdIn.readInt()
n: Int = 5
# 返回集合类型为Vector
scala> var result=for(i <- 0 to n) yield i*i
result: scala.collection.immutable.IndexedSeq[Int] = Vector(0, 1, 4, 9, 16, 25)

for循环中,如果嵌套了if判断,就构成了守卫循环。

# 打印100以内5的倍数,但是不是7的倍数
scala> for(i <- 0.to(100,5)){
     | if(i%7!=0){
     | println(i)
     | }
     | }
5
10
15
20
25
30
40
45
50
55
60
65
75
80
85
90
95
100
# 可以使用守卫循环
scala> for(i <- 0.to(100,5) if(i%7!=0)){
     | println(i)
     | }
5
10
15
20
25
30
40
45
50
55
60
65
75
80
85
90
95
100

for循环嵌套,scala中还有大括号写法。

# 普通写法
scala> for(i <- 1 to 9){
     |   for(j <- 1 to i){
     |   print(i+"*"+j+"="+i*j+"	")
     |   }
     | println()
     | }
1*1=1
2*1=2   2*2=4
3*1=3   3*2=6   3*3=9
4*1=4   4*2=8   4*3=12  4*4=16
5*1=5   5*2=10  5*3=15  5*4=20  5*5=25
6*1=6   6*2=12  6*3=18  6*4=24  6*5=30  6*6=36
7*1=7   7*2=14  7*3=21  7*4=28  7*5=35  7*6=42  7*7=49
8*1=8   8*2=16  8*3=24  8*4=32  8*5=40  8*6=48  8*7=56  8*8=64
9*1=9   9*2=18  9*3=27  9*4=36  9*5=45  9*6=54  9*7=63  9*8=72  9*9=81
# 两个循环嵌套到一起,scala中不推荐使用分号
scala> for(i <- 1 to 9; j <- 1 to i){
     | print(i+"*"+j+"="+i*j+"	")
     | if(i==j) println()
     | }
1*1=1
2*1=2   2*2=4
3*1=3   3*2=6   3*3=9
4*1=4   4*2=8   4*3=12  4*4=16
5*1=5   5*2=10  5*3=15  5*4=20  5*5=25
6*1=6   6*2=12  6*3=18  6*4=24  6*5=30  6*6=36
7*1=7   7*2=14  7*3=21  7*4=28  7*5=35  7*6=42  7*7=49
8*1=8   8*2=16  8*3=24  8*4=32  8*5=40  8*6=48  8*7=56  8*8=64
9*1=9   9*2=18  9*3=27  9*4=36  9*5=45  9*6=54  9*7=63  9*8=72  9*9=81
# 大括号写法
scala> for{
     | i <- 1 to 9
     | j <- 1 to i
     | }
     | {
     | print(i+"*"+j+"="+i*j+"	")
     | if(i==j){
     |   println()
     |   }
     | }
1*1=1
2*1=2   2*2=4
3*1=3   3*2=6   3*3=9
4*1=4   4*2=8   4*3=12  4*4=16
5*1=5   5*2=10  5*3=15  5*4=20  5*5=25
6*1=6   6*2=12  6*3=18  6*4=24  6*5=30  6*6=36
7*1=7   7*2=14  7*3=21  7*4=28  7*5=35  7*6=42  7*7=49
8*1=8   8*2=16  8*3=24  8*4=32  8*5=40  8*6=48  8*7=56  8*8=64
9*1=9   9*2=18  9*3=27  9*4=36  9*5=45  9*6=54  9*7=63  9*8=72  9*9=81

Break

scala中,没有break和continue关键字,如果要退出循环,可以使用Break类来跳出循环。

(1)跳出循环,报异常,并且for循环后面语句不执行。

for(i <- 0.to(50,5)){
      if(i==35) Breaks.break()
      println("我今年"+i+"岁,我依然很帅")
    }
//测试是否执行后面语句
println("finished~~")

# console显示没有打印
我今年0岁,我依然很帅
我今年5岁,我依然很帅
我今年10岁,我依然很帅
我今年15岁,我依然很帅
我今年20岁,我依然很帅
我今年25岁,我依然很帅
我今年30岁,我依然很帅
Exception in thread "main" scala.util.control.BreakControl

如果按以下这么写,即使跳出循环,依然执行for循环后面的语句。

Breaks.breakable(
  for(i <- 0.to(50,5)){
    if(i==35) Breaks.break() 
    println("我今年"+i+"岁,我依然很帅")
  }
)
//测试是否执行后面语句
println("finished~~")

# console显示打印了后面的语句
我今年0岁,我依然很帅
我今年5岁,我依然很帅
我今年10岁,我依然很帅
我今年15岁,我依然很帅
我今年20岁,我依然很帅
我今年25岁,我依然很帅
我今年30岁,我依然很帅
finished~~

# import scala.util.control.Breaks._后,这么写也行
breakable(
  for(i <- 0.to(50,5)){
    if(i==35) break()
    println("我今年"+i+"岁,我依然很帅")
  }
)

println("finished~~")

函数

函数在scala中地位非常高,是"一等公民",它可以作为函数的参数,也可以作为函数的返回值,并且可以定义在任何地方,函数定义的基本语法如下。

  /*
   * 函数的格式
   * def 函数名(参数列表):返回值类型={
   *   函数体
   *   return 返回值
   *   }
   */

入门使用

常规写法。

scala> def max(i:Int,j:Int):Int={
     | return if(i>j) i else j
     | }
max: (i: Int, j: Int)Int
# 调用
scala> max(1,3)
res0: Int = 3

如果没有return,默认将函数体最后一行的结果返回。

scala> def sum(i:Int,j:Int):Int={i+j}
sum: (i: Int, j: Int)Int

scala> sum(1,3)
res1: Int = 4

如果函数体只有一句,大括号可以省略,如果可以根据最后一行推导出返回值类型,方法返回值类型也可以省略。

scala> def sum(i:Int,j:Int)=i+j
sum: (i: Int, j: Int)Int

scala> sum(1,3)
res2: Int = 4

如果没有指定方法的返回值类型,方法体不能用return,根据函数体最后一行来推导。

scala> def sum(i:Int,j:Int)={return i+j}
<console>:10: error: method sum has return statement; needs result type
       def sum(i:Int,j:Int)={return i+j}
                             ^

方法如果没有返回值类型,可以使用Unit作为返回值类型。

scala> def printShape(row:Int,col:Int):Unit={
     | for(i <- 1 to row)
     |   println("*"*col)
     | }
printShape: (row: Int, col: Int)Unit

scala> printShape(3,3)
***
***
***

如果没有写返回值类型,也没写等号,则默认返回值类型为Unit。

scala> def printTest(){
     | println("this is return Unit")
     | }
printTest: ()Unit
# 如果函数没有参数,可以省略括号
scala> printTest
this is return Unit

函数进阶

scala可以直接调用java api。

scala> def getRandom()=(Math.random()*11).toInt
getRandom: ()Int
# 调用时加括号
scala> getRandom()
res7: Int = 0
# 没有参数,调用时括号可以省略
scala> getRandom
res8: Int = 10
# 方法没有参数,方法括号也可以省略
scala> def getRandom=(Math.random()*11).toInt
getRandom: Int
scala> getRandom
res9: Int = 3

scala中函数可以重载,类似java,根据方法签名来绑定执行的方法。

//scala中函数可以重载
def square(x:Int,y:Int)=x*y
def square(r:Int)=3.14*r*r

//根据参数不同,重载,好像一次只能调用一个,同时调用两个也只显示一个的结果

函数定义时如果没有(),调用时就不能加()。

scala> def test="this is test"
test: String
# 正常调用
scala> test
res15: String = this is test
# 加括号调用就报错
scala> test()
<console>:12: error: not enough arguments for method apply: (index: Int)Char in class StringOps.
Unspecified value parameter index.
       test()
           ^

高阶函数

将函数作为参数,或者函数的返回值也是函数,这样的函数叫做高阶函数,前面的闭包和柯理化就是高阶函数,返回值类型是函数。

(1)闭包

函数中嵌套函数,可以延长变量的生命周期,这是函数的闭包。

# 定义函数,打印最小值,如果有必要可以获取两个较大值的和
# 返回值为函数
scala> def min(x:Int,y:Int,z:Int)={
     | var min=x
     | if(y<min) min=y
     | if(z<min) min=z
     | println("最小值为:"+min)
     |   def sum(){ # 内部定义函数,返回值类型为Unit
     |   println(x+y+z-min) 
     |   }
     | sum _ # 将sum一整个函数作为返回值,调用它才执行sum方法,不调用就不执行
     | }
min: (x: Int, y: Int, z: Int)() => Unit
# 打印最小值,返回值类型是一个函数,() => Unit就是min方法返回值类型
scala> min(1,2,3)
最小值为:1
res17: () => Unit = <function0>
# 调用sum方法
scala> min(1,2,3)()
最小值为:1
5

min函数执行完后,会移除出栈内存,如果将参数也移除,那接下来如果想调用sum方法就没有了参数,在scala中,只会将min函数移除出栈内存,而参数x,y,z会保留,供sum函数使用,这样延长了x,y,z变量的生命周期,这样使用函数嵌套延长变量生命周期的方式叫做函数的闭包。

闭包的应用练习如下,求两个数的积,如果有必要添加第三个数来求积。

scala> def product(x:Int,y:Int):Int => Int={
     | println("两个数的积为:"+x*y)
     |  def add(z:Int):Int={
     |  println("三个数的积为:"+x*y*z)
     |  x*y*z
     |  }
     | add _
     | }
product: (x: Int, y: Int)Int => Int

scala> product(3,4)
两个数的积为:12
res19: Int => Int = <function1>

scala> product(3,4)(5)
两个数的积为:12
三个数的积为:60
res20: Int = 60
(2)柯理化

柯理化是闭包的一种简化形式,写法更加简洁,但是使用没有闭包灵活,必须传入指定数目参数才能执行。

# 比较两个值的较大值
# 1 闭包写法
scala> def max(x:Int)={
     |  def max1(y:Int)={
     |  if (x>y) x else y
     |  }
     | max1 _
     | }
max: (x: Int)Int => Int

scala> max(1)
res21: Int => Int = <function1>

scala> max(1)(2)
res22: Int = 2
# 2 柯理化写法
scala> def max2(x:Int)(y:Int)={
     | if (x>y) x else y
     | }
max2: (x: Int)(y: Int)Int

scala> max2(1)(2)
res23: Int = 2
(3)函数作为参数

函数可以作为参数,这也是高阶函数的一种。

# 函数作为参数传入
scala> def printResult(x:Int,y:Int,f:(Int,Int)=>Int):Unit={
     | println(f(x,y))
     | }
printResult: (x: Int, y: Int, f: (Int, Int) => Int)Unit
# 定义函数1
scala> def sum(x:Int,y:Int)=x+y
sum: (x: Int, y: Int)Int
# 定义函数2
scala> def minus(x:Int,y:Int)=x-y
minus: (x: Int, y: Int)Int
# 函数作为参数传入,根据函数不同,得到的结果也不同
scala> printResult(3,4,sum)
7
scala> printResult(3,4,minus)
-1 

函数作为参数传入,可以直接将匿名函数传入刚才定义的函数。

scala> printResult(3,4,(x:Int,y:Int)=>x*y)
12

如果参数类型确定,可以省略类型。

scala> printResult(3,4,(x,y)=>x*y)
12

如果参数只在匿名函数方法体中出现一次,可以用下划线来代替。

scala> printResult(3,4,_*_)
12

函数的递归

类似java和python,scala中也可以使用递归,需要注意的是递归需要指定返回值类型,因为无法通过类型自动推导。

# 求阶乘
scala> def getfactorial(num:Int):Int={
     | if (num==1) return 1 //不写return,会报StackOverflowError,因为执行到此处,会依然往后执行,函数使用最后一行结果作为返回
     | return num*getfactorial(num-1)
     | }
getfactorial: (num: Int)Int

scala> getfactorial(5)
res33: Int = 120

可变参数

类似java中参数可以传入可变参数(使用...),scala中也可以实现,使用 " 变量类型* " 来表示任意多个可变参数。

# 求和,参数变长
scala> def sum(arr:Int*)={
     | var sum=0
     | for(i <- arr) sum+=i # 增强型for循环
     | sum
     | }
sum: (arr: Int*)Int
scala> sum(1,2,3,4,5,6,7,8,9,10)
res35: Int = 55

函数中可设定默认值

函数中可以给参数默认值,如果方法没有传入此参数就按默认值来计算。

scala> def getOffPrice(money:Double,offFactor:Double=1)=money*offFactor
getOffPrice: (money: Double, offFactor: Double)Double
# 按默认值计算
scala> getOffPrice(100)
res36: Double = 100.0
# 按给定值计算
scala> getOffPrice(100,0.8)
res37: Double = 80.0

懒值

scala中设置懒值,可以让懒值加载时才执行。

package boe.com.clyang.function

/**
 * 懒值
 * 加载时才真的运行
 */
object LazyValue {

  def sum(i:Int,j:Int)={
    println("sum is running~~~")
    i+j
  }

  def main(args: Array[String]): Unit = {
    var i=3
    var j=4
    //将sum求和设置为懒值
    lazy val result=sum(i,j)

    # 线程等待3秒,自动回来,模拟这里有很多的代码
    println("sleeping")
    Thread.sleep(3000)

    //只有这里真正加载时才执行sum求和
    println(result)

  }
}

控制台结果,可以看出虽然sum方法在写在前面,但是因为设置了懒加载,在等待3秒后,println打印时才真正加载执行。

sleeping
sum is running~~~
7

以上,是scala入门基础知识,记录一下备用。

参考资料:

(1)菜鸟教程

(2)scala官网

(3)https://blog.csdn.net/qq_31573519/article/details/82749188 守护循环

以上是关于Scala学习-变量常量运算符流程控制和函数的主要内容,如果未能解决你的问题,请参考以下文章

Scala学习笔记一之基础语法,条件控制,循环控制,函数,数组,集合

2020寒假学习进度报告2

大数据周会-本周学习内容总结09

Scala学习(变量和数据类型流程控制)

Scala学习(变量和数据类型流程控制)

Scala 基础 :运算符和流程控制