Scala学习教程笔记二之函数式编程、Object对象、伴生对象、继承、Trait、

1:Scala之函数式编程学习笔记:

1:Scala函数式编程学习:
    1.1:Scala定义一个简单的类,包含field以及方法,创建类的对象,并且调用其方法:
        class User {
            
          private var name = "张三";
          def hello(): Unit ={
            println("hello : " + name)
          }
          //注:如果定义方法时不带括号,则调用的时候也不可以加括号,否则报错。
          def getName = name;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {
            
          def main(args: Array[String]): Unit = {
            var user = new User;
            user.hello();
            println(user.getName)
          }
        }
2:Scala中field字段的getter和setter详解教程:
    2.1:定义不带private的var field,此时scala生成的面向jvm的类时,会定义为Private的name字段,并提供public的getter和setter的方法:
        class User {

          var name = "张三";
          var age = 15;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            user.name="李四";
            user.age=20;
            println(user.name + " " + user.age)
          }
        }

    2.2:如果使用private修饰field,则生成的getter和setter也是private的:
        class User {

          private var name = "张三";
          private var age = 15;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            //如果使用private修饰field,则生成的getter和setter也是private的,所以调用会报错。
            //println(user.name + " " + user.age)
          }
        }
    2.3:如果定义val field,则只会生成getter方法:
        class User {

          val name = "张三";
          val age = 15;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            //如果定义val field,则只会生成getter方法。所以调用setter方法会报错的。
            //user.name_="张三";
            println(user.name + " " + user.age)
          }
        }
    2.4:如果不希望生成setter和getter方法,则将field声明为private[this]:
        class User {

          private[this] var name = "张三";
          private[this] var age = 15;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            //则将field声明为private[this],不生成setter和getter方法。所以调用setter和getter方法报错
            //user.name_="张三";
            //println(user.name + " " + user.age)
          }
        }
    2.5:调用getter和setter方法,分别叫做name和name_=
        class User {

          var name = "张三";
          var age = 15;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            println(user.name + " " + user.age)
            //调用setter方法来修改值
            user.name="李四";
            println(user.name + " " + user.age)
          }
        }
    2.6:如果只是希望拥有简单的getter和setter方法,那么就按照scala提供的语法规则,根据需求为field选择合适的修饰符就好:var,val,private,private[this];
        注意:如果希望能够自己对gettter和setter进行控制,则可以自定义getter和setter方法,自定义setter方法的时候一定要注意scala的语法限制,签名,=,参数间不能有空格
        class User {
          var name : String = "张三";
          var age : Int = 15;
          def getName = "my name is : " + name;
          def setName_=(newName : String): Unit ={
            println("不可以修改你的姓名。")
          }
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            user.name="李思思"
            println(user.getName + " " + user.age)

          }
        }
    2.7:如果不希望field有setter方法,则可以定义为val,但是此时就再也不能更改field的值了。如果希望能够仅仅暴漏出一个getter方法,并且还能通过某些方法更改field的值,那么需要综合使用private以及自定义getter方法。此时,由于field是private的,所以setter和getter都是private的,对外界没有暴漏,自己可以实现修改field值的方法;自己可以覆盖getter方法。
        class User {

          private var myName : String = "张三";
          def updateName(newName : String): Unit ={
            if(newName == "李四"){
              myName = newName;
            }else{
              println("此值不可以修改")
            }
          }

          def name = "you name is :" + myName;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            user.updateName("李四");
            println(user.name)

          }
        }
    2.8:如果将field使用private来修饰,那么代表这个field是类私有的,在类的方法中,可以直接访问类的其他对象的private field;这种情况下,如果不希望field被其他对象访问到,那么可以使用private[this],意味着对象私有的field,只有本对象内可以访问到。
        class User {

         private[this] var myAge : Int = 0;
         def age_=(newAge : Int): Unit ={
           if(newAge > 0){
              myAge = newAge;
           }else println("不合法的年龄")
         }

          def age = myAge;
          //使用private[this],意味着对象私有的field,只有本对象内可以访问到。
          //def orderAge(user : User) ={
            //myAge > user.myAge;
          //}
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            user.age_=(20);
            println(user.age)
            user.age = 24;
            println(user.age)
            var user2 = new User;
            user2.age_=(25);
            if(user2.orderAge(user)){
              println("user2 大于 user")
            }else{
              println("user2 小于 user")
            }
          }
        }
    2.9:Scala的getter和setter方法的命名与java是不同的,是field和field_=的方式,如果要让scala自动生成java风格的getter和setter方法,只要给field添加@BeanProperty注解即可;此时会生成4个方法,name:String,name_=(newName:String):Unit,getName():String,setName_(newValue:String):Unit;
        class User {

          @BeanProperty var name : String = _;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            user.setName("张三");
            println(user.getName);
            user.name_=("李思思");
            println(user.name)
          }
        }        
