快学scala-第七章 包和引入

知识点:

1. Scala、Java、C++的包的目的都是为了管理大型程序中的名称。与对象或类的定义不同,同一个包可以定义在多个文件当中。或者在同一个文件中,为多个包贡献内容。

2. Scala的包和其他作用域一样的支持嵌套,可以访问上层作用域中的名称。Scala会遇到默认引入包中的类和自定义包中的类的冲突问题,Java中不会出现这样的问题,包名总是绝对的,从包层级的最顶端开始,但是在Scala中,包名是相对的,就像内部类的名称一样。任何人都可以在任何时候向任何包添加内容。

3. 串联式包语句 com.horstmann.impatient 这样的包语句限定了可见的成员,文件顶部标记法是在文件顶部使用package语句,这比较适用于文件中所有代码属于同一个包的情况。

4. 包对象,包可以包含类、对象和特质,但不能包含函数或变量的定义,这是Java虚拟机的局限,包对象解决了这个局限。每个包都可以有一个包对象,需要在父包中定义它,且名称与子包一样,包对象被编译成带有静态方法和字段的JVM类,名为package.class,位于相应的包下。

5. 包可见性,在Java中,没有被声明为public、private或protected的类成员在包含该类的包中可见,在Scala中,可以通过修饰符达到同样的效果。

pacage com.horstmann.impatient.people

private[people] ….(自己的包中可见)

private[impatient]….(上层包,将可见性扩展到上层包)

6. 引入语句可以使用更短的名称而不是原来较长的名称,引入某个包的全部成员,使用类似于import java.awt._, 这和Java中的通配符*一样,在Scala中,*是合法的标识符。在Scala中,任何地方都可以声明引入,import语句效果一直延伸到包含该语句的块末尾。通过将引入放置在需要这些引入的地方可以大幅减少可能的名称冲突。

如果想要引入包中的几个成员,可以使用选取器,eg: import java.awt.{Color,Font}

选取器语法还允许重命名选到的成员,eg:

      import java.util.{HashMap => JavaHashMap}

      import scala.collection.mutable._

JavaHashMap就是java.util.HashMap,而HashMap则对应scala.collection.mutable.HashMap.

7.隐式引入,每个Scala程序都隐式地以如下代码开始:

import java.lang._

import scala._

import Predef._

这个引入被允许覆盖。

 

练习:参考网址

1.编写示例程序,展示为什么
package com.horstmann.impatient 
不同于
package com 
package horstmann 
package impatient

第一种方式子类不可以使用父包里的类.

package com{
  class T1(){}
  package horstmann{
    class T2(t:T1){}
    package impatient{
      class T3(t1:T1,t2:T2){}
    }
  }
}

package com.horstmann.impatient{
  class T4(t1:T1,t3:T3){}//not found:type T1
}

2.编写一段让你的Scala朋友们感到困惑的代码,使用一个不在顶部的com包

不懂.

3.编写一个包random,加入函数nextInt():Int,nextDouble():Double,setSeed(seed:Int):Unit。生成随机数的算法采用线性同余生成器: 
后值 = (前值 * a + b)mod 2^n 
其中,a = 1664525,b=1013904223,n = 32,前值的初始值为seed

package random{
  package object random{
    var seed:Int = _
    val a = BigDecimal(1664525)
    val b = BigDecimal(1013904223)
    val n = 32

    def nextInt:Int={
      var temp = (seed * a + b) % BigDecimal(2).pow(n)
      seed = temp.toInt
      seed
    }
    def nextDouble:Double={
      var temp = (seed * a + b) % BigDecimal(2).pow(n)
      seed = temp.toInt
      temp.toDouble
    }
  }
}

package test{
  import random.random
  object Test extends App{
    random.seed = 4
    println(random.nextInt)
    println(random.nextInt)
    println(random.nextInt)
    println(random.nextDouble)
    println(random.nextDouble)
    println(random.nextDouble)
  }
}

4.在你看来Scala的设计者为什么要提供package object语法而不是简单的让你将函数和变量添加到包中呢?

虚拟机的局限

5.private[com] def giveRaise(rate:Double)的含义是什么?有用吗?

private[com]定义包可见性,除了com包其他包都不能访问。

6.编写一段程序,将Java哈希映射中的所有元素拷贝到Scala哈希映射。用引入语句重命名这两个类.

import java.util.{HashMap => JavaHashMap}
import scala.collection.mutable.HashMap

object Test extends App{
  val map = new JavaHashMap[String,String]()
  map.put("1", "a")
  map.put("2", "b")
  map.put("3", "c")

  val smap = new HashMap[String,String]()
  for(key <- map.keySet().toArray){
    smap += (key.toString() -> map.get(key))
  }

  println(smap.mkString)
}

7.在前一个练习中,将所有引入语句移动到尽可能小的作用域里.

object Test extends App{
  import java.util.{HashMap => JavaHashMap}
  val map = new JavaHashMap[String,String]()
  map.put("1", "a")
  map.put("2", "b")
  map.put("3", "c")

  import scala.collection.mutable.HashMap
  val smap = new HashMap[String,String]()
  for(key <- map.keySet().toArray){
    smap += (key.toString() -> map.get(key))
  }

  println(smap.mkString)
}

8.以下代码的作用是什么?这是个好主意吗?

import java._ 
import javax._

