一.本章要点

  • 标识符由字母,数字或运算符构成
  • 一元和二元操作符其实是方法调用
  • 操作符优先级取决于第一个字符,而结合性取决于最后一个字符
  • apply和update方法在对expr(args)表达式求值时被调用
  • 提取器从输入中提取元组或值的序列

二.标识符

   变量,函数,类的名称统称为标识符。标准的(Java)字符和数字组成的序例,以下划线或者字母开头,以及Unicode字符。

  注:在Scala中,除了标准的,还可以使用任意序列的操作符字符:

      • 除字母、数字、下划线、括号(){}[]或分隔符.,;'"之外的ascii码,如!@#%&等等;
      • Unicode的数学符号,或Unicode的Sm和So类别中的其他符号。如**等
      • 还可以用''包含任意字符(包括关键字),例val 'val'=...

三.中置操作符

  a 标识符 b,标识符表示一个带有两个参数的方法(一个隐式的参数,一个显式的参数),如1 to 10实际调用的事1.to(10),1->10等同于1.->10,是中置表达式,操作符位于两个参数中间。

  定义操作符,例:

class Fraction(n:Int,d:Int){
private int num
private int den
def *(other:Fraction)=new Fraction(num * other.num,den*other.den)
}

四.一元操作符

  中置操作符是二元的,有两个参数,只有一个参数的操作符称为一元操作符。

  如果出现在参数之后就是后置操作符,如a 标识符,1 toString同于1.toString();+,-,!,~可以作为前置操作符,出现在参数之前,被转换成名为unary_操作符的方法调用,如-a等同于a.unary_-。

五.赋值操作符

  赋值操作符名称形式为操作符=,如a 操作符=b等同于a=a 操作符 b。例:a+=b等同于a=a+b。

  注:

      • <=、>=、!=不是赋值操作符;
      • 以=开头的操作符不是赋值操作符(==,===,=/=等)
      • 如果a有一个名为操作符=的方法,那么该方法直接引用

六.优先级

  当一次使用对个操作符,又没有括号,执行顺序按优先级执行。 

七.结合性

  当有一系列相同优先级的操作符时,操作符的结合性决定了它们是从左到右求值还是从右到左求值。

  注:在Scala中,除了以冒号(:)结尾的操作符(例:用于构造列表的::,2::Ni1等同于Ni1.::(2)),赋值操作符以外,都是左结合的  

八.apply和update方法

  Scala允许f(arg1,arg2,...)调用,如果f不是函数或者方法,则等同于f.apply(arg1,arg2,...),或者出现在等号的左侧如f(arg1,arg2,...)=value等同于f.update(arg1,arg2,...,value)。

  这个机制被用于数组和映射:

val scores=new scala.collection.mutable.HashMap(String,Int)
scores("Bob")=100//调用scores.update("Bob",100)
val bobScore=score["Bob"]//调用scores.apply("Bob")

  apply还常用于伴生对象中,构造对象不用显示的应用new,例:

class Fraction(n:Int,d:Int){
...} object Fraction{
def apply(n:Int,d:int)=new Fraction(n,d)
}

九.提取器

  提取器就是一个带有unapply方法的对象。可以当作伴生对象apply(接受参数,构造对象)的反向操作,接受一个对象,然后提取值,通常是用来构造该对象的值。

  例:

var Fraction(a,b)=Fraction(3,4)*Fraction(2,5)
//a和b分别被初始化称运算结果的分子和分母 case Fraction(a,b)=>...//a和b分别被绑到分子和分母 //模式匹配可能失败,因此unapply方法返回的是一个Option
object Fraction{
def unapply(input:Fraction)={
if (input.den==0) None else Some(input.num,inout.den)}  }

  注:每一个样例类都自动具备apply和unapply方法。

十.带单个参数或无参的提取器

  在Scala中,没有值带一个组件的元组,如果unapply要提取单值,则应该返回一个目标类型的Option。

  例:

object Number{
def unapply(input:String):Option[Int]=
{
try{
Some(Integer.parseInt(inut.trim))
}
catch{
case ex:NUmberFromatException=>None
}
}
}

十一.unapplySeq方法

  要提取任意长度的值的序列,应该使用unapplySeq来命名。返回一个Option[Seq[A]],其中A是被提取的值的类型,例:

object Name{
def unapplySeq(input:String):Option[Seq[String]]=
if(input.trim=="")None else Some(input.trim.split("\\s+"))
}

  

十二.练习

    1.左结合,从左往右依次执行

  2.

  3.

class Fraction {
var n:Int=_
var d:Int=_
def +(that:Fraction):Fraction=Fraction(this.n*that.d+that.n*this.d,this.d*this.d)
def -(that:Fraction):Fraction=Fraction(this.n*that.d-that.n*this.d,this.d*that.d)
def *(that:Fraction):Fraction=Fraction(this.n*that.n,this,d*that.d)
def /(that:Fraction):Fraction=Fraction(this.n*that.d,this.d*that.n)
def this(n:Int,d:Int){
this()
this.n=n
this.d=d
simplify()
}
def simplify(): Unit ={
var r=n%d
if(r==0){
n/=d
d=1
}
else if(d%r==0){
n/=r
d/=r
}
} override def toString: String = 1.0*n/d toString }
object Fraction{
def apply(n:Int,d:Int):Fraction=new Fraction(n,d)
}