3:Scala中constructor详解:
    3.1:Scala中,可以给类定义多个辅助constructor,类似于java中的构造函数重载;辅助constructor之间可以互相调用,而且必须第一行调用主constructor
        class User {

          @BeanProperty var name : String = _;
          @BeanProperty var age : Int =_;

          def this(name: String){
            this();
            this.name = name;
          }
          def this(name : String, age :Int){
            this(name);
           this.age = age;
          }
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User;
            var user2 = new User("张三");
            var user3 = new User("李四",26);
          }
        }
    3.2:Scala中,主构造constructor是与类名放到一起的,与java不同。而且类中,没有定义在任何方法或者是代码块之中的代码,就是主constructor的代码,这是感觉没有java那么清晰;
        注意:如果主constructor传入的参数什么修饰都没有,比如name:String。那么类内部的方法使用到了,则会声明为private[this] name,否则没有该field,就只能被constructor代码使用而已。
        class User(name : String, age : Int) {
          println("you name is :" + name + ", you age is : "+ age)

          var id : Int = _;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User("张三",16);
            user.id_=(20);
            println("you id is :" +user.id)

          }
        }
    3.3:主构造方法constructor方法中还可以通过使用默认参数,来给参数默认的值:
        class User(val name : String="李思思", age : Int = 18) {
          println("you name is :" + name + ", you age is : "+ age)

          var id : Int = _;
        }
        //创建一个object来调用练习的实体类对象.
        object Hello {

          def main(args: Array[String]): Unit = {
            var user = new User();
            user.id_=(20);
            println("you id is :" +user.id)

          }
        }
4:Scala中内部类的介绍:
    4.1:在Scala中,同样可以在类中定义内部类,但是与java不同的是,每个外部类的对象的内部类,都是不同的类:
        class User {
                
          class Student(val name : String){};
          val students = new ArrayBuffer[Student];
          def getStudent(name : String) = {
            new Student(name);
          }
        }
        //创建一个object来调用练习的实体类对象.
            object Hello {

              def main(args: Array[String]): Unit = {
                val user1 = new User();
                val stu = user1.getStudent("张三");
                println(stu)
                user1.students += stu;

                var user2 = new User;
                var stu2 = user2.getStudent("李四");
                println(stu2)
                //下面这一行报错,好好体会一下
                //user1.students += stu2;
              }
            }

2:Scala之Object对象学习笔记:

1:Object对象:
    1.1:object对象,相当于class的单个实例,通常在里面放一些静态的field或者method;第一次调用object的方法时候,就会执行object的constructor构造方法,也就是Object内部不在method中的代码;但是Object不能定义接受参数的constructor;object通常用于作为单例模式的实现,或者放class的静态成员,比如工具方法;
        注意:object的constructor只会在其第一次被调用的时候执行一次,以后再次调用就不会执行constructor了。
        object Hello {

          private val name = "张三";
          println("this is object Hello");
          def show(): Unit ={
            println("一步一个脚印");
          }
          def getName = name;
          def main(args: Array[String]): Unit = {
            show()
            println(getName);
            println(Hello.getName)
          }
        }

3:Scala之伴生对象学习笔记:

1:伴生对象,如果有一个class,还有一个与class同名的Object,那么就称这个object是class的伴生对象,class是object的伴生类;伴生类与伴生对象必须存放在一个.scala文件之中;伴生类与伴生对象,最大的特点就是在于,互相可以访问private field;

4:Scala之继承学习笔记:

1:让object继承抽象类:
    1.1:object的功能其实和class类似,除了不能定义接受参数的constructor之外,object也可以继承抽象类,并且覆盖抽象类中的方法:
        abstract class User(val name : String) {

          def hello(name : String): Unit ={
            //println("you name is : "+ name)
          }
        }
        //创建一个Object继承User类
        object UserImpl extends User("张三"){

          override def hello (name: String): Unit = {
            println("you name is :" + name)
          }
        }
        //创建一个Object来进行测试
        object Test {

          def main(args: Array[String]): Unit = {
            //object的功能其实和class类似,除了不能定义接受参数的constructor之外
            var ui = UserImpl;
            //方式一
            ui.hello("李四");
            //方式二
            UserImpl.hello("王五");
          }
        }
