文章目录

  • 一,介绍
  • 二,类Class
    • 2.1 类定义
    • 2.2 构造器
      • 1.基本语法
      • 2.构造器参数
    • 2.3 私有成员和Getter/Setter语法
  • 三,单例对象Object
  • 四,伴生对象和伴生类介绍
  • 五,伴生类和伴生对象反编译后结果
  • 六,总结

一,介绍

Scala是一个完全面向对象的语言,故Scala去掉了Java中非面向对象的元素,如static关键字,void类型,为了能够调用静态语法(模拟静态语法),采用伴生对象单例的方式

  1. Scala源码中包含了main方法,在编译后自动形成了public static void main

  2. scala在编译源码时,会生成两个字节码文件,静态main方法执行另一个字节码文件中的成员main方法

  3. Scala是完全面向对象的语言,那么没有静态的方法,只能通过模拟生成静态方法

  4. scala无static关键字,由object实现类似静态方法的功能(类名.方法名)

  5. class关键字和Java中的class关键字作用相同,用来定义一个类

  6. 编译时将当前类生成一个特殊的类==>Scala01_HelloWorld$,然后创建对象来调用这个对象的main方法

  7. 一般情况下,将加$的类的对象,称之为伴生对象

  8. 伴生对象中的内容,都可以通过类名访问,来模拟java中的静态语法

  9. 伴生对象的语法规则:使用object声明[加上就一定能够使用类名来访问]

二,类Class

Scala中的类是用于创建对象的蓝图,其中包含了方法、常量、变量、类型、对象、特质、类,这些统称为成员。

object中定义的均为静态的,class中均是非静态的。

2.1 类定义

一个最简的类的定义就是关键字class+标识符,类名首字母应大写。

class User
val user1 = new User

关键字new被用于创建类的实例。User由于没有定义任何构造器,因而只有一个不带任何参数的默认构造器。然而,你通常需要一个构造器和类体。下面是类定义的一个例子:

class Point(var x: Int, var y: Int) {def move(dx: Int, dy: Int): Unit = {x = x + dxy = y + dy}override def toString: String =s"($x, $y)"
}val point1 = new Point(2, 3)
point1.x  // 2
println(point1)  // prints (2, 3)

Point类有4个成员:变量xy,方法movetoString。与许多其他语言不同,主构造方法在类的签名中(var x: Int, var y: Int)move方法带有2个参数,返回无任何意义的Unit类型值()。这一点与Java这类语言中的void相当。另外,toString方法不带任何参数但是返回一个String值。因为toString覆盖了AnyRef中的toString方法,所以用了override关键字标记。

2.2 构造器

和 Java 一样,Scala 构造对象也需要调用构造方法,并且可以有任意多个构造方法。

1.基本语法

Scala 类的构造器包括:主构造器和辅助构造器

class  类名(形参列表) {    // 主构造器// 类 体def    this(形参列表) {    // 辅助构造器}def    this(形参列表) {    // 辅助构造器可以有多个...}
}

说明:

  1. 辅助构造器,函数的名称 this,可以有多个,编译器通过参数的个数及类型来区分。
  2. 辅助构造方法不能直接构建对象,必须直接或者间接调用主构造方法。
  3. 构造器调用其他另外的构造器,要求被调用构造器必须提前声明。

如果主构造器无参数,小括号可省略,构建对象时调用的构造方法的小括号也可以省略。

//(1)如果主构造器无参数,小括号可省略
//class Person (){
class Person {var name: String = _ var age: Int = _def this(age: Int) {this() this.age = ageprintln("辅助构造器")}def this(age: Int, name: String) { this(age)this.name = name}println("主构造器")
}object Person {def main(args: Array[String]): Unit = { val person2 = new Person(18)}
}

说明实例