Scala学习十一——操作符的更多相关文章

  1. scala学习笔记——操作符

    中置操作符(二元操作符),操作符位于两个参数之间.操作符包括字母,比如to,也可以包括操作符字符,比如1->10,等同于方法调用1.->(10) a 标识符 b 其中的标识符是一个带有两个 ...

  2. Scala学习随笔——Scala起步

    实验楼学习Scala语言的笔记,课程网址为https://www.shiyanlou.com/courses/490 一.Scala简介 Scala 是一门多范式的编程语言,类似于 Java .设计初 ...

  3. 基于.net的分布式系统限流组件 C# DataGridView绑定List对象时,利用BindingList来实现增删查改 .net中ThreadPool与Task的认识总结 C# 排序技术研究与对比 基于.net的通用内存缓存模型组件 Scala学习笔记:重要语法特性

    基于.net的分布式系统限流组件   在互联网应用中,流量洪峰是常有的事情.在应对流量洪峰时,通用的处理模式一般有排队.限流,这样可以非常直接有效的保护系统,防止系统被打爆.另外,通过限流技术手段,可 ...

  4. Scala学习笔记及与Java不同之处总结-从Java开发者角度

    Scala与Java具有很多相似之处,但又有很多不同.这里主要从一个Java开发者的角度,总结在使用Scala的过程中所面临的一些思维转变. 这里仅仅是总结了部分两种语言在开发过程中的不同,以后会陆续 ...

  5. Scala学习资源

    Scala学习资源: Scala官方网站:http://www.scala-lang.org/ Scala github:https://github.com/scala/scala Twitter ...

  6. Linq学习之操作符

    一.环境搭建 下面将逐步搭建我们学习的环境,这个环境不仅仅是这次需要使用,以后的教程一样需要使用这个环境.所以请大家务必按照 搭建这里的环境否则会影响你后面的学习. 我们用到的几张表 通知消息表: 用 ...

  7. 【Todo】【读书笔记】大数据Spark企业级实战版 & Scala学习

    下了这本<大数据Spark企业级实战版>, 另外还有一本<Spark大数据处理:技术.应用与性能优化(全)> 先看前一篇. 根据书里的前言里面,对于阅读顺序的建议.先看最后的S ...

  8. Spark基础-scala学习(一、入门)

    Scala解析器的使用 REPL:Read(取值)-> Evaluation(求值)-> Print(打印)->Loop(循环).scala解析器也被称为REPL,会快速编译scal ...

  9. 【转载】 强化学习(十一) Prioritized Replay DQN

    原文地址: https://www.cnblogs.com/pinard/p/9797695.html ------------------------------------------------ ...

随机推荐

  1. koa 实现下载文件

    文件下载需要使用到koa-send这个插件,该插件是一个静态文件服务的中间件,它可以用来实现文件下载的功能. 1.下载页面 static/download.html <!DOCTYPE html ...

  2. pytorch-VGG网络

    VGG网络结构 第一层: 3x3x3x64, 步长为1, padding=1 第二层: 3x3x64x64, 步长为1, padding=1 第三层: 3x3x64x128, 步长为1, paddin ...

  3. buildscript和allprojects的作用和区别是什么?

    在Android Studio的Project的build.gradle中, // Top-level build file where you can add configuration optio ...

  4. Mac配置jdk以及maven

    一 Mac配置JDK和Maven 1.安装成功jdk后 2.打开终端后,输入vim ~/.bash_profile 3.输入: export MAVEN_HOME=/usr/local/maven/a ...

  5. 实时更新DataGridView 合计值

    public partial class Form1 : Form { public Form1() { InitializeComponent(); dataGridView1.DataSource ...

  6. driver.switch_to.window(driver.window_handles[0])切换到最新打开窗口

    有时,使用selenium  定位页面元素时,浏览器明明打开的是需要定位的页面,但就是定位不到元素.打印一下元素page_source,会发现源码与页面不同. 主要问题是页面没有加载完成导致,需要设置 ...

  7. [GPU] Machine Learning on C++

    一.MPI为何物? 初步了解:MPI集群环境搭建 二.重新认识Spark 链接:https://www.zhihu.com/question/48743915/answer/115738668 马铁大 ...

  8. docker-搭建efk收集docker日志

    新建docker-compose.yml文件 version: '2' services: fluentd: build: ./fluentd volumes: - ./fluentd/conf:/f ...

  9. Vue组件v-if新渲染的组件不更新

    Vue组件v-if新渲染的组件不更新:可能原因是Vue识别到是相似组件(高度相似甚至相同)不会更新元素.给原来的组件和新组件分别给不同的key值让Vue识别为不同的组件.

  10. Elasticsearch常见错误与配置简介

    一.常见错误 1.1 root用户启动elasticsearch报错 Elasticsearch为了安全考虑,不让使用root启动,解决方法新建一个用户,用此用户进行相关的操作.如果你用root启动, ...