2:Apply方法,object中非常重要的一个特殊方法,就是apply方法。通常在伴生对象中实现apply方法,并在其中实现构造伴生类的对象的功能。而创建伴生类的对象时,通常不会使用new Class的方式,而是使用Class()的方式,隐式的调用伴生对象得到apply方法,这样会让对象创建更加简洁:
    2.1:比如,Array类的伴生对象的apply方法就实现了接受可变数量的参数,并且创建一个Array对象的功能:
        var arr = Array(1,2,3,4,5,6,7,8,9);
    2.2:比如,定义自己的伴生类和伴生对象:    
        class Person(val name : String) {
      
        }    
        //创建伴生对象
        object Person{
            def apply(name : String) = new Person(name)
        }
        //创建Object进行测试
        object Test {

          def main(args: Array[String]): Unit = {
            val p1 = new Person("张三");
            println(p1.name);
            val p2 = Person("李思思");
            println(p2.name);
          }
        }
3:main方法,就如同java中,如果要运行一个程序,必须编写一个包含main方法类一样,在scala中,如果想要运行一个应用程序,那么必须有一个main方法,作为入口;
    3.1:注意:scala中的main方法定义为def main(args: Array[String]): Unit = {}。而且必须定义在object中;
        App Trait的工作原理,App Trait继承自DelayedInit Trait,scalac命令进行编译时候,会把继承App Trait的object的construtcor代码都放到DelayedInit Trait的delayedInit方法中执行;
        
        object Test {

          def main(args: Array[String]): Unit = {
            

          }
        }        
    3.2:除了自己实现main方法以外,还可以继承App Trait,然后将需要在main方法中运行的代码,直接作为Object的construstor代码。而且用args可以接受传入的参数:
        object Test extends App{

          if(args.length > 0){
            println("hello : " + args(0))
          }else println("hello 你妹啊 hello.")
        }    
4:用object来实现枚举功能:
    4.1:scala没有直接提供类似于java中的Enum这样的枚举特性,如果要实现枚举,则需要用Object继承Enumeration类,并且调用value方法来初始化枚举值:
        object Season extends Enumeration{

          val SPRING,SUMMER,AUTUMN,WINTER = Value;
        }
    4.2:还可以通过value传入枚举值的id和name,通过id和toString可以获取,还可以通过id和name来查找枚举值:
        object Season extends Enumeration{

        //  val SPRING,SUMMER,AUTUMN,WINTER = Value;
          val SPRING = Value(0,"spring");
          val SUMMER = Value(1,"summer");
          val AUTUMN = Value(2,"autumn");
          val WINTER = Value(3,"winter");

          def main (args: Array[String]): Unit = {
            println(Season(0));
            println(Season.withName("spring"));
          }
        }
    4.3:使用枚举object.values可以遍历枚举值:
        object Season extends Enumeration{

          //val SPRING,SUMMER,AUTUMN,WINTER = Value;
          val SPRING = Value(0,"spring");
          val SUMMER = Value(1,"summer");
          val AUTUMN = Value(2,"autumn");
          val WINTER = Value(3,"winter");

          def main (args: Array[String]): Unit = {
            for(i <- Season.values){
              println(i)
            }
          }
        }
5:Scala中,让子类继承父类,与Java一样,也是使用extends关键字;
    5.1:继承就代表,子类可以从父类继承父类的field和method,然后子类可以在自己内部放入父类没有的field或者method;子类拥有特有的field和method,使用继承可以有效的复用代码。
        class Person {

          private var name : String = "张三";
          def getName = name;
          def setName_=(name : String): Unit ={
            println("private修改的字段,生成的setter和getter也是私有的。")
          }
        }
        //创建一个Student类来继承Person类:
        class Student extends Person{

          private  var score : Int = 60;
          def getScore = score;
          def setScore_=(score : Int): Unit ={
            println("private修改的字段,生成的setter和getter也是私有的。")
          }
        }
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student();
            println("my name is : " + student.getName + ", and my score is : " + student.getScore);
          }
        }
    5.2:子类可以覆盖父类的field和method,但是如果父类用final修饰,field和method用final修饰,则该类是无法被继承的,field和method是无法被覆盖的。
        //父类用final修饰
        final class Person {

          private var name : String = "张三";
          def getName = name;
          def setName_=(name : String): Unit ={
            println("private修改的字段,生成的setter和getter也是私有的。")
          }
        }
        //创建一个Student类来继承Person类:
        class Student extends Person{

              private  var score : Int = 60;
              def getScore = score;
              def setScore_=(score : Int): Unit ={
                println("private修改的字段,生成的setter和getter也是私有的。")
              }
            }
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student();
            //父类用final修饰,则该类是无法被继承的,所以下面会报错
            //println("my name is : " + student.getName + ", and my score is : " + student.getScore);
          }
        }