// TODO Scala中的构造方法分为两类: 主构造方法(一个) & 辅助构造方法(多个)
// Scala构建对象可以通过辅助构造方法创建,但是 TODO 必须要调用主构造方法
// Scala是完全面向函数的语言,所以类也是函数
// 类是函数,可以使用小括号作为函数的参数列表
// 类所代表的函数其实就是这个类的构造方法
// 默认情况下,scala也是给类提供无参构造方法,所以小括号可以省略
// 在类的后面声明的构造方法就是主构造方法
// 在主构造方法中声明的构造方法就是辅助构造方法
// 辅助构造器的声明不能和主构造器的声明一致,会发生错误(构造器名重复)// 如果想让主构造器变成私有的,可以在()之前加上private,这样用户不能直接通过主构造器来构造对象了(可以通过伴生对象访问)class User09 private(s: String) { // 主构造方法,如果主构造方法无参数,小括号可以省略// 类体 & 构造方法体println("主构造方法")println(s)// 辅助构造方法无论是直接或者间接,最终都一定要调用主构造器,执行主构造器的逻辑// 而且需要放在辅助构造器的第一行def this(s: String, ss: String) { // 辅助构造方法this(s) // 直接调用主构造器println("辅助构造方法2")}// 声明辅助构造方法,方法名为this// TODO 构造方法调用其他的构造方法时,应该保证被调用的构造方法已经声明过def this() { // 辅助构造方法可以有多个this("辅助构造方法1", "xxxx") // 间接调用主构造器,因为def this(s: String, ss: String)中调用了主构造器}}

2.构造器参数

Scala 类的主构造器函数的形参包括三种类型:未用任何修饰、var 修饰、val 修饰

  1. 未用任何修饰符修饰,这个参数就是一个局部变量
  2. var 修饰参数,作为类的成员属性使用,可以修改
  3. val 修饰参数,作为类只读属性使用,不能修改

案例实操