导入Java和javax中的所有类,这两个包中中只有子包,不包含类,所以无用。

9.编写一段程序,引入java.lang.System类,从user.name系统属性读取用户名,从Console对象读取一个密码,如果密码不是"secret",则在标准错误流中打印一个消息;如果密码是"secret",则在标准输出流中打印一个问候消息。不要使用任何其他引入,也不要使用任何限定词(带句点的那种).

import java.lang.System

object Test extends App{
  val password = Console.readInt

  if(password equals "secret")
    System.out.println("Hello " + System.getProperty("user.name"))
  else System.err.println("password error")
}
时间: 2024-07-30 09:47:24

快学scala-第七章 包和引入的相关文章

快学scala 第十一章 操作符 读书笔记及习题答案代码

chapter 11 操作符 标签:快学scala 一.笔记 scala种可以在反引号中包含几乎任何字符序列, val 'val' = 42 所有的操作符都是左结合的,除了以冒号(:)结尾的操作符,和赋值操作符.用于构造列表的::操作符是又结合的.1::2::Ni1的意思是1::(2::Ni1),先创建出包含2的列表,这个列表又被作为尾巴拼接到以1作为头部的列表中. 2. 函数调用语法:f(arg1, arg2,...)扩展到可以应用于函数之外的值,如果f不是函数或方法,那么这个表达式等于f.a

快学Scala第14章----模式匹配和样例类

本章要点 match表达式是一个更好的switch,不会有意外掉入到下一个分支的问题. 如果没有模式能够匹配,会抛出MatchError.可以用case _ 模式来避免. 模式可以包含一个随意定义的条件,称作守卫. 你可以对表达式的类型进行匹配:优先选择模式匹配而不是isInstanceOf/asInstanceOf. 你可以匹配数组.元组和样例类的模式,然后将匹配到的不同部分绑定到变量. 在for表达式中,不能匹配的情况会被安静的跳过. 样例类继承层级中的公共超类应该是sealed的. 用Op

快学Scala第10章----特质

本章要点 类可以实现任意数量的特质 特质可以要求实现它们的类具备特定的字段.方法或超类 和Java接口不同,Scala特质可以提供方法和字段的实现 当你将多个特质叠加在一起时,顺序很重要--其方法先被执行的特质排在更后面 为什么没有多重继承 Scala和Java一样不允许类从多个超类继承:从多了超类继承可能会导致许多问题,例如两个超类有相同的方法,子类该如何使用和菱形继承.在java 中类只能扩展自一个超类,它可以实现任意数量的接口,但接口只能包含抽象方法,不能包含字段. Scala提供了特质(

快学Scala第13章----集合

本章要点 所有集合都扩展自Iterable特质 集合有三大类:序列.集.映射 对于几乎所有集合类,Scala都同时提供了可变的和不可变的版本 Scala列表要么是空的,要么拥有一头一尾,其中尾部本身又是一个列表 集是无先后次序的集合 用LinkedhashSet 来保留插入顺序,或者用SortedSet来按顺序进行迭代 '+' 将元素添加到无先后次序的集合中: +: 和 :+ 向前或向后追加到序列: ++将两个集合串接在一起: -和–移除元素 Iterable和Seq特质有数十个用于常见操作的方

快学Scala 第17章 - 类型参数 习题解答

1. 定义一个不可变类Pair[T,S],带一个swap方法,返回组件交换过位置的新对偶. package ex17_01 object Main extends App {   val p = new Pair(97 -> 'a')   val a = p.swap   println(a) } class Pair[T, S](val p: (T, S)) {   def swap = {     (p._2, p._1)   } } /*output: (a,97) */ 2. 定义一个可

快学Scala 第13章 集合 - 练习解答

1. 编写一个函数,给定字符串,产出一个包含所有字符的下标的映射.举例来说:indexes("Mississippi")应返回一个映射,让'M'对应集{0},'i'对应集{1,4,7,10},依此类推. 使用字符到可变集的映射.另外,你如何保证集是经过排序的? 回答:使用SortedSet可以保证集是经过排序的. package ex13_01 import scala.collection.mutable.SortedSet import scala.collection.mutab

快学Scala 第6章 对象 - 练习

1. 编写一个Conversions对象,加入inchesToCentimeters.gallonsToLiters和milesToKilometers方法. object Conversions {     def main(args: Array[String]){         printf("1 inch = %g centimeters\n", inchesToCentimeters(1))         printf("2 gallons = %g liter

快学Scala 第18章 高级类型 习题解答

1. 实现一个Bug类,对沿着水平线爬行的虫子建模.move方法向当前方向移动,turn方法让虫子转身,show方法打印出当前的位置.让这些方法可以被串接调用.例如: bugsy.move(4).show().move(6).show().turn().move(5).show() 上述代码应显示 4 10 5. package ex18_01 class Bug {   var x = 0   var y = 0   var curr_direction = 0   def move(len:

快学Scala第2章–控制结构和函数 笔记

条件表达式 在Scala中,if/else 表达式是有值的,这个就是跟在if或者else之后的表达式的值.例如: val s = if(x > 0) 1 else -1 // 类似于 var s = 0 if(x > 0) s = 1 else s = -1 Scala允许使用混合类型的返回值,例如: if(x > 0) "positive" else -1 上式表达式返回的类型是它们类型的公共超类型, 在这里java.lang.String 和 Int 它们的公共超