6:Scala中的override和super:
    6.1:Scala中,如果子类要覆盖一个父类中的非抽象方法,则必须使用override关键字;override关键字可以帮助我们尽早的发现代码里面的错误,比如,override修改的父类方法的方法名我们拼写错误了,比如要覆盖的父类方法的参数我们写错了等等。此外,在子类覆盖父类方法以后,如果我们在子类中就要调用父类的被覆盖的方法呢?那就可以使用super关键字,现实的指定要调用父类的方法。
        class Person {

          private var name : String = "张三";
          def getName = name;
          def setName_=(name : String): Unit ={
            println("private修改的字段,生成的setter和getter也是私有的。")
          }
        }
        //创建一个Student类来继承Person类:
        class Student extends Person{

              private  var score : Int = 60;
              def getScore = score;
              def setScore_=(score : Int): Unit ={
                println("private修改的字段,生成的setter和getter也是私有的。")
              }
              //方法的覆盖,使用关键词override和super
              override def getName: String = "Student类继承Person,且覆盖getName方法:" + super.getName;
            }
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student();
            println("my name is : " + student.getName + ", and my score is : " + student.getScore);
          }
        }
    6.2:Scala中,子类可以覆盖父类的val field,而且子类的val field还可以覆盖父类的val field的getter方法;只要在子类中使用override关键字即可;
        class Person {

          val name : String = "张三";
          def age : Int = 0;
        }
        //创建一个Student类来继承Person类:    
        class Student extends Person{

          override val name : String = "李四";

          override val age : Int = 20;
        }    
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student();
            println("my name is : " + student.name +   ",my age is :" + student.age);
          }
        }
7:isInstanceOf和asInstanceOf,如果我们创建了子类的对象,但是又将其赋予了父类类型的变量。则在后续的程序中,我们又需要将父类类型的变量转换为子类类型的变量。首先,需要使用isInstanceOf判断对象是否是指定类的对象,如果是的话,则可以使用asInstanceOf将对象转换为指定类型。
    注意:如果对象是null,则isInstanceOf一定返回false,asInstanceOf一定返回null;
         如果没有用isInstanceOf先判断对象是否为指定类的实例,就直接用asInstanceOf转换,则可能会抛出异常;
        class Person {

          var name : String = "张三";
          var age : Int = 20;
        }
        //创建一个Student类来继承Person类:    
        class Student extends Person{

          var sex : String ="男";

        }
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            //父类的变量引用了子类的对象.
            val p : Person = new Student;
            var s : Student = null;
            //注意是[]不是(),否则报错。
            if(p.isInstanceOf[Student]){
              s = p.asInstanceOf[Student];
            }
            println(" my name is :"+ s.name + " ,my age is :" + s.age)
          }
        }
8:getClass和classOf,isInstanceOf只能判断出对象是否是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象。如果要求精确的判断对象就是指定类的对象,那么只能使用使用getClass和classOf了。
    用法如下所示:
        对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类没然后使用==操作符即可判断:    
        class Student extends Person{

          var sex : String ="男";
        }
        //创建一个Student类来继承Person类:    
        class Person {

          var name : String = "张三";
          var age : Int = 20;
        }
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            //父类的变量引用了子类的对象.
            val p : Person = new Student;
            var s : Student = null;
            //注意是[]不是(),否则报错。
            if(p.isInstanceOf[Student]){
              s = p.asInstanceOf[Student];
            }
            println("======================================================================")
            //注意,isInstanceOf不可以精确判断是子类还是父类的。
            if(p.isInstanceOf[Person] && p.isInstanceOf[Student]){
              println("0:p指向了Person,也指向了Student");
            }
            println("======================================================================")
            println(" my name is :"+ s.name + " ,my age is :" + s.age)
            if(p.getClass == classOf[Person]){
              //p指向的是Student;
              println("1:getClass的用法:" + p.getClass + ",classOf的用法:" + classOf[Person]);
            }
            println("======================================================================")
            if(p.getClass == classOf[Student]){
              println("2:getClass的用法:" + p.getClass + ",classOf的用法:" + classOf[Student]);
            }
            println("======================================================================")
            val p2 : Person = new Person;
            if(p2.getClass == classOf[Person]){
              //p2指向的是Person;
              println("3:getClass的用法:" + p2.getClass + ",classOf的用法:" + classOf[Person]);
            }
          }
        }
