0


Scala之面向对象(2)

单例对象(伴生对象):

Scala语言是完全面向对象的语言,所以并没有静态的操作(即在Scala中没有静态的概 念)。但是为了能够和Java语言交互(因为Java中有静态概念),就产生了一种特殊的对象 来模拟类对象,该对象为单例对象。若单例对象名与类名一致,则称该单例对象这个类的伴 生对象,这个类的所有“静态”内容都可以放置在它的伴生对象中声明。

创建对象常用的两种方法:

(1)使用构造器构造(new)

    val person0 = new Person09()

(2)使用静态方法

(在Scala中没有静态方法与之等价的是object---伴生对象)

    val person01: Person09 = Person09.getPerson()

伴生对象的定义:

object Person09{
  def getPerson():Person09=new Person09
}
      方法2主要是在将构造器私有化的时候是无法new的  这时就需要使用伴生对象构造           
class Person09 private{
  val name :String="张三"
}

apply方法:

在Scala中引入了一个apply()方法模板 来替代上述的伴生对象的构造方法 在伴生对象中创建了方法,直接在main方法中调用apply方法即可

apply()方法在Scala中是做出了改进 在函数进行调用的时候可以不写方法名,直接进行调用

apply方法的调用:

    val person0: Person09 = Person09()

apply方法也是可以进行任意自定义的 没必要一定跟模板一样创建一个对象

特点就是可以不写方法名

直接Person09() 就能调用apply方法

系统在集合当中使用非常广泛,调用对象的apply方法

下边集合 Array就是一个对象 使用.var 自动生成对象 其实是Array集合

    val ints: Array[Int] = Array(1, 2, 3, 4)  

单例模式:

单例模式是确保一个类只有一个实例,即不能随便被创建对象,所以构造方法要使用private进行私有化,不能被外界进行实例化,这个实例是当前类的类成员变量,即类静态成员变量,这个静态变量需要提供给外界,所以需要提供静态方法向外界提供当前实例

单例模式分为两种:

1.懒汉式

占用内存少(减少创建对象的数量),线程不安全(在某个时间段 都进入if判断,都为真,都会赋值产生变量 就会产生线程安全---只会出现在第一次初始化的时候)

  class Singleton09 private{
  }
  object Singleton09{        //相当于静态方法 在静态方法中创建对象 便于外边的访问
    //先声明属性     先声明为null  后边获取的时候再赋值
    private var singleton:Singleton09=null;
    def apply(): Singleton09 = {
              if (singleton==null)
                {
                  singleton = new Singleton09
                }
                  singleton
    }
  }

2.饿汉式:(推荐)

饿汉式没有线程安全问题,写起来简单,占用内存多(提前创建) 它是在伴生对象中先进行创建对象 在jvm中进行类加载,这样就避免了线程安全问题

object Singleton09 {
  private var singleton: Singleton09 = new Singleton09;
  def apply(): Singleton09 = singleton
}

判断单例模式:

  def main(args: Array[String]): Unit = {
    //判断单例模式是否正确
    val singleton:Singleton09=Singleton09();
    val singleton1:Singleton09=Singleton09();
    println(singleton)
    println(singleton1)
  }

可以看到地址值相同

特质(Trait):

Scala 语言中,采用特质 trait(特征)来代替接口的概念,也就是说,多个类具有相同

的特质(特征)时,就可以将这个特质(特征)独立出来,采用关键字 trait 声明。

Scala 中的 trait 中即可以有抽象属性和方法,也可以有具体的属性和方法,**一个类可 **

以混入(mixin)多个特质。这种感觉类似于 Java 中的抽象类(现在就是接口--->现在的接口是可以有具体的属性和方法)。

一个类具有某种特质(特征),就意味着这个类满足了这个特质(特征)的所有要素,

所以在使用时,也采用了 extends 关键字,如果有多个特质或存在父类,那么需要采用 with

关键字连接。

基本语法:

没有父类:class 类名 extends特质 1 with特质 2 with特质 3 …

