下面的代码包含了基本的Scala的语法内容。包括:判断,循环代码片段,方法定义,调用。 虚拟类,继承,接口,case,package,单例模式
package org.exinglo.scala
class LearnScala {
}
object Test{
val variable1 = "this is my first scala variable"
def func1(): String = "hello world" //不带参数的函数被调用时,可以不加括号
def func2(who : String): String = { return "hello world " + who}
def func3 = (x : Int, y : Int) => x + y //这是个匿名函数的写法,调用的时候,func3(1,2)
//这里给了y一个默认值,如果用户传过来,那就用用户的
def func4(x: Int)(y : Int = 1): Int = x + y //加脂法的写法,调用时, func4(1)(2)
// * 代表多个参数的意思,比如 printEveryChar("c","b", "a"),可以传递多个string
def printEveryChar(c : String*) = {
c.foreach(x => println(x))
}
def main(args : Array[String]) {
println(variable1)
for(i <- 0 to 10 if i%2 == 0){
println("i is a even number: " + i)
}
for(i <- 0 until 10){
println("")
}
var (n, r) = (10, 0)
while(n > 0){
n = n - 1
}
val p = new Person //这里可以省略括号
println("information about person:" + p.name + " " + p.age) //如果p.name没有赋值的话,这里会显示为null
}
}
package org.exinglo.scala
class Person{ //类默认是public类型的
var name : String = _ //会生成setter和getter方法
val age = 10 //只会生成getter方法
private[this] val gender = "male" //无法在下面的main方法访问gender,这个属性只能在对象内部访问
}
//生成对象时 val p = new Person2("Java", 10)
//如果参数声明时不带val或者var,那么这个变量相当于内部变量,比如 class Person(name: String) 这里的name就无法在外部被访问到了
class Person2(val name : String, val age : Int){//括号中的内容属于主构造器,其中的参数会作为类的属性存在
println("this is the person2's constructor")//这里属于默认构造器,生成对象时,会执行这句话
var gender : String = _ //这里要求初始化
//附属构造器必须叫做this,必须首先调用已经存在的子构造器或者其他附属构造器
def this(name : String, age : Int, gender : String){
this(name, age)//必须调用默认的构造器,=
this.gender = gender
}
}
//成成Student对象的时候,会先打印父类构造器的打印语句,然后但因student类的语句
class Student(val grade : Int, name : String, age : Int) extends Person2(name, age) {
println("this is the subclass of Person, grade is:" + grade)
override def toString() : String = {""} //覆盖父类的方法,一定要加上override关键字,其实不仅仅是父类的方法,覆盖父类的var val都要加上override
}
package org.exinglo.abstractandtrait
class ABstractAndTrait {
}
//抽象类(abstract class)
//• 类的⼀一个或者多个⽅方法没有没完整的定义, 当然也可以没有任何抽象方法。
// 对于子类覆盖父类的非抽象方法,一定要override关键字
//• 声明抽象⽅方法不需要加abstract关键字,只需要不写⽅方法体
//• ⼦子类重写⽗父类的抽象⽅方法时不需要加override
//• ⽗父类可以声明抽象字段(没有初始值的字段)
//• ⼦子类重写⽗父类的抽象字段时不需要加override
abstract class PersonX{
def speak
val name:String
val age:Int
}
class Student extends PersonX{
//必须要实现抽象方法,像这种不写等于号的情况,表示返回类型是Unit类型
def speak{
println("speak")
}
val name:String = ""
val age:Int = 3
}
trait Logger{
def log(msg:String)
}
//这里的trait consoleLogger可以不实现父类的方法,也可以覆盖父类的方法实现,也可以实现父类的虚方法
trait ConsoleLogger extends Logger{
def log(msg:String){
println("save money:" + msg)
}
}
//因为是覆盖上级的方法,必须使用override关键字
trait MessageLogger extends ConsoleLogger{
override def log(msg:String){
println("save money to account:" + msg)
}
}
//如果一个类没有继承其他的类,实现的第一个trait只能用extends,其他的用with关键字
class Test extends ConsoleLogger{
def test{
log("hellp")
}
}
abstract class Account{
def save
}
class MyAccount extends Account with ConsoleLogger{
def save{
log("100")
}
}
object Run extends App{
//这里是一个很好玩的地方,单独给一个对象混入了一个特质trait,并且这个特质与类的定义继承的特质有冲突
//打印出来的是MessageLogger中的log方法
val acc = new MyAccount with MessageLogger
acc.save
}
package org.exinglo.abstractandtrait
class ApplyTest(val msg:String) {
def apply() = msg
def test{
println(msg)
}
}
object ApplyTest{
//下面是apply方法的最常用的方式
def apply() = new ApplyTest("object calls new")
//object的方法相当于java中的静态方法
def static{
println("I am a static method")
}
}
object Basic extends App{
//类似于下面,嗲用object的方法
ApplyTest.static
val t = new ApplyTest("new")
println(t())
//据说下面这样可以调用object ApplyTest中的apply方法,但是我的编译器总是报错
//后来发现是object的apply方法一开始声明没有加上小括号造成的,如def apply = xxxx
val tt = ApplyTest()
println(tt())
// ApplyTest()是调用object的apply方法
// val t = new ApplyTest(); t() 调用的是class ApplyTest的apply方法
}
package org.exinglo.abstractandtrait
/*
* package com.xyz{
* //-------------
* package spark
* }
* }
*
* { //这里只在程序块中可见
* import com.yyy
*
* }
*
* import java.util.{HashMap =>JavaHashMap} //定义一个别名
*
*
* package aa.bb.cc.dd
*
* class Xxx{
* private[dd] def variable = {} //根据private[]中指定的不同,可见范围不同
* }
*
* */
class PackageAndCase {
}
object BasicT extends App{
val value = 3
val result = value match{
case 1 => "one"
case 2 => "two"
case _ => "some other number"
}
val result2 = value match{
case i if i == 1 => "one"
case i if i == 2 => "two"
case _ => "other number"
}
def t(obj : Any) = obj match{
case x:Int => println("Int")
case s:String => println("String")
case _ => println("unkonwn type")
}
t(2)
}