9:使用模式匹配进行类型判断:
    9.1:在实际开发中,比如Spark的源码中,大量的地方都是使用了模式匹配的方式进行类型的判断,这种方式更加的简洁明了,而且代码的维护性和可扩展性也很高。
        使用模式匹配,功能性上来说,与instanceOf一样,也是判断主要是该类以及该类的子类的对象即可,不是精确判断的:
        class Student extends Person{

          var sex : String ="男";
        }
        //创建一个Student类来继承Person类:    
        class Person {

          var name : String = "张三";
          var age : Int = 20;
        }
        //创建一个Object来调用创建的Student或者Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            //父类的变量引用了子类的对象.
            val p : Person = new Student;

            //模式匹配
            p match {
              case person : Person => println("It is Person class");
              case student : Student => println("It is Student class");
              case _ => println("不知道是什么类型的.");
            }
          }
        }
10:Protected关键字,跟java一样,scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字,直接就可以访问field和method;
    10.1:注意:还可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法通过其他子类对象访问父类的field和method;
        class Person {

          protected var name : String = "张三";
          protected[this] var age : Int = 20;
        }
        //创建一个Student类来继承Person类:    
        class Student extends Person{

          var sex : String ="男";
          def showPerson(): Unit ={
            println("my name is : " + name);
          }
          def makeFriends(s : Student): Unit ={
            //还可以使用protected(this)则只能在当前子类对象中访问父类的field和method,
            //无法通过其他子类对象访问父类的field和method;
            //下面的s.age会报错的。protected[this] var age : Int = 20;
            //println("my age is : " + age  + ",you age is : " + s.age);
          }
        }
11:调用父类的constructor,在scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor;因此子类的辅助constructor是一定不可能直接调用父类的constructor的;
    注意:只能在子类的主constructor中调用父类的constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数;如果是父类中接受的参数,比如name和age,子类中接受时,就不要用任何val或者var来修饰了,否则会认为是子类要覆盖父类的field;
        class Person(val name : String,val age : Int) {

        }
        //创建一个Student类来继承Person类:        
        class Student(name : String,age : Int,var score : Int) extends Person(name,age){

          def this(name : String){
            this(name,0,0);
          }
          def this(age : Int){
            this("张三",age,0);
          }
        }
        //创建一个Student类来继承Person类:    
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student("李思思",22,100);
            println(student.name + " "  + student.age + " " + student.score)
            var student2 = new Student("王五");
            println(student2.name + " "  + student2.age + " " + student2.score)
            var student3 = new Student(20);
            println(student3.name + " "  + student3.age + " " + student3.score)

          }
        }
12:匿名子类,在Scala中,匿名子类是非常常见,而且非常强大的。Spark的源码中也大量使用了这种匿名子类。
    匿名子类,也就是说,可以定义一个类的没有名称的子类,并且直接创建其对象,然后将对象的引用赋予一个变量。之后甚至可以将该匿名子类的对象传递给其他函数。
        class Person(protected val name : String) {

          def hello()= "hello, I am :" + name;
        }
        //创建一个Student类来继承Person类:    
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            //匿名内部类
            var p = new Person("张三"){
              override def hello(): String = "匿名内部类,hello :" + name;
            }
            //调用
            println(p.hello());
          }
        }