class Person(name: String, var age: Int, val sex: String) {}
object Test {def main(args: Array[String]): Unit = {var person = new Person("bobo", 18, "男")// (1)未用任何修饰符修饰,这个参数就是一个局部变量// printf(person.name)// (2)var 修饰参数,作为类的成员属性使用,可以修改person.age = 19println(person.age)// (3)val 修饰参数,作为类的只读属性使用,不能修改// person.sex = "女"println(person.sex)}
}

构造器可以通过提供一个默认值来拥有可选参数:

class Point(var x: Int = 0, var y: Int = 0)val origin = new Point  // x and y are both set to 0
val point1 = new Point(1)
println(point1.x)  // prints 1

在这个版本的Point类中,xy拥有默认值0所以没有必传参数。然而,因为构造器是从左往右读取参数,所以如果仅仅要传个y的值,你需要带名传参

class Point(var x: Int = 0, var y: Int = 0)
val point2 = new Point(y=2)
println(point2.y)  // prints 2

2.3 私有成员和Getter/Setter语法

成员默认是公有(public)的。使用private访问修饰符可以在类外部隐藏它们。

class Point {private var _x = 0private var _y = 0private val bound = 100def x = _xdef x_= (newValue: Int): Unit = {if (newValue < bound) _x = newValue else printWarning}// 类似于java中的get方法def y = _y// 类似于java中的set方法def y_= (newValue: Int): Unit = {if (newValue < bound) _y = newValue else printWarning}private def printWarning = println("WARNING: Out of bounds")
}val point1 = new Point
point1.x = 99
point1.y = 101 // prints the warning

在这个版本的Point类中,数据存在私有变量_x_y中。def xdef y方法用于访问私有数据。def x_=def y_=是为了验证和给_x_y赋值。

注意下对于setter方法的特殊语法:这个方法在getter方法的后面加上_=,后面跟着参数。

主构造方法中带有valvar的参数是公有的。然而由于val是不可变的,所以不能像下面这样去使用。

class Point(val x: Int, val y: Int)
val point = new Point(1, 2)
point.x = 3  // <-- does not compile

不带valvar的参数是私有的,仅在类中可见。

class Point(x: Int, y: Int)
val point = new Point(1, 2)
point.x  // <-- does not compile

三,单例对象Object

object是类的单例对象,开发人员无需用new关键字实例化。如果对象的名称和类名相同,这个对象就是伴生对象

Scala object相当于java中的单例

object中定义的均为静态的,class中均是非静态的。

静态的可以直接用Object名.属性/方法来调用

Object就相当于一个静态的工具类,main方法要写在object中。

object中不可以传参,当创建一个object时,如果传入参数,那么会自动寻找object中的相应参数个数的apply方法。

apply可以写多个 保证参数不同即可

object ClassAndObj {// object中定义的属性都是静态的val name = "wangwu"// 当给对象传入参数时,默认会调用apply方法def apply(i: Int) = {println("Score is " + i)}// apply可以重写def apply(i: Int, s: String) = {println("name is " + s + ",Score is " + i)}def main(args: Array[String]): Unit = {// 对象不可以传参,当看到代码中出现为对象传参的情况,就说明代码中肯定会有一个apply方法ClassAndObj(30)ClassAndObj(30,"zhangsan")}
}

一个单例对象是就是一个值。单例对象的定义方式很像类,但是使用的关键字是object

object Box

下面例子中的单例对象包含一个方法:

package logging
object Logger {def info(message: String): Unit = println(s"INFO: $message")
}

方法info可以在程序中的任何地方被引用。像这样创建功能性方法是单例对象的一种常见用法。

下面让我们来看看如何在另外一个包中使用info方法:

import logging.Logger.infoclass Project(name: String, daysToComplete: Int)class Test {val project1 = new Project("TPS Reports", 1)val project2 = new Project("Website redesign", 5)info("Created projects")  // Prints "INFO: Created projects"
}

因为import语句import logging.Logger.info,方法info在此处是可见的。

import语句要求被导入的标识具有一个“稳定路径”,一个单例对象由于全局唯一,所以具有稳定路径。

注意:如果一个object没定义在顶层而是定义在另一个类或者单例对象中,那么这个单例对象和其他类普通成员一样是“路径相关的(path-dependent)”。这意味着有两种行为,class Milkclass OrangeJuice,一个类成员object NutritionInfo“依赖”于包装它的实例,要么是牛奶要么是橙汁。milk.NutritionInfo则完全不同于oj.NutritionInfo

四,伴生对象和伴生类介绍

在同一个文件中当一个单例对象和某个类同名时,这个单例对象称为伴生对象。 同理,这个类被称为是这个单例对象的伴生类。类和它的伴生对象可以互相访问其私有成员。使用伴生对象来定义那些在伴生类中不依赖于实例化对象而存在的成员变量或者方法。

scala反编译之后其实就是把Object中的静态的内容全都揉进class中

import scala.math._case class Circle(radius: Double) {import Circle._def area: Double = calculateArea(radius)
}object Circle {private def calculateArea(radius: Double): Double = Pi * pow(radius, 2.0)
}val circle1 = Circle(5.0)circle1.area

这里的class Circle有一个成员area是和具体的实例化对象相关的,单例对象object Circle包含一个方法calculateArea,它在每一个实例化对象中都是可见的。

伴生对象也可以包含工厂方法:

class Email(val username: String, val domainName: String)object Email {def fromString(emailString: String): Option[Email] = {emailString.split('@') match {case Array(a, b) => Some(new Email(a, b))case _ => None}}
}val scalaCenterEmail = Email.fromString("scala.center@epfl.ch")
scalaCenterEmail match {case Some(email) => println(s"""Registered an email|Username: ${email.username}|Domain name: ${email.domainName}""")case None => println("Error: could not parse email")
}

伴生对象object Email包含有一个工厂方法fromString用来根据一个String创建Email实例。在这里我们返回的是Option[Email]以防有语法分析错误。

注意:伴生类和它的伴生对象必须定义在同一个源文件里。如果需要在 REPL 里定义类和其伴生对象,需要将它们定义在同一行或者进入 :paste 模式。

五,伴生类和伴生对象反编译后结果

scala反编译之后其实就是把Object中的静态的内容全都揉进class中。这个class就是这个Object的伴生类,这个Object就是这个class的伴生对象。这样就可以互相调用他们之间的私有的属性和私有的方法。

scala代码

package com.cw/*** @author 陈小哥cw* @date 2021/3/30 13:52*/
class Person(name: String, var age: Int, val sex: String) {var phone: Long = _// 类体 & 构造方法体println(Person.objAge)println(Person.objName)println(Person.getName())def this(name: String, age: Int, sex: String, phone: Long) {this(name, age, sex)this.phone = phone}}object Person {var objName: String = "Obj"var objAge = 20def main(args: Array[String]): Unit = {var person = new Person("Mike", 18, "男")// 未用任何修饰符修饰,这个参数就是一个局部变量// println(person.name)// 无法访问// var 修饰参数,作为类的成员属性使用,可以修改println(person.age)person.age = 20println(person.age)// val 修饰参数,作为类的只读属性使用,不能修改// person.sex = "女" //无法修改println(person.sex)}def getName(): String = {objName}
}

反编译为java代码后

//decompiled from Person$.class
package com.cw;import scala.Predef.;
import scala.runtime.BoxesRunTime;public final class Person$ {public static Person$ MODULE$;private String objName;private int objAge;static {new Person$();}public String objName() {return this.objName;}public void objName_$eq(final String x$1) {this.objName = x$1;}public int objAge() {return this.objAge;}public void objAge_$eq(final int x$1) {this.objAge = x$1;}public void main(final String[] args) {Person person = new Person("Mike", 18, "男");.MODULE$.println(BoxesRunTime.boxToInteger(person.age()));person.age_$eq(20);.MODULE$.println(BoxesRunTime.boxToInteger(person.age()));.MODULE$.println(person.sex());}public String getName() {return this.objName();}private Person$() {MODULE$ = this;this.objName = "Obj";this.objAge = 20;}
}//decompiled from Person.class
package com.cw;import scala.Predef.;
import scala.reflect.ScalaSignature;
import scala.runtime.BoxesRunTime;@ScalaSignature(bytes = "\u0006\u000114Aa\u0006\r\u0001;!AA\u0005\u0001B\u0001B\u0003%Q\u0005\u0003\u00051\u0001\t\u0005\r\u0011\"\u00012\u0011!)\u0004A!a\u0001\n\u00031\u0004\u0002\u0003\u001f\u0001\u0005\u0003\u0005\u000b\u0015\u0002\u001a\t\u0011u\u0002!Q1A\u0005\u0002yB\u0001b\u0010\u0001\u0003\u0002\u0003\u0006I!\n\u0005\u0006\u0001\u0002!\t!\u0011\u0005\n\u000f\u0002\u0001\r\u00111A\u0005\u0002!C\u0011\u0002\u0014\u0001A\u0002\u0003\u0007I\u0011A'\t\u0013=\u0003\u0001\u0019!A!B\u0013I\u0005\"\u0002!\u0001\t\u0003\u0001v!B+\u0019\u0011\u00031f!B\f\u0019\u0011\u00039\u0006\"\u0002!\u000e\t\u0003A\u0006bB-\u000e\u0001\u0004%\tA\u0010\u0005\b56\u0001\r\u0011\"\u0001\\\u0011\u0019iV\u0002)Q\u0005K!9a,\u0004a\u0001\n\u0003\t\u0004bB0\u000e\u0001\u0004%\t\u0001\u0019\u0005\u0007E6\u0001\u000b\u0015\u0002\u001a\t\u000b\rlA\u0011\u00013\t\u000b)lA\u0011A6\u0003\rA+'o]8o\u0015\tI\"$\u0001\u0002do*\t1$A\u0002d_6\u001c\u0001a\u0005\u0002\u0001=A\u0011qDI\u0007\u0002A)\t\u0011%A\u0003tG\u0006d\u0017-\u0003\u0002$A\t1\u0011I\\=SK\u001a\fAA\\1nKB\u0011a%\f\b\u0003O-\u0002\"\u0001\u000b\u0011\u000e\u0003%R!A\u000b\u000f\u0002\rq\u0012xn\u001c;?\u0013\ta\u0003%\u0001\u0004Qe\u0016$WMZ\u0005\u0003]=\u0012aa\u0015;sS:<'B\u0001\u0017!\u0003\r\tw-Z\u000b\u0002eA\u0011qdM\u0005\u0003i\u0001\u00121!\u00138u\u0003\u001d\tw-Z0%KF$\"a\u000e\u001e\u0011\u0005}A\u0014BA\u001d!\u0005\u0011)f.\u001b;\t\u000fm\u001a\u0011\u0011!a\u0001e\u0005\u0019\u0001\u0010J\u0019\u0002\t\u0005<W\rI\u0001\u0004g\u0016DX#A\u0013\u0002\tM,\u0007\u0010I\u0001\u0007y%t\u0017\u000e\u001e \u0015\t\t#UI\u0012\t\u0003\u0007\u0002i\u0011\u0001\u0007\u0005\u0006I\u001d\u0001\r!\n\u0005\u0006a\u001d\u0001\rA\r\u0005\u0006{\u001d\u0001\r!J\u0001\u0006a\"|g.Z\u000b\u0002\u0013B\u0011qDS\u0005\u0003\u0017\u0002\u0012A\u0001T8oO\u0006I\u0001\u000f[8oK~#S-\u001d\u000b\u0003o9CqaO\u0005\u0002\u0002\u0003\u0007\u0011*\u0001\u0004qQ>tW\r\t\u000b\u0006\u0005F\u00136\u000b\u0016\u0005\u0006I-\u0001\r!\n\u0005\u0006a-\u0001\rA\r\u0005\u0006{-\u0001\r!\n\u0005\u0006\u000f.\u0001\r!S\u0001\u0007!\u0016\u00148o\u001c8\u0011\u0005\rk1CA\u0007\u001f)\u00051\u0016aB8cU:\u000bW.Z\u0001\f_\nTg*Y7f?\u0012*\u0017\u000f\u0006\u000289\"91\bEA\u0001\u0002\u0004)\u0013\u0001C8cU:\u000bW.\u001a\u0011\u0002\r=\u0014'.Q4f\u0003)y'M[!hK~#S-\u001d\u000b\u0003o\u0005DqaO\n\u0002\u0002\u0003\u0007!'A\u0004pE*\fu-\u001a\u0011\u0002\t5\f\u0017N\u001c\u000b\u0003o\u0015DQAZ\u000bA\u0002\u001d\fA!\u0019:hgB\u0019q\u0004[\u0013\n\u0005%\u0004#!B!se\u0006L\u0018aB4fi:\u000bW.\u001a\u000b\u0002K\u0001"
)
public class Person {private int age;private final String sex;private long phone;public static String getName() {return Person$.MODULE$.getName();}public static void main(final String[] args) {Person$.MODULE$.main(var0);}public static void objAge_$eq(final int x$1) {Person$.MODULE$.objAge_$eq(var0);}public static int objAge() {return Person$.MODULE$.objAge();}public static void objName_$eq(final String x$1) {Person$.MODULE$.objName_$eq(var0);}public static String objName() {return Person$.MODULE$.objName();}public int age() {return this.age;}public void age_$eq(final int x$1) {this.age = x$1;}public String sex() {return this.sex;}public long phone() {return this.phone;}public void phone_$eq(final long x$1) {this.phone = x$1;}public Person(final String name, final int age, final String sex) {this.age = age;this.sex = sex;super();.MODULE$.println(BoxesRunTime.boxToInteger(Person$.MODULE$.objAge()));.MODULE$.println(Person$.MODULE$.objName());.MODULE$.println(Person$.MODULE$.getName());}public Person(final String name, final int age, final String sex, final long phone) {this(name, age, sex);this.phone_$eq(phone);}
}

单例模式饿汉式:在类初始化时初始化一个对象,并让MODULE$引用

MODULE$即静态对象,所谓的静态方法与静态变量只不过是MODULE$的成员方法与属性

从底层原理看,伴生对象实现静态特性是依赖于public static final MODULE$实现的。

可以看到

