快学Scala-第二章 控制结构和函数
知识点:
1.条件表达式 if(x>0) 1 else 0
scala每个表达式都有一个类型,如果一致则为它们的类型,如果为混合类型表达式,则为它们的公共超类型Any。
if(x>0) 1 可能没有输出值,在scala中,每个表达式都应该有某种值,引入Unit类,写作()。
if(x>0) 1 等同于 if(x>0) 1 else ()
2. 语句终止,如果在写较长的语句,需要分两行来写的话,就要确保第一行以一个不能用作语句结尾的符号结尾,通常用操作符。
3.块表达式和赋值,块中最后一个表达式的值就是块的值,赋值语句的值是Unit类型,因此不能串接一起。
4.打印:print\println,用readLine函数从控制台读取一行输入,用readInt、readByte、readDouble、readShort、readLong、readFloat、readBoolean、readChar读取不同数值字符。
5.循环
for(i <- 表达式) 让变量i遍历<- 右边表达式的所有值
to方法:1 to n 返回数字1到n(含)的Range区间,until方法返回一个并不包含上限的区间。
for(i <- 0 until “scala”.length) 等同于 for(i <- “scala”)
可以以 变量<- 表达式 的形式提供多个生成器,用分号隔开,每个生成器都可以带一个守卫。
for(i <- 1 to 3 ; j <- 1 to 3 if i != j) print((10 * i +j) +” ”) //将打印 12 13 21 23 31 32
for推导式:如果for循环的循环体以 yield 开始,则该循环会构造出一个集合,每次迭代生成集合中的一个值。for推导式生成的集合与它的第一个生成器是类型兼容的。
for(i <- 1 to 10) yield i % 3 //生成 Vector(1,2,0,1,2,0,1,2,0,1)
6.方法对对象进行操作,函数不是。Scala的函数也包括 名称、参数和函数体。必须给出所有参数的类型,只要函数不是递归的(编译器无法校验递归函数的返回类型)就不需要指定返回类型,函数体中最后一个表达式的值就是函数返回值。
7.函数的默认参数和带名参数,相对参数的数量,给出的值不够,默认参数会从后往前逐个应用进来,带名参数并不需要跟参数列表的顺序完全一致,如果混用未命名参数和带名参数,只要那些未命名的参数是排在前面的即可。
8.变长参数
def sum(args:Int*)={ var result = 0; for(arg <- args) result += arg; result}
val s = sum(1,2,3,4,6)
函数得到的是一个类型为Seq的参数,用for循环访问每一个元素。但如果已经有一个值得序列,则不能直接传入函数中,需要告诉编译器希望这个参数被当做参数序列处理,追加:_*.
val s = sum(1 to 5:_*)
9.过程:如果函数体包含在花括号当中但没有前面的=号,那么返回类型就是Unit,这样的函数成为过程,过程不返回值。
10.当val被声明为lazy时,它的初始化将被推迟,直到首次对它取值。懒值对于开销较大的初始化语句有用,是介于val和def的中间状态,val在被定义时即被取值;lazy在被首次使用时取值;def在每一次被使用时取值。
11.异常Scala没有“受捡”异常,不需要声明说函数或方法可能会抛出某种异常。控制器将在离抛出点最近的处理器中恢复,如果没有找到符合要求的异常处理器,则程序退出。throw表达式有特殊的类型Nothing,在if/else表达式中很有用,如果一个分支的类型是Nothing,那么if/else的类型就是另一个分支的类型。
练习:(参考CSDN上下载的资料)
1.一个数字如果为正数,则它的signum为1;如果是负数,则signum为-1;如果为0,则signum为0.编写一个函数来计算这个值。
scala> def signum(num:Int)={
| var r=0
| if(num>0) r=1
| else if(num<0) r= -1
| else r=0
| r}
signum: (num: Int)Int scala> print(signum(5))
1
scala> print(signum(0))
0
scala> print(signum(-8))
-1
一定要注意函数的返回值有个等号,不然就是返回Unit。r= –1,等号和负号要隔开,不然报错“value =- is not a member of Int”。
Scala中已经有此方法了。
scala> BigInt(5).signum
res6: Int = 1
2.一个空的块表达式{}的值是什么?类型是什么?
scala> print({})
()
scala> def test={}
test: Unit
scala> var kong = {}
kong: Unit = ()
类型是Unit,值为()。
3.指出在Scala中何种情况下赋值语句x=y=1是合法的。(提示:给x找个合适的类型定义)
书本中说赋值语句的类型是Unit,所以给x定义类型Unit即合法了。
scala> var x={}
x: Unit = () scala> var y=1
y: Int = 1 scala> x=y=0
x: Unit = ()
4.针对下列Java循环编写一个Scala版本:
for(int i=10;i>=0;i–)System.out.println(i);
scala> for(i <- 0 to 10 reverse) print(i+"\n")
5.编写一个过程countdown(n:Int),打印从n到0的数字。
scala> def countdown(n:Int){
| for(i <- 0 to n reverse)
| print(i+" ")
| }
countdown: (n: Int)Unit scala> countdown(5)
5 4 3 2 1 0
6.编写一个for循环,计算字符串中所有字母的Unicode代码的乘积。举例来说,"Hello"中所有字符串的乘积为9415087488L。
scala> var l:Long = 1
l: Long = 1 scala> for(i <- "Hello"){
| l=l*i.toLong
| } scala> l
res14: Long = 9415087488
7.同样是解决前一个练习的问题,但这次不使用循环。(提示:在Scaladoc中查看StringOps)
scala> var t:Long = 1
t: Long = 1 scala> "scala".foreach(t *= _.toLong) scala> t
res18: Long = 11569118220
8.编写一个函数product(s:String),计算前面练习中提到的乘积。
scala> def product(s:String):Long={
| var t:Long = 1
| for(i <- s){
| t *= i.toLong
| }
| t
| }
product: (s: String)Long scala> print(product("scala"))
11569118220
9.把前一个练习中的函数改成递归函数
scala> def product(s:String):Long={
| if(s.length == 1) return s.charAt(0).toLong
| else s.take(1).charAt(0).toLong * product(s.drop(1))
| }
product: (s: String)Long scala> print(product("scala"))
11569118220
10.编写函数计算xn,其中n是整数,使用如下的递归定义:
xn=y2,如果n是正偶数的话,这里的y=x(n/2)
xn = x*x(n-1),如果n是正奇数的话
x0 = 1
xn = 1/x(-n),如果n是负数的话
不得使用return语句。
scala> def mten(x:Double,n:Int):Double={
| if(n==0) 1
| else if(n>0 && n%2==0) mten(x,n/2) * mten(x,n/2)
| else if(n>0 && n%2==1) x * mten(x,n-1)
| else 1/mten(x,-n)
| }
mten: (x: Double, n: Int)Double scala> print(mten(2,3))
8.0
scala> print(mten(2,-3))
0.125
快学Scala-第二章 控制结构和函数的更多相关文章
- 《快学Scala》第二章 控制结构和函数
- 快学Scala 第二课 (apply, if表达式,循环,函数的带名参数,可变长参数,异常)
apply方法是Scala中十分常见的方法,你可以把这种用法当做是()操作符的重载形式. 像以上这样伴生对象的apply方法是Scala中构建对象的常用手法,不再需要使用new. if 条件表达式的值 ...
- 快学Scala 第二十二课 (apply和unapply)
apply和unapply: apply方法经常用在伴生对象中,用来构造对象而不用显式地使用new. unapply是当做是伴生对象的apply方法的反向操作.apply方法接受构造参数,然后将他们变 ...
- 快学Scala 第二十一课 (初始化trait的抽象字段)
初始化trait的抽象字段: trait Logged { println("Logged constructor") def log(msg: String){ println( ...
- 快学Scala 第二十课 (trait的构造顺序)
trait的构造顺序: 首先调用超类构造器 特质构造器在超类构造器之后,类构造器之前执行 特质从左向右被构造 每个特质当中,父特质先被构造 如果多个特质共有一个父特质,而那个父特质已经被构造,则不会被 ...
- 《快学scala》读书笔记(2)
第二章 控制结构和函数 1.条件表达式 (1)scala中if/else表达式有值,这个值就是跟在if或者else之后的表达式的值.如: if (x > 0) 1 else -1 这个表达式的 ...
- 快学Scala习题解答—第一章 基础
1 简介 近期对Scala比较感兴趣,买了本<快学Scala>,感觉不错.比<Programming Scala:Tackle Multi-Core Complexity on th ...
- 《快学Scala》
Robert Peng's Blog - https://mr-dai.github.io/ <快学Scala>Intro与第1章 - https://mr-dai.github.io/S ...
- 快学Scala 第十九课 (trait的abstract override使用)
trait的abstract override使用: 当我看到abstract override介绍的时候也是一脸懵逼,因为快学scala,只介绍了因为TimestampLogger中调用的super ...
随机推荐
- 朱丽叶—Cuda+OSG
#include <cuda_runtime.h> #include <osg/Image> ; typedef struct cuComplex { float r; flo ...
- P2P直连?经服务器中转?
当同一个系统的两个客户端A.B相互发送消息给对方时,如果它们之间存在P2P通道,那么消息传送的路径就有两种:直接经P2P通道传送.或者经服务器中转.如下图所示: 通常就一般应用而言,如果P2P通道能够 ...
- LeetCode OJ 42. Trapping Rain Water
Given n non-negative integers representing an elevation map where the width of each bar is 1, comput ...
- 【入门一】一些简单的C程序及VS的安装
实例1 用 printf 函数打印字符串 # include <stdio.h> // 预处理指令,结尾不加 ';' int main(void) // int表示该函数将返回一个整数值, ...
- 3.1 Data Member的绑定
文章开始提出了一段示例代码,并讨论了返回哪个x的问题.然后 a) 给出了我们普遍认为正确的回答,并肯定了这个想法: b) 提醒大家,这在以前的编译器实现中,并非正确. 在早期的实现中,首 ...
- 自定义Button成进度条
ProgressButton源码: package com.example.progressbutton; import android.content.Context; import android ...
- echarts图表里遇到的问题
1. 取消echarts的默认动画 : series:[{ animation:false }] 2. 控制散点图里散点的大小: series:[{ symbolSize:5 }] 3. 去除内部网格 ...
- 一个有用的shell脚本
#!/bin/bash #if [ $1 -eq null ]; then # echo "please input params1!" # exit #fi #if [ $2 - ...
- Sikuli:创新的图形化编程技术
Sikuli是一种使用截图进行UI自动化测试的技术.Sikuli包括sikul脚本,基于Jython的API以及sikuli IDE.Sikuli可以实现任何你可以在显示器上看到ui对象的自动化,你可 ...
- 如何查找僵尸进程并Kill之,杀不掉的要查看父进程并杀之
转自:如何查找僵尸进程并Kill之,杀不掉的要查看父进程并杀之 用ps和grep命令寻找僵尸进程#ps -A -ostat,ppid,pid,cmd | grep -e '^[Zz]'命令注解:-A ...