有父类:class 类名 extends父类with特质 1 with特质 2 **with **特质 3…

说明

(1)类和特质的关系:使用继承的关系。

(2)当一个类去继承特质时,第一个连接词是 extends,后面是 with。

(3)如果一个类在同时继承特质和父类时,应当把父类写在 extends 后

(4)一个类可以混入多个特质 


(5)所有的Java接口都可以当作Scala的特质进行使用 
  class Person11 extends Young11 with old11 with java.io.Serializable{}

(6)动态混入:可灵活的扩展类的功能

    (6.1)动态混入:创建对象时混入 trait,而无需使类混入该 trait 

    (6.2)如果混入的 trait 中有未实现的方法,则需要实现

    (6.3)如果特质中有抽象的方法和属性也是需要重写的

(7)特质冲突:如果继承的父类和特质中有相同的名称的具体数据(非抽象属性--抽象的可 以重写)会报错冲突

如果属性是val 是可以在类中创建一个新的对象进行赋值改变

如果属性是var 是不可以的 报错误

特质叠加:

    由于一个类可以混入(mixin)多个 trait,且 trait 中可以有具体的属性和方法,若混入 

的特质中具有相同的方法(方法名,参数列表,返回值均相同),必然会出现继承冲突问题。

冲突分为以下两种:

(1)第一种,一个类(Sub)混入的两个 trait(TraitA,TraitB)中具有相同的具体方法,且

两个 trait 之间没有任何关系,解决这类冲突问题,直接在类(Sub)中重写冲突方法。

(2)第二种,一个类(Sub)混入的两个 trait(TraitA,TraitB)中具有相同的具体方法,且

两个 trait 继承自相同的 trait(TraitC),及所谓的“钻石问题”,解决这类冲突问题,Scala

采用了特质叠加的策略。

  钻石问题测试: 
package chapter04
object Test11_TraitMinxin2 {
  def main(args: Array[String]): Unit = {
    val person1 = new Person12
    println(person1.Info())
  }
}
trait Age12{
val age:Int
  def Info()="age "
}
//下边两个特质均继承自Age12
trait Young12 extends Age12{
  override val age: Int = 12
  override def Info(): String = super.Info()+"Young "
}
trait Old12 extends  Age12{
  override val age: Int = 13
  override def Info(): String = super.Info()+"Old12 "
}
//钻石效应
class Person12 extends Young12 with  Old12 {
  override def Info(): String = super.Info()+"Person12 "
}

测试结果:

上述结果说明,在产生钻石问题的时候会按照继承关系进行一个排序来进行调用

当我们如果只想调用某一个类的时候方法的时候,可以对继承的类进行指定

对Person12类进行了修改:

class Person12 extends Young12 with  Old12 {
  override def Info(): String = super[Young12].Info()+"Person12 "
}

执行结果:

特质和抽象类的区别:

1.优先使用特质。一个类扩展(继承)多个特质是很方便的,但却只能扩展一个抽象类。

2.如果你需要构造函数参数,使用抽象类。因为抽象类可以定义带参数的构造函数,

而特质不行(只有无参构造)。

   面向对象的扩展: 
 

   (1)类型检查和转换 
 

   在使用多态的时候,无法调用子类独有的方法,所以需要进行强制类型转换(在Scala中是使用方法进行强制类型转换) 
 

   代码样例: 
 
package chapter04
object Test_14_Rxtends {
  def main(args: Array[String]): Unit = {
//多态
    var son:father14 =new son14
    val son1: son14 = son.asInstanceOf[son14]
    son1.sayHi()
  }
}
class father14{
}
class son14 extends father14 {
def sayHi(): Unit ={
  println("hi son")
}
}
标签: scala java 开发语言

本文转载自: https://blog.csdn.net/m0_61469860/article/details/130116663
版权归原作者 小唐同学爱学习 所有, 如有侵权,请联系我们删除。

“Scala之面向对象(2)”的评论:

还没有评论