  1. val 和 private val 修饰的成员变量被编译成 private final, 并提供getter
  2. var 和 private var 修饰的成员变量被编译成 private, 并提供getter和setter
  3. 没有修饰词修饰的和private[this]不会被编译成成员变量,但构造函数会带上

六,总结

  1. 在 Java 中static成员对应于Scala中的伴生对象的普通成员。
  2. Java代码中调用伴生对象时,伴生对象的成员会被定义成伴生类中的static成员。这称为静态转发。这种行为发生在当你自己没有定义一个伴生类时。
  3. scala 中的object是单例对象,相当于java中的工具类,可以看成是定义静态的方法的类。
  4. object不可以传参,当看到代码中出现为对象传参的情况,就说明代码中肯定会有一个apply方法。
  5. 当创建一个object时,如果传入参数,那么会自动寻找object中的相应参数个数的apply方法。
  6. class可以传参,object不可以传参。 对象(object)要传参,使用apply方法。
  7. scala中的class类默认可以传参数,传参一定要指定类型, 有了参数就有了默认的构造
  8. class重写构造函数的时候,必须要调用默认的构造函数。
  9. 类中重写构造时,构造中第一行必须先调用默认的构造def this(){}
  10. class 类属性自带getter ,setter方法。但是要set的话,属性必须是变量。Var声明。
  11. Scala中当new class时,类中除了方法不执行[这里的方法不包括构造方法],其它都执行
  12. 使用object时,不用new,使用class时要new ,并且new的时候,class中除了方法(不包括构造方法)不执行,其他都执行。这些执行的内容就相当于是java 中的构造方法。
  13. 如果在同一个文件中,object对象和class类的名称相同,则这个对象就是这个类的伴生对象,这个类就是这个对象的伴生类。可以互相访问私有变量。
  14. 伴生类和伴生对象:因为scala中剥离了静态和非静态的属性和方法,需要解决私有属性和方法的调用问题
  15. Scala中,将非静态的全部定义在class中,静态的全部定义在Object中。
  16. scala反编译之后其实就是把Object中的静态的内容全都揉进class中。这个class就是这个Object的伴生类,这个Object就是这个class的伴生对象。这样就可以互相调用他们之间的私有的属性和私有的方法。
  17. class和Object中的属性默认是public的,可以使用private来修饰私有
  18. 伴生类和它的伴生对象必须定义在同一个源文件里
package com.cw.scala/*** Scala:* 1.Scala object相当于java中的单例,object中定义的全是静态的.相当于java中的工具类* 2.Scala中定义变量使用var,定义常量使用val,变量可变,常量不可变* 3.Scala中每行后面都会有分号自动推断机制,不用显式写出";"* 4.Scala 变量和常量的类型可以省略不写,会自动推断* 5.建议在Scala中命名使用驼峰命名法* 6.Scala类中可以传参,传参一定要指定类型,有了参数就有了默认的构造* 7.类中的属性默认有getter和setter方法,但是要set的话,属性必须是变量。Var声明。* 8.类中重写构造时,构造中第一行必须先调用默认的构造,def this(){}* 9,Scala中当new class时,类中除了方法不执行[这里的方法不包括构造方法],其它都执行* 10.在同一个scala文件中,class名称和Object名称一样时,这个类叫做这个对象的伴生类,这个对象叫做这个类的伴生对象,他们之间可以互相访问私有变量* 11.class和Object中的属性默认是public的,可以使用private来修饰私有* 12.Object不可以传参 class可以传参 对象(object)要传参,使用apply方法.*///在xname前不使用val和var修饰,相当于xname是私有的
// 使用val和var修饰后可以被其它类调用p.xname  如 class Person(var xname: String, xage: Int)
class Person(xname: String, xage: Int) {//class中定义的属性都是非静态的val name = xnamevar age = xagevar gender = 'M'//  println("****** Person Class ***********")//重写构造: 构造方法{}前不能加=,因为它没有返回值def this(yname: String, yage: Int, ygender: Char) {this(yname, yage)this.gender = ygender}def sayName() = {println("hello world..." + ClassAndObj.name)//object中的属性和方法都是静态的ClassAndObj.show()}//  println("======== Person Class ======")
}object ClassAndObj {//  println("######## ClassAndObj Object #######")//object中定义的属性都是静态的val name = "wangwu"//当给对象传入参数时,默认会调用apply方法def apply(i: Int) = {println("Score is " + i)}//apply可以重写def apply(i: Int, s: String) = {println("name is " + s + ",Score is " + i)}def main(args: Array[String]): Unit = {//对象不可以传参,当看到代码中出现为对象传参的情况,就说明代码中肯定会有一个apply方法ClassAndObj(30)ClassAndObj(30,"zhangsan")//    val p = new Person("zhangsan", 20);//    print(p.name)//    println(p.gender)//    //有了参数就有了默认的构造//    val p1 = new Person("diaochan", 18, 'F')//    println(p1.gender)//    p.age = 200 //使用默认setter方法来修改//    println(p.name)//    println(p.age)//    p.sayName()//    val a: Int = 100//    var b = 200//    b = 300//    println(b)}def show() = {println("hello")}
}

Scala中Object和Class区别(伴生类和伴生对象)相关推荐

