Scala(五)-②-面相对象高级-静态属性和方法、特质(上)

① 伴生对象和伴生类

①-① Why

  • Scala语言是完全面相对象的,并不支持静态这个概念,也就没有静态成员(静态成员变量和静态成员方法).
  • 但是java又支持静态这个概念,有些需求需要用到静态.所以Scala就使用伴生对象这个技术来模拟静态,使得我们能够实现和静态一样的效果.

①-② How

语法和规则
  • 伴生对象必须和伴生类同名.
class Person {  // 半生类Person

}

object Person {  // 伴生对象Person.写在里面的成员可以模拟Java的static效果.

}

  • 可以在伴生对象实现apply方法.之后直接用类名(apply形参列表)就能够触发apply方法.所以一般创建对象.
object 类名 {
    def apply(形参列表) : 类名 = new 类名(形参列表)
}
Demo
伴生对象模拟静态方法
package com.sweetcs.bigdata.scala.day05.chapter08._01_accompobject

/**
  * @author sweetcs
  */
object ChildGameDemo {

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

    val child0 = new Child("lisi")
    val child1 = new Child("zhangsan")
    val child2 = new Child("wangwu")

    Child.count(child0) // 使用类能够直接调用伴生对象中的成员
    Child.count(child1)
    Child.count(child2)
    Child.showNumberOfChild()

  }
}

class Child(inName :String) {

  var name :String = inName

}
object Child {

  var numberOfChild : Int = 0

  def count(child: Child): Unit = {
    numberOfChild += 1
  }

  def showNumberOfChild(): Unit = {
    println(s"numberOfChild = $numberOfChild")
  }
}
伴生对象里apply
object ApplyDemo {
  def main(args: Array[String]): Unit = {

    val pig1 = Pig()
    val pig2 = Pig("pei qi")

    println(pig1.name)
    println(pig2.name)

  }
}

class Pig(inName :String) {

  var name :String = inName

  def this() {
    this("")
  }

}

object Pig {

  def apply(inName: String): Pig = new Pig(inName)

  def apply(): Pig = new Pig()

  
}

①-③ What

伴生对象是怎么实现让对应的伴生类能通过直接调用 伴生对象里面的成员?

如下分析,先看源代码对应的反编译的代码

源代码
/**
  * @author sweetcs
  */
object AccompObjectDemo {

  def main(args: Array[String]): Unit = {
      PersonOfAccompanyClass.name = "test"
      PersonOfAccompanyClass.show()
  }
}

// 半生类
class PersonOfAccompanyClass {

}

// 半生对象
object PersonOfAccompanyClass {
  var name :String = ""
  def show(): Unit = {
    println(s"name = ${name}")
  }
}
反编译的AccompObjectDemo和PersonOfAccompanyClass代码
public final class AccompObjectDemo$
{
  public static final  MODULE$;
  
  static
  {
    new ();
  }
  
  public void main(String[] args)
  {
    PersonOfAccompanyClass..MODULE$.name_$eq("test");
    PersonOfAccompanyClass..MODULE$.show();
  }
  
  private AccompObjectDemo$()
  {
    MODULE$ = this;
  }
}
public final class PersonOfAccompanyClass$
{
  public static final  MODULE$;
  private String name;
  
  public String name()
  {
    return this.name;
  }
  
  public void name_$eq(String x$1)
  {
    this.name = x$1;
  }
  
  public void show()
  {
    Predef..MODULE$.println(new StringContext(Predef..MODULE$.wrapRefArray((Object[])new String[] { "name = ", "" })).s(Predef..MODULE$.genericWrapArray(new Object[] { name() })));
  }
  
  private PersonOfAccompanyClass$()
  {
    MODULE$ = this;this.name = "";
  }
  
  static
  {
    new ();
  }
}

从上面代码可以看出,底层是通过以下两句实现了我们写的代码.而这里的MODULE$核心所在,它是一个静态实例,类型就是PersonOfAccompanyClass$类型.所以底层其实是创建了一个对应的类名$的类,并将伴生对象中的成员分按访问控制权限放入, 并且仅有一个类名$的类型的实例(因为MODULE$是静态的).

    PersonOfAccompanyClass$.MODULE$.name_$eq("test");
    PersonOfAccompanyClass$.MODULE$.show();

①-④ Details

  • 伴生对象必须和伴生类同名.如果只有伴生对象,其会自动生成一个对应的空的半生类,如果只有伴生类其不会自动生成伴生对象.
  • 如果半生对象中定义一个apply(apply形参列表),则可以不用new就创建对象,直接通过类名(apply形参列表)即可以触发对应的apply方法返回对象.

② 特质

学习特质之前我们先来看下Java中的接口.

Java中的接口特点

  • 在Java中, 一个类可以实现多个接口
  • 在Java中,接口之间支持多继承
  • 接口中属性是常量
  • 接口中的方法是抽象的

Java中的接口有以下缺点

  • 如果B实现了Ia接口,并且C继承自B,则C中会多出哪个被实现的方法,但是很多时候我们并不想要.而Scala可以通过mixin动态混入技术实现该功能
  • 无法在接口中实现方法.(在Java8之前), Scala的trait可以实现该功能.

