===================== Scala语言的面向对象编程 ========================
一、面向对象的基本概念:把数据和操作数据的方法放到一起,作为一个整体(类 class)
面向对象的特征:
(1)封装
(2)继承
(3)多态
二、定义类: class,没有类的修饰符: public、protected、private
class Student1 { //定义属性 private var stuName:String = "Tom" private var stuAge:Int = 20 //定义方法(函数) def getStuName():String = stuName def setStuName(newName:String) = this.stuName = newName def getStuAge():Int = stuAge def setStuAge(newAge:Int) = this.stuAge = newAge } //开发一个测试程序,创建学生对象 // Java: public static void main()****** //概念:objec对象可以跟类名不一样,如果一样,这个object对象就是该类的伴生对象 object Student1{ //都是静态 def main(args: Array[String]): Unit = { //创建学生对象 var s1 = new Student1 //第一次输出 println(s1.getStuName()+"\t" + s1.getStuAge()) //调用set方法 s1.setStuName("Mary") s1.setStuAge(25) //第二次输出 println(s1.getStuName()+"\t" + s1.getStuAge()) //第三次输出: 直接访问类的私有成员 //问题:为什么可以直接访问类的私有成员??? ----> 讨论:属性的set和get方法 println(s1.stuName + "\t" + s1.stuAge) } }
运行:
三、属性的get和set方法
/* 属性的get和set方法 1、当定义属性的时候,如果是private,Scala会自动生成对应的set和get方法 private var stuName:String = "Tom" (1) get方法:stuName (2) set方法: stuName_ 2、如何只有get方法,没有set方法? ---> 将属性定义为: 常量 val private val money:Int = 1000 3、不希望生成get和set方法: private[this] 该属性只属于该对象私有 */ class Student2 { //定义属性 private var stuName:String = "Tom" //只有get方法 private val money:Int = 1000 } object Student2{ def main(args: Array[String]): Unit = { var s2 = new Student2 println(s2.stuName) //修改money值 ===> error 错误 //s2.money = 2000 } }
四、嵌套类(内部类): 类:Student 包含:课程Course
import scala.collection.mutable.ArrayBuffer class Student3 { //定义一个内部类(嵌套类): 学生选修的课程 //通过主构造器 class Course(val courseName:String,val credit:Int){ //其他的方法 } //属性 private var stuName:String = "Tom" private var stuAge:Int = 20 //定义一个数组来保存该学生选修的课程 private var courseList = new ArrayBuffer[Course]() //定义方法:往学生信息中添加新的课程 def addNewCourse(cname:String,credit:Int): Unit ={ //创建一门课程 var c = new Course(cname,credit) //加入list courseList += c } } object Student3{ def main(args: Array[String]): Unit = { //创建学生 var s3 = new Student3 //给学生添加课程 s3.addNewCourse("Chinese",3) s3.addNewCourse("English",3) s3.addNewCourse("Math",3) //输出 println(s3.stuName+"\t"+s3.stuAge) for(s <- s3.courseList) println(s.courseName + "\t" + s.credit) } }
运行:
五、构造器:(1)主构造器 (2)辅助构造器
package main.scala /** * Created by YOGA on 2018/2/3. * (1)主构造器:和类的申明在一起,只能有一个主构造器 * (2)辅助构造器:多个,通过关键字this */ class Student4(val stuName:String,val stuAge:Int) { //定义一个辅助构造器 def this(age:Int){ //调用主构造器 this("No Name",age) } } object Student4{ def main(args: Array[String]) { //使用主构造器创建学生对象 val s4 = new Student4("Tom",24) println(s4.stuName+ "\t" + s4.stuAge) //使用辅助构造器创建学生对象 var s5 = new Student4(25) println(s5.stuName+"\t"+s5.stuAge) } }
运行:
六、Object对象:相当于static关键字
1、单例模式:一个类只有一个对象
举例:生成信用卡的卡号
package main.scala /** * Created by YOGA on 2018/2/3. */ object CreditCard { //变量:保存信用卡的卡号 //该属性只属于该对象 private[this] var creditCardNumber:Long = 0 //产生卡号 def generateNewCCNumber() = { creditCardNumber += 1 creditCardNumber } //测试 def main(args: Array[String]) { //得到新的卡号:通过类名.方法 println(CreditCard.generateNewCCNumber()) println(CreditCard.generateNewCCNumber()) println(CreditCard.generateNewCCNumber()) println(CreditCard.generateNewCCNumber()) } }
运行:
2、应用程序对象: App -----> 可以省略main方法
package main.scala object MainAppObject extends App { //创建一个main方法 // def main(args: Array[String]): Unit = { // println("Hello World") // } println("Hello World") }
七、类的apply方法:省略new关键字
举例:val s1 = new Student
val s2 = Student ---> 必须要定义Student类的apply方法
package main.scala /** * 主构造器 */ class Student5(val stuName:String){ } object Student5 { //定义类的Apply方法 位置:定义在类的伴生对象中 def apply(stuName:String) ={ println("调用到了apply方法") //调用主构造器 new Student5(stuName) } def main(args: Array[String]) { //创建学生对象 var s1 = new Student5("Tom") println(s1.stuName) //省略new 关键字 var s2 = Student5("Mary") println(s2.stuName) } }
运行:
八、继承 extends
子类与父类同名时,需要加override
package main.scala //继承 /* * 1、基本的继承 * 2.复写父类方法 * 2.使用匿名子类 * */ //定义父类 class Person(val name:String,val age:Int){ //方法(函数) def sayHello():String = "Hello " + name + " and the age is "+age; } //定义子类 class Employee(override val name:String,override val age:Int,salary:Int) extends Person(name,age) { //重写父类中的sayhello override def sayHello():String = "子类中的sayHello方法" } object Demo1 { def main(args: Array[String]): Unit = { //创建一个Person对象 var p1 = new Person("Tom",20) println(p1.sayHello()) //创建一个子类 var p2:Person = new Employee("Mike",25,1000) println(p2.sayHello()) //创建一个匿名子类,从Person继承 var p3:Person = new Person("jerry",26){ //在匿名子类中重写父类的方法 override def sayHello():String = "匿名子类中的sayHello方法" } println(p3.sayHello()) } }
运行:
九、抽象类:方法只声明,不实现
package main.scala //抽象类 //父类: 抽象 交通工具 abstract class Vehicle{ //定义一个抽象方法 def checkType():String } //子类 class Car extends Vehicle{ def checkType():String = {"I am a car"} } class Bike extends Vehicle{ def checkType():String = {"I am a bike"} } object Demo2 { def main(args: Array[String]):Unit = { var v1:Vehicle = new Car var v2:Vehicle = new Bike println(v1.checkType()) println(v2.checkType()) } }
运行:
十、抽象字段:没有初始值的字段
package main.scala.p1 //父类:抽象 abstract class Person{ //就是一个抽象字段 var id:Int var name:String } //一种做法 abstract class Employee1 extends Person{ //var id:Int = 1 var name:String = "No Name" } //另一种做法: 定义一个主构造器,在主构造器中,提供抽象字段 class Employee2(var id:Int) extends Person{ //只提供name var name:String = "No Name" } class Demo3 { }
十一、特质trait:就是抽象类
与抽象类的最大区别:支持多重继承 :extends... with
package main.scala.p2 //特质:trait //有点像接口、也有点像抽象类 支持多重继承 trait Human{ //抽象字段 var id:Int var name:String //方法:可以是抽象,也可以不是 def sayHello():String = "Hello" + name } trait Action{ //定义一个抽象方法 def getActionName():String } //定义一个子类,从上面两个继承 //关键字: extends... with class Student6(var id:Int,var name:String) extends Human with Action{ //实现Action中的getActionName def getActionName():String = "Action is running" } object Demo4 { def main(args: Array[String]) { var s1 = new Student6(1,"Tom") println(s1.sayHello()) println(s1.getActionName()) } }
运行:
十二、包和包对象
举例:package ****{
class ****
def 函数
val 常量
等等
}
包可以被引用在程序的任何地方
可以把工具函数或者常量添加到包对象里,在其他的类里导入这个包对象就可以引用。
包对象:常量,变量,特质,方法,类,对象