13:抽象类,如果在父类中,有某些方法无法立即实现,而需要依赖不同的子类来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法;
    13.1:注意:一个类中如果有一个抽象方法,那么类就必须用abstract来声明为抽象类,此时抽象类是不可以实例化的。
        在子类中覆盖抽象类的抽象方法时,不需要使用override关键字;
        abstract class Person(val name : String) {

          def hello() : Unit;
        }
        //创建一个Student类来继承Person类:    
        class Student(name : String) extends  Person(name){

          override def hello(): Unit = println("hello : " + name);
        }
        //创建一个Student类来继承Person类:
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student("张三")
            student.hello()
          }
        }
    13.2:抽象field,如果在父类中,定义了field,但是没有给出原始值,则此field为抽象field;
         抽象field意味着,scala会根据自己的规则,为var或者val类型的field生成对应的getter和setter方法,但是父类中是没有该field的。
         子类必须覆盖field,以定义自己的具体field,并且覆盖抽象field,不需要使用override关键字;
        abstract class Person {

          val name : String;
        }    
        //创建一个Student类来继承Person类:    
        class Student extends  Person{

          val name : String= "张三";
        }
        //创建一个Student类来继承Person类:    
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var student = new Student
            println(student.name)
          }
        }

5:Scala之面向对象编程之Trait学习笔记:

1:trait基础知识:
    1.1:将trait作为接口使用:
        a、Scala中的trait是一种特殊的概念,首先我们可以将trait作为接口来使用,此时的trait就与Java中的接口非常类似;
        b、在trait中可以定义抽象方法,就与抽象类中的抽象方法一样,只要不给出方法的具体体现即可;
        c、类可以使用extends关键字来继承trait,注意,这里不是implement,而是extends,在scala中没有implmemts的概念,    无论继承还是trait,统一都是extends;
        d、类继承trait后,必须实现其中的抽象方法,实现时候不需要使用override关键字;
        e、scala不支持对类进行多继承,但是支持多重继承trait,使用with关键字即可;
        trait Person {

          def hello(name : String);
        }
        //定义一个MakeFriends的trait
        trait MakeFriends {

          def makeFriends(friend: Friend);
        }
        //定义一个类来继承上面两个trait
        class Friend(val name : String) extends Person with MakeFriends with Cloneable with Serializable{

          def hello(name: String) = println("my name is : " +name)

          def makeFriends(friend: Friend)  = println("hello,my name is : " + name + ",your name is :" + friend.name);
        }
        //定义一个object来测试实现的类
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var friend = new Friend("张三");
            var friend2 = new Friend("李四");
            friend.hello("李四");
            friend.makeFriends(friend2);
          }
        }
    1.2:在trait中定义具体方法:
        Scala中的trait可以不是只定义抽象方法,还可以定义具体方法,此时trait更像是包含了通用工具方法的东西,有一个专有的名词来形容这种情况,就是说trait的功能混入了类。举例来说,trait中可以包含一些很多类都通用的功能方法,比如打印日志等等,Spark中就使用了trait来定义了通用的日志打印方法:
        trait Logger {

          def log(message : String) = println(message);
        }
        //定义一个类来实现trait接口
        class User(val name : String) extends Logger{

          def makeFriends(user: User): Unit ={
            println("hello, i am " + name + " i am nice to meet you :" + user.name);
            log("makeFriends logger User[name="+user.name+"]");
          }
        }
        //定义一个object来测试实现的类
        object HelloWorld {
        
        def main(args: Array[String]): Unit = {
            val user = new User("张三");
            val user2 = new User("李思思");
            user.makeFriends(user2);
          }
        }
 
    1.3:在trait中定义具体字段:
      Scala中的triat可以定义具体field,此时继承triat的类就自动获得了triat中定义的field,但是这种获取field的方式与继承class不同:如果是继承class获取的field,实际是定义在父类中的,而继承triat获取的field,就直接被添加到了类中。
        trait Person {

          def hello();
          //定义一个field
          val eyeNum : Int =2;
        }
        //定义一个类来实现trait接口
        class Friend(val name : String) extends Person{

          def hello() = println("my name is : " +name + ",and i have : " +eyeNum + "eyes");

        }
        //定义一个object来测试实现的类
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var friend = new Friend("张三");
            friend.hello();
          }
        }
    1.4:在trait中定义抽象字段:
      Scala中的Triat可以定义抽象field,而Triat中的具体方法则可以基于抽象field来编写,但是继承Triat的类,则必须覆盖抽象field,提供具体的值:
        trait Person {

          //定义一个抽象的field
          val msg : String;
          //Triat中的具体方法则可以基于抽象field来编写
          def hello(name : String) = println(msg + " ," + name);
        }
        //定义一个类来实现trait接口
        class Friend(val name : String) extends Person{
          //但是继承Triat的类,则必须覆盖抽象field,提供具体的值;
          val msg : String = "hello";
          def makeFriends(p : Person)={
            hello(name)
            println("my name is : " + name,",and i want to make friends with you.")
          }
        }
        //定义一个object来测试实现的类
        object HelloWorld {

          def main(args: Array[String]): Unit = {
            var friend = new Friend("张三");
            var friend2 = new Friend("张三");
            friend.makeFriends(friend2)
          }
        }