②-① Why

  • trait拥有接口和抽象类的特点.学习trait,是因为trait拥有java中接口的特点抽象类的特点,其可以很方便的解决以上接口存在的问题.所以我们可以用trait来替代接口.
  • trait可以动态扩展类的功能而不通过继承.trait可以让一个类(包含抽象类),能够不通过继承trait就可以动态的扩展trait里的非抽象方案,增强这个类的功能.

②-② How

规则和语法
  • 如果使用传统方法(extends),则子类依旧会继承父类实现的接口方法
  • 如果使用mixin(动态混入),则子类不会继承父类实现的接口方法
获取数据库连接(trait的接口特点)-传统方法
object TraitDemo02 {

  def main(args: Array[String]): Unit = {
    val mySQLDriver = new MySQLDriver
    mySQLDriver.getConnection()

    val oracleDriver = new OracleDriver
    oracleDriver.getConnection()

  }
}

trait DBDriver{
  def getConnection()
}

class A {}
class B extends A {}
class MySQLDriver extends A with DBDriver {
  override def getConnection(): Unit = {
      println("连接MySQL成功")
  }
}

class D {}
class OracleDriver extends D with DBDriver {
  override def getConnection(): Unit = {
    println("连接Oracle成功")
  }
}
class F extends D {}

动态混入(mixin)
/**
  * @author sweetcs
  */
object TraitDemo04Mixin {
  def main(args: Array[String]): Unit = {

    val oracle = new OracleWithMixin with DBDriver02
    oracle.insert()
  }
}

class OracleWithMixin {

}

trait DBDriver02 {

  def insert(): Unit = {
    println("正在插入数据到数据库中")
  }
}

②-③ What

  • 特质中如果还有普通方法,在底层是如何实现这个普通方法的调用呢?其在底层是放在哪个类里?
反编译后的代码
  • Animal.class
public abstract interface Animal
{
  public abstract void sayHi();
  
  public abstract void eat();
}
  • Animal$class.class
public abstract class Animal$class
{
  public static void eat(Animal $this)
  {
    Predef$.MODULE$.println("I am eating~~~");
  }
  
  public static void $init$(Animal $this) {}
}
  • sheep.class
public class Sheep
  implements Animal
{
  public void eat()
  {
    Animal$class.eat(this);
  }
  
  public Sheep()
  {
    Animal$class.$init$(this);
  }
  
  public void sayHi()
  {
    Predef$.MODULE$.println("hello human");
  }
}

可以看到trait中的普通方法其实真正的实现是在特质名$class这个类里,并且其将普通方法转换成一个公共的静态方法.Sheep中调用普通方法,本质是是去调用特质名$class.普通方法名()

  • Scala底层是如何实现Mixin动态混入技术的,即如何做到不继承trait而又能用其功能?如何做到使用动态混入创建出来的类不会影响其子类?

如下代码是动态混入TraitDemo04Mixin中的反编译代码

public final class TraitDemo04Mixin$
{
  public static final  MODULE$;
  
  static
  {
    new ();
  }
  
  public void main(String[] args)
  {
    OracleWithMixin oracle = new OracleWithMixin()
    {
      public void insert()
      {
        DBDriver02$class.insert(this);
      }
    };
    ((DBDriver02)oracle).insert();
  }
  
  private TraitDemo04Mixin$()
  {
    MODULE$ = this;
  }
}
  • DBDriver02$class
public abstract class DBDriver02$class
{
  public static void insert(DBDriver02 $this)
  {
    Predef$.MODULE$.println("正在插入数据到数据库");
  }
  
  public static void $init$(DBDriver02 $this) {}
}
  • OracleWithMixin
public class OracleWithMixin {}

分析:

  • 我们知道trait中的普通方法最终在底层是在trait名$calss这个抽象类中,并且会将其变成一个静态的方法.这里对应的就是DBDriver02$class中的insert公共静态方法.
  • TraitDemo04Mixin$是程序真正的main入口,这里可以看到mixin的底层技术本质其实就是new了一个匿名子类,并且实现了对应的insert接口,并且在接口中调用的是DBDriver02$class.insert方法.
  • 因为使用的是new 匿名子类,所以这种方式并不是继承trait(如上OracleWithMixin并没有继承DBDriver02).自然其子类也就不会有对应的insert方法.

②-④ Details

  • scala的trait即特质需要首字母大写
  • scala中java中的接口都可以当做trait使用.
  • scala中特质的使用,用extends关键字,如果有多个就使用extends 特质1 with 特质2 with 特质3,如果有父类要继承,那么extends后面发父类.
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容

  • Scala与Java的关系 Scala与Java的关系是非常紧密的!! 因为Scala是基于Java虚拟机,也就是...
    灯火gg阅读 3,437评论 1 24
  • 面向对象编程之类 定义一个简单的类 // 定义类,包含field以及方法 // 创建类的对象,并调用其方法 get...
    义焃阅读 780评论 0 2
  • 这篇讲义只讲scala的简单使用,目的是使各位新来的同事能够首先看懂程序,因为 scala 有的语法对于之前使用习...
    MrRobot阅读 2,912评论 0 10
  • Databricks Scala 编程风格指南 本文转载自 https://github.com/databric...
    雪轩辕阅读 1,172评论 0 1
  • 亲爱的小胖,今天妈妈的死党少燕阿姨把妈妈馋得不行的肉丸和肉饼寄来了,可是,因为阿姨没有经验,加上天气热,结果...
    zhuo舍舍阅读 153评论 0 0