  1. 每天学一点Scala之 伴生类和伴生对象

    1.  object 相当于class的单个实例,因此,在object里声明的变量都是静态变量,静态方法 2.  在object里声明的变量,都是全局变量,也就是类的公共属性,或者类似于java中父类 ...

  2. 大数据开发语言Scala(三)——伴生类和伴生对象

    Scala语言是完全面向对象的语言,所以并没有静态的操作(即在Scala中没有静态的概念).但是为了能够和Java语言交互(因为Java中有静态概念),就产生了一种特殊的对象来模拟类对象,该对象为单例 ...

  3. Scala伴生类和伴生对象

    单例对象与类同名时,这个单例对象被称为这个类的伴生对象,而这个类被称为这个单例对象的伴生类.伴生类和伴生对象要在同一个源文件中定义,伴生对象和伴生类可以互相访问其私有成员.不与伴生类同名的单例对象称为 ...

  4. [scala-spark]5. 伴生类和伴生对象

    单例对象与类同名时,这个单例对象被称为这个类的伴生对象,而这个类被称为这个单例对象的伴生类.伴生类和伴生对象要在同一个源文件中定义,伴生对象和伴生类可以互相访问其私有成员.不与伴生类同名的单例对象称为 ...

  5. 样例类,样例对象,伴生类,伴生对象

