一.本章要点

  • 类中的字段自动带有getter方法和setter方法
  • 你可以用定制的getter/setter方法替换掉字段的定义,而不必修改使用类的客户端——这就是所谓的”统一访问原则“
  • 用@BeanProperty注解来生成JavaBean的getXxx/setXxx方法
  • 每个类都有一个主要的构造器,这个构造器和类定义”交织“在一起。它的参数直接称为类的字段。主构造器执行类体中所有的语句
  • 辅助构造器是可选的。它们叫做this

二.简单类和无参方法

Class Counter{
private var value=0 //你必须初始化字段
def increment(){ value=+1}// 方法默认是共有的
def current=value //可以用不带()的方式强制风格
}

  注:调用无参方法时,可以写上(),也可以不写,这里有一种风格(对于改值器方法[改变对象状态的方法]使用(),而对于取值器方法[不会改变对象状态的方法]去掉())

三.带getter和setter的属性

  Scala为每个字段都提供getter和setter方法 (gettter和setter分别对应xxx和xxx_)。

  注:

      • 如果字段是私有的,则getter和setter方法也是私有的;
      • 如果字段是val,则只有getter方法被生成;
      • 如果你不需要仍个getter和setter,可以将字段声明为private[this]    

四.只带getter的属性

  有时候你需要一个只读属性,有getter但没有setter,则可以使用val字段,这样Scala会生成一个私有的final字段和一个getter方法。

  注:

    通过这种方式客户端不能任意改值,但如果需要可以通过其他方式根据自己的需求改变,且在Scala中不能实现只写属性,例:

class Counter{
private var value=0
def increment(){value+=1}
def current=value }

  小结:

      • var foo:Scala会自动合成一个getter和setter
      • val foo:Scala自动合成一个getter
      • 由你来定义foo和foo_=方法
      • 由你来定义foo方法  

五.对象私有字段

  在Scala中,和Java/C++等一样,方法可以访问所有对象的私有字段,例:

Class Counter{
private var value=0
de increment(){value+=1}
def isLess(other:Counter)=value<other.value
//同样可以访问另一个对象的私有字段
}

  于是Scala允许我们定义更严格的访问限制,通过private[this]修饰符来实现。(private[this] var value=0 //类似某个对象.value这样的访问将不被允许,因此Counter类的方法只能访问当前对象的value字段,而不能访问同样是Counter类型的其他对象的字段,这样的访问有时被称为对象私有的[对于对象私有的字段,Scala根本不会生成getter和setter方法])。

  注:Scala允许你将访问权赋予指定的类,private[类名]修饰符可以定义仅有指定类的方法可以访问给定的字段。这里的类名必须是当前定义的类或者是包含该类的外部类。

六.Bean属性

  Scala会对定义的字段提供setter和getter方法,但是这些名称都不是预期的(Java工具依赖于setFoo和getFoo这样的命名)。使用@BeanProperty会自动生成四个方法,例子:

impoer scala.reflect.BeanProperty
class Person{
@BeanProperty var name:String=_
}

  提供的四个方法:

    1.name:String;

    2.name_=(new Value:String):Unit;

    3.getName():String;

    4.setName(new Value:String):Unit  

七.辅助构造器

  Scala和Java一样,可以有任意多个构造器(一个主构造器[重要]多个辅助构造器)。

  注意(构造器与Java/C++不同点):

    1.辅助构造器的名称是this(而不像Java中与类名相同);  

    2.每一个辅助构造器都必须以一个对先前已经定义的其他辅助构造器或主构造器的调用开始

  例:

class Person{
private var name=""
private var age=0
def this(name:String){
//一个辅助构造器
this()//调用主构造器
}
def this(name:String,age:Int){
this(name) //调用前一个辅助构造器
this.age=age
} }

八.主构造器

  如果没有显式定义主构造器,则自动拥有一个无参的主构造器(简单的执行类体中的所有语句而已)。

  要点:

    1.主构造器的参数直接放置在类名之后:

class Person(val name:String,val age:Int){
  }

    2.主构造器会执行类定义中的所有语句(如配置某个字段的特性时非常有用)

class MyProg{
private val props=new Properties
props.load(new FileReader("myprog.properties"))
//上述语句是主构造器的一部分
}

  注:

    主构造器的参数可以是任意形态的,如private,val,var的;

    也可以是普通的方法参数(不加val/var),这样的参数如何处理取决于它们在类中如何被使用,如:

      1.不带val/var的参数至少被一个方法使用,则升格为字段,类似于private[this] val字段的效果;

      2.否则,该参数不被保存为字段(仅仅是可以被主构造器访问的普通参数)

    如果想把构造器变成私有的,可以calss Person private(val id:Int){...}

九.嵌套类

  Scala中,几乎可以任何语法结构嵌套任何语法结构(函数中嵌套函数,类中嵌套类)

  

import scala.collection.mutable.ArrayBuffer
class Network{
class Member(val name:String){
val contacts=new ArrayBuffer[Member]
}
private val members=new ArrayBuffer[Member]
def join(name:sSring)={ val m=new Member(name)
member+=m
m
}
}

  每个实例对象都有自己的Member类,这和Java不同(Java的内部类从属于外部类)。

每个实例只能在各自中添加成员,而不能跨实例。

  如果不希望是上图那样,可以使用伴生对象或者类型投影:

伴生对象

类型投影

建立引用别名

十.练习

  

  1.

class Counter {
private var value = 0 def increment = if (value == Int.MaxValue) value else value += 1 def current = value } object Counter {
def main(args: Array[String]): Unit = {
val counter = new Counter
for (i <- 1 to Int.MaxValue)
counter.increment
println(counter.current) }

  2.

class BankAccount {
private var balance: Double = 0.0 def deposit(depamount: Double) = {
balance += depamount
} def withdraw(drawamount: Double) = {
balance -= drawamount
} def current = balance } object BankAccount {
def main(args: Array[String]): Unit = {
val bankAccount = new BankAccount
bankAccount.deposit(1000)
bankAccount.withdraw(200)
print(bankAccount.current)
}
}

  3.

class Time {
private var hours: Int = 0
private var minute: Int = 0 def this(hr: Int, min: Int) = {
this()
this.hours = hr
this.minute = min
} def before(other: Time): Boolean = {
hours < other.hours || (hours == other.hours && minute < other.minute)
} def <(other: Time) = before(other)
}
object Time{
def main(args: Array[String]): Unit = {
val t1=new Time(20,20)
val t2=new Time(20,25)
print(t1 < t2)
}
}

  4.

class TimeMinute {
private var minute: Int = 0 def this(hr: Int, min: Int) = {
this()
this.minute = min + hr*60
} def before(other: TimeMinute): Boolean = {
minute < other.minute
} def <(other: TimeMinute) = before(other)
} object TimeMinute {
def main(args: Array[String]): Unit = {
val t1 = new TimeMinute(22, 20)
val t2 = new TimeMinute(22, 25)
print(t1 < t2)
}
}

  5.

import scala.beans.BeanProperty

class Student {
@BeanProperty var id: Long = 0L
@BeanProperty val name: String = null }

  6.

class Person(ag: Int) {
val age = if (ag > 0) ag else 0
}
object Person{
def main(args: Array[String]): Unit = {
val p=new Person(-12)
println(p.age)
}
}

  7.

class Person(namestr: String) {
val firstName = namestr.split(" ")(0)
val lastName = namestr.split(" ")(1) }

  8.

class Car(val producer: String, val modelName: String, val modelYear: Int = -1, var carCode: String = "") {
}

  9.

 public class Car {
private String producer;
private String modelName;
private int modelyear;
private String carCode; public Car() { } public Car(String producer, String modelName) {
this.producer = producer;
this.modelName = modelName;
this.modelyear = -1;
this.carCode = "";
} public Car(String producer, String modelName, String carCode) {
this.producer = producer;
this.modelName = modelName;
this.modelyear = -1;
this.carCode = carCode;
} public Car(String producer, String modelName, int modelYear, String carCode) {
this.producer = producer;
this.modelName = modelName;
this.modelyear = modelYear;
this.carCode = carCode;
} public String getProducer() {
return producer;
} public void setProducer(String producer) {
this.producer = producer;
} public String getModelName() {
return modelName;
} public void setModelName(String modelName) {
this.modelName = modelName;
} public int getModelyear() {
return modelyear;
} public void setModelyear(int modelyear) {
this.modelyear = modelyear;
} public String getCarCode() {
return carCode;
} public void setCarCode(String carCode) {
this.carCode = carCode;
}
}

  10.

class Employ(val name:String,var salary:Double){
}

Scala学习五——类的更多相关文章

  1. Scala学习(五)---Scala中的类

    Scala中的类 摘要: 在本篇中,你将会学习如何用Scala实现类.如果你了解Java或C++中的类,你不会觉得这有多难,并且你会很享受Scala更加精简的表示法带来的便利.本篇的要点包括: 1. ...

  2. Scala学习(五)练习

    Scala中的类&练习 1. 改进Counter类,让它不要在Int.MaxValue时变成负数 程序代码: class Counter { private var value=100 def ...

  3. scala学习笔记——类和对象

    基础语法关于Scala程序,这是非常要注意以下几点. 区分大小写 - Scala是大小写敏感的,这意味着标识Hello 和 hello在Scala中会有不同的含义. 类名 - 对于所有的类名的第一个字 ...

  4. PHP面向对象学习五 类中接口的应用

    类中接口的应用 接口:一种成员属性全部为抽象的特殊抽象类,在程序中同为规范的作用   抽象类:1.类中至少有一个抽象方法.2.方法前需要加abstract 接口: 1.类中全部为抽象方法,抽象方法前不 ...

  5. scala 学习笔记(04) OOP(上)主从构造器/私有属性/伴生对象(单例静态类)/apply方法/嵌套类

    一.主从构造器 java中构造函数没有主.从之分,只有构造器重载,但在scala中,每个类都有一个主构造器,在定义class时,如果啥也没写,默认有一个xxx()的主构造器 class Person ...

  6. Scala中的类学习

    Scala中的类学习 从java了解类的情况下,了解Scala的类并不难.Scala类中的字段自动带getter和setter方法,用@BeanProperty注解生成javaBean对象的getXX ...

  7. Scala学习随笔——深入类和对象

    函数化对象(又称方程化对象)指的是所定义的类或对象不包含任何可以修改的状态. 本篇随笔就是着重记录函数化对象.定义了一个有理数类定义的几个不同版本,以介绍 Scala 类定义的几个特性:类参数和构造函 ...

  8. scala学习-类与对象

    类 / 对象 [<快学Scala>笔记] 一.类 1.Scala中的类是公有可见性的,且多个类可以包含在同一个源文件中: class Counter{ private var value ...

  9. scala学习笔记(四)样本类与模式匹配

    访问修饰符 格式:private[x]或protected[x],x指某个所属包.类或单例对象,表示被修饰的类(或方法.单例对象),在X域中公开,在x域范围内都可以访问: private[包名]:在该 ...

随机推荐

  1. group by用法提示:select涉及字段规则

    工资表t_salary如下:  id month  name  salary  1 201601  Jim  12  2 201601  Bruce  30  3 201601  Peter  23 ...

  2. SpringBoot整合guava缓存

    1.pom文件 <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...

  3. How to get full path of StreamWriter

     How to get full path of StreamWriter   In my version of the framework, this seems to work: string f ...

  4. github pr

    github----向开源框架提交pr的过程 https://blog.csdn.net/vim_wj/article/details/78300239github 的 pr 使用方法 https:/ ...

  5. Python写的大小写转换小工具

    几行代码的小工具,用于进行如下转换 TRANSACTIONS ON CLOUD COMPUTING => Transactions On Cloud Computing orig = 'TRAN ...

  6. Flink初探wordCout

    知识点 Flink介绍 1.无界数据-->数据不断产生 2.有界数据-->最终不再改变的数据 3.有界数据集是无界数据集的一个特例 4.有界数据集在flink内部是以一种终态数据集进行处理 ...

  7. layui上传文件前加入确认提示

    //上传文件 upload: function () { layui.use('upload', function () { var upload = layui.upload; //执行实例 var ...

  8. java IO流的API

    常用的IO流API有:[InputStream.OutputStream] [FileInputStream.FileOutputStream] [BufferedInputStream.Buffer ...

  9. WMPageController设置menuView的左右视图

    效果图如下: 绿色的是自定义的emenuView的rightView哟!!! 代码实现如下: // // CategoryVC.m // JSHui // // Created by Apple on ...

  10. coreDNS域名无法解析问题

    问题: 在pod内无法解析域名 解决: busybox的镜像有bug,导致ping可以解析,但是nslookup无法解析 kubectl run -it --rm --image=infoblox/d ...