每日一篇大数据每日一篇Scala程序员技术栈

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

2018-11-16  本文已影响3人  sixleaves

① 伴生对象和伴生类

①-① Why

①-② How

语法和规则
class Person {  // 半生类Person

}

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

}

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

② 特质

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

Java中的接口特点

Java中的接口有以下缺点

②-① Why

②-② How

规则和语法
获取数据库连接(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

反编译后的代码
public abstract interface Animal
{
  public abstract void sayHi();
  
  public abstract void eat();
}
public abstract class Animal$class
{
  public static void eat(Animal $this)
  {
    Predef$.MODULE$.println("I am eating~~~");
  }
  
  public static void $init$(Animal $this) {}
}
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.普通方法名()

如下代码是动态混入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;
  }
}
public abstract class DBDriver02$class
{
  public static void insert(DBDriver02 $this)
  {
    Predef$.MODULE$.println("正在插入数据到数据库");
  }
  
  public static void $init$(DBDriver02 $this) {}
}
public class OracleWithMixin {}

分析:

②-④ Details

上一篇 下一篇

猜你喜欢

热点阅读