2:trait高级知识:
    2.1:位实例对象混入trait:
      有时候,我们可以在创建类的对象的时候,指定该对象混入某个trait,这样,就只有这个对象混入该trait的方法,而类的其他对象则没有:
        import scala.util.logging.Logged

        trait MyLogged extends Logged{

          override def log(msg: String): Unit = {
            println("log : " + msg);
          }
        }
        //定义一个类来实现trait接口
        import scala.util.logging.Logged

        class Person(name : String) extends Logged{
          def hello(): Unit ={
            println("hi , i am is :" + name);
          }
          log("hello is invoked");
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            var person1 = new Person("张三");
            person1.hello();
            var person2 = new Person("李思思") with MyLogged;
            person2.hello()
          }
        }
    2.2:trait调用链:
      a、Scala中支持让类继承多个Trait后,依次调用多个trait中的同一个方法,只要让多个trait的同一个方法中,在最后都执行super方法即可;
        b、类中调用多个trait中都有的这个方法时,首先会从最右边的trait的方法开始执行,然后依次往左执行,形成一个调用链条;
        c、这种特性非常强大,其实就相当于设计模式中的责任链模式的一种具体实现依赖;
        trait Handler {

          def handler(data : String){}
        }
        //定义一个trait来实现trait接口
        trait DataValidHandler extends Handler{

          override def handler(data: String): Unit = {
            println("check data : " + data);
            super.handler(data)
          }
        }
        //定义一个trait来实现trait接口
        trait SignatureValidHandler extends Handler{

          override def handler(data: String): Unit = {
            println("signature : " + data)
            super.handler(data)
          }
        }
        //定义一个类来实现trait接口
        class Person(val name : String) extends SignatureValidHandler with DataValidHandler{

          def hello(): Unit ={
            println("hello :" + name);
            handler(name);
          }
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            val p1 = new Person("张三");
            p1.hello();
          }
        }
    2.3:在trait中覆盖抽象方法【注意语法】:
      在Trait中,是可以覆盖父trait的抽象方法的。但是覆盖时,如果使用了super.方法的代码,则无法通过编译。因为super.方法就会去掉用父trait的抽象方法,此时子trait的该方法还是会被认为是抽象的。此时如果要通过编译,就得给子trait的方法加上abstract override修饰:
        trait MyLogged extends Logged{

          abstract override def log(msg: String): Unit = {
            super.log(msg)
          }
        }
    2.4:混合使用trait的具体方法和抽象方法:
      trait Valid {

          def getName : String;
          def valid : Boolean = {
            getName == "张三";
          }
        }
        //定义一个类来实现trait接口
        class Person(val name : String) extends Valid{

          println(valid)

          def getName = name;
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            var p1 = new Person("张三");

          }
        }
    2.5:trait的构造机制:
      在Scala中,trait也是有构造方法的,也就是trait中的,不包含在任何方法中的代码。而继承了trait的类的构造机制如下所示:
        a、父类的构造函数执行。
        b、trait的构造代码执行,多个trait从坐到右依次执行。
        c、构造trait的时候会先构造父类trait,如果多个trait继承同一个父trait,则父trait只会构造一次。
        d、所有trait构造完毕以后,子类的构造函数执行。
        trait Logger {

          println("logger constructor")
        }
        //定义一个trait来实现trait接口
        trait Mylogger extends Logger{

          println("Mylogger constructor")
        }    
        //定义一个trait来实现trait接口
        trait TimeLogger extends Logger{

          println("TimeLogger constructor");
        }
        //定义一个class
        class Person{
            
          println("person construcotr")
        }
        //定义一个class继承类和trait
        class Student extends Person with Mylogger with TimeLogger{

          println("Student constructor")
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            val s1 = new Student();
          }
        }
    2.6:trait字段的初始化:
      在Scala中,trait也是有构造方法的,也就是trait中的,不包含在任何方法中的代码。而继承了trait的类的构造机制如下所示:
        a、父类的构造函数执行。
        b、trait的构造代码执行,多个trait从坐到右依次执行。
        c、构造trait的时候会先构造父类trait,如果多个trait继承同一个父trait,则父trait只会构造一次。
        d、所有trait构造完毕以后,子类的构造函数执行。
        trait Logger {

          println("logger constructor")
        }
        //定义一个trait来实现trait接口
        trait Mylogger extends Logger{

          println("Mylogger constructor")
        }    
        //定义一个trait来实现trait接口
        trait TimeLogger extends Logger{

          println("TimeLogger constructor");
        }
        //定义一个class
        class Person{
            
          println("person construcotr")
        }
        //定义一个class继承类和trait
        class Student extends Person with Mylogger with TimeLogger{

          println("Student constructor")
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            val s1 = new Student();

          }
        }
    2.7:让trait继承类:
       在Scala中,trait是没有接受参数的构造函数的,这是trait与class的唯一区别,但是如果需求就是要trait能够对field进行初始化,那么只能使用Scala中非常特殊的一种高级特性--提前定义:
        //1:第一种方式实现:
        trait Hello {

          val msg : String;
          println(msg.toString);
        }
        //定义一个class继承类和trait
        class Person extends {

          val msg : String = "init";
        }with Hello{}
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            var p1 = new Person();
          }
        }
        //2:第二种方式实现:
        trait Hello {

          lazy val msg : String = null;
          println(msg.toString);
        }
        //定义一个class继承类和trait
        class Person extends Hello{

          override lazy val msg: String = "init"
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            var p1 = new Person();
          }
        }
    2.7:让trait继承类:
        在Scala中,trait也可以继承自class,此时这个class就会成为所有继承该trait的类的父类;
        class MyUtil {

          def printMessage(msg : String) = println("msg : " + msg);
        }    
        //定义一个trait实现class
        trait Logger extends MyUtil{

          def log(msg : String) = printMessage("log :" + msg);
        }
        //定义一个class实现trait
        class Person(val name : String) extends Logger{

          def hello(): Unit ={
            log("hi , i am : " + name);
            printMessage("hi , i am : " + name);
          }
        }
        //定义一个object来测试实现的类
        object Object {

          def main(args: Array[String]): Unit = {
            val p = new Person("张三");
            p.hello()
          }
        }

