码迷,mamicode.com
首页 > 其他好文 > 详细

Scala入门系列(五):面向对象之类

时间:2017-12-11 14:26:02      阅读:206      评论:0      收藏:0      [点我收藏+]

标签:类对象   news   add   ack   调用   不同   ber   setter   server   

定义类

// 定义类,包含field以及method
class HelloWorld {
private var name = "Leo"
def sayHello() { println("Hello" + name)}
def getName = name
}
defined class HelloWorld
// 创建类对象,并调用其方法
scala> val helloworld = new HelloWorld
helloworld: HelloWorld = HelloWorld@4f49f6af
scala> helloworld.sayHello() // 也可以不加括号
HelloLeo
scala> print(helloworld.getName) // 不能加括号,因为定义方法的时候没有加括号
Leo

封装(setter与getter)

  • 定义不加参数的var filed,Scala会将该字段定义为private,并提供public的getter和setter方法
  • 如果使用private修饰field, 则生成的getter和setter也是private的
  • 如果定义val field, 则只会生成getter方法
  • 如果使用private[this]修饰field,则不会生成getter和setter方法 
    总结:Scala提供的四种修饰符:var、val、private、private[this]
class Student {
var name = "sparks"
}
defined class Student
scala> val spark = new Student
spark: Student = Student@6337c201
// 调用自动生成的setter和getter方法
scala> print(spark.name)
sparks
scala> spark.name = "hahah"
spark.name: String = hahah

如果希望能够自己对getter和setter进行控制,则可以自定义getter与setter方法,使用field和fild_=的方式。

class Student{
private var myName = "Sparks"
def name = "Your name is" + myName
def name_=(newName: String) {
myName = newName
}
}
val spark = new Student
print(spark.name)
spark.name = "Leo"

注意:自义定setter方法一定要注意scala的语法限制,签名、_=参数间不能有空格。

private[this]的详解

如果字段是private修饰的,那么代码这个字段时私有的,在类的方法中,可以直接访问类的其他对象的private field。 
例如:

class Student {
private var myAge = 0
def age_=(newValue: Int){
if(newValue>0) { myAge = newValue}
else print("illegal age!")
}
def age = myAge
def older(s: Student) = {
myAge > s.myAge // 调用另一个对象s的私有字段并不报错
}
}

  
但是如果是用private[this]修饰,那么意味着该对象私有字段只有本对象内才可以访问,上面的代码就会报错

<console>:16: error: value myAge is not a member of Student
myAge > s.myAge // 调用另一个对象s的私有字段并不报错

Java风格的getter和setter方法

如果要让scala自动生成java风格的getter和setter方法,只要给field添加@BeanProperty注解即可。 
此时会生成4个方法,name: String、 name_=(newValue: String): Unit、 getName(): String、 setName(newValue: String): Unit

// 定义类
import scala.reflect.BeanProperty
class Student{
@BeanProperty var name: String = _
}
defined class Student
// 测试类
scala> val s = new Student
s: Student = Student@5f303ecd
scala> s.setName("leo")
scala> s.getName()
res9: String = leo
scala> s.name
res10: String = leo
scala> s.name = "spark"
s.name: String = spark
scala> s.getName()
res11: String = spark
// 定义Java风格类的第二种方式
class Student(@BeanProperty var name: String)

构造函数

主构造函数

在Scala中,主constructor是与类名放在一起的,而且类中没有定义在任何方法或者是代码块之中的代码,就是主constructor的代码,这点很新颖,但感觉没有java清晰。

class Student(val name: String, val age: Int) {
println("your name is " + name + ", your age is " + age)
}
defined class Student
scala> val s = new Student("spark", 30)
your name is spark, your age is 30
s: Student = Student@43b40233
// 还可以使用默认参数
class Student(val name: String = "leo", val age: Int = 30) {
println("your name is " + name + ", your age is " + age)
}
defined class Student
// 这样新建对象时就不用传递参数了
scala> val s = new Student
your name is leo, your age is 30

注意:如果主constucutor传入的参数什么修饰都没有,比如name: String, 那么如果类内部的方法使用到了,则会生命为private[this] name; 否则没有该field,就只能被constructor代码使用而已。

辅助构造函数(this)

Scala中,可以给类定义多个辅助constructor,类似于java中的构造函数重载 
辅助constructor之间可以互相调用,而且必须第一行调用主constructor

// 定义类
class Student{
private var name = ""
private var age = 0
def this(name: String){
this() // 必须调用主构造函数
this.name = name
}
def this(name: String, age:Int) {
this(name) // 必须调用主构造函数
this.age = age
}
}
defined class Student
// 测试类
scala> val s1 = new Student
s1: Student = Student@387bf2d9
scala> val s2 = new Student("sparks")
s2: Student = Student@5e746d37
scala> val s3 = new Student("spark", 23)
s3: Student = Student@5524b72f

类部类

在Scala中,同样可以在类中定义内部类;但是与java不同的是,每个外部类对象的内部类,都是不同的类

// 定义内部类
import scala.collection.mutable.ArrayBuffer
class Class{
class Student(val name: String){}
val students = new ArrayBuffer[Student]
def getNewStudent(name: String) = {
new Student(name)
}
}
defined class Class
scala> val c1 = new Class
scala> val s1 = c1.getNewStudent("spark")
scala> c1.students += s1
scala> val c2 = new Class
scala> val s2 = c2.getNewStudent("leo")
scala> c1.students += s2
// 出错,因为每个外部类对象的内部类都是不同的类
<console>:15: error: type mismatch;
found : c2.Student
required: c1.Student
c1.students += s2
^

Scala入门系列(五):面向对象之类

标签:类对象   news   add   ack   调用   不同   ber   setter   server   

原文地址:http://www.cnblogs.com/LiCheng-/p/8022251.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!