    样例类 使用case class 类名 定义 case class Person(var name:String, var age:Int) 可以直接使用 : Person("张三" ...

  6. scala中object和class的理解---apply方法是初始化方法

    1.class Scala的类和C#中的类有点不一样,诸如: 声明一个未用priavate修饰的字段 var age,scala编译器会字段帮我们生产一个私有字段和2个公有方法get和set ,这和C ...

  7. [Scala基础]-- 伴生类和伴生对象

    Scala比 Java 更面向对象的一个方面是 Scala 没有静态成员.替代品是,Scala 有: 单例对象:singleton object. 除了用 object 关键字替换了 class 关键 ...

  8. scala的伴生类和伴生对象所属类是什么?

    文章目录 说在前面 运行原理 代码体现 解析 说在前面 scala是纯面向对象的语言,C是纯面向过程的语言,Java就是面向过程+面向对象的语言. 在说到伴生对象之前,要先说到java的static关 ...

  9. scala中sortBy与sortWith区别

    (1)Scala中sortBy是以方法的形式存在的,并且是作用在Array或List集合排序上,并且这个sortBy默认只能升序,除非实现隐式转换或调用reverse方法才能实现降序. (2)sort ...

最新文章

  1. meson 中调用shell script
  2. Linux那些事儿之我是Sysfs(11)sysfs 创建普通文件
  3. 小鹏,该兑现PPT了
  4. ubuntu安装(owncloud-docker安装)
  5. HTML标记语法总结
  6. SAP UI5 control id generation by jQuery
  7. 进程前台运行后台运行的相关命令
  8. yum install git Could not resolve host: ftp.twaren.net; Unknown error
  9. 国际C语言混乱代码大赛
  10. centos7 菜鸟第一天--输入法在哪
  11. Mac OSX上折腾JDK
  12. 全国计算机三级网络技术题库南开,计算机三级上机:适用于三级网络技术、数据库等(南开100题)...
  13. java类和对象的生命周期
  14. ios java模拟器 2017_Visual Studio 2017(Xamarin)未显示iPhone模拟器列表
  15. iOS获取设备IP地址(WiFi,蜂窝数据两种情况)
  16. Python 电脑上自动阅读东方头条
  17. Java程序员的职业规划(五年)
  18. _()---由此知度娘和google的冷暖
  19. nginx 响应头详解
  20. 问题事件名称: APPCRASH 应用程序名: xxx.EXE 应用程序版本:

热门文章

  1. 你还在为购买PowerBI Pro账号而烦恼吗?(PowerBI+钉钉可以为你解决)
  2. 三七互娱2018校招前端笔试题
  3. 升级WIN10 (9879)后IE无响应的解决办法
  4. C# KeyDown事件无响应解决方法
  5. MapReduce序列化之统计各部门员工薪资总和
  6. Eclipse,静态,继承,封装,this,super,重写,重载,多态,final,冒泡排序,选择排序,抽象类,接口,内部类
  7. boot spring 叶子暗的_抓住MySQL面试重点知识,让你征服面试官半小时
  8. 用VLOOKUP函数在Excel中批量替换值
  9. mutt配置文件_配置mutt做邮件客服端
  10. Linux入门,非常适合小白