待续......

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏黑泽君的专栏

java基础学习_基础语法(下)01_day05总结

============================================================================= ==...

971
来自专栏python学习指南

python正则表达式

本篇将介绍python正则表达式,更多内容请参考:【python正则表达式】 什么是正则表达式 正则表达式,又称规则表达式,通常被用来检索、替换那些符合某...

2556
来自专栏编程

对Python中的类做简要的分析

在Python中,定义类是通过class关键字,class后面紧接着是类名,即Student,类名通常是大写开头的单词,紧接着是(object),表示该类是从哪...

20210
来自专栏伦少的博客

scala 下划线使用指南

原文地址:https://my.oschina.net/joymufeng/blog/863823   作者:joymufeng

3648
来自专栏Android干货园

Kotlin中级(7)- - - Kotlin类之抽象类、重载和重写.md

abstract 其中值得注意的是:抽象可以分为抽象类、抽象函数、抽象属性。而一个抽象类和普通类的区别在于抽象类除了可以有其自己的属性、构造函数、方法等组成部...

1192
来自专栏和蔼的张星的图像处理专栏

c++ primer2 变量和基本类型。

这四种初始化方式c++11都是支持的。c++11中用花括号来初始化变量得到了全面应用。

1191
来自专栏塔奇克马敲代码

第6章 函数

2067
来自专栏python学习指南

Python爬虫(十)_正则表达式

本篇将介绍python正则表达式,更多内容请参考:【python正则表达式】 什么是正则表达式 正则表达式,又称规则表达式,通常被用来检索、替换那些符合某...

2676
来自专栏PHP在线

欢迎来到phpdaily

1.Null类型,表示空对象指针,使用typeof检测会返回object。 如果定义的变量在将来用于保存对象,最好将该变量初始化为NUll.可以体现null作为...

3297
来自专栏编程

Python面向对象5:特殊方法

特殊方法,两边带双下划线的方法。比如__init__(self,...) 、__del__(self) 、__call__(self, *args) 、__st...

19510

扫码关注云+社区

领取腾讯云代金券