import org.junit.Test

/**
 * 关于scala Class的get、set方法的描述
 */
class C4_Class {

}

//定义一个简单的类
// 定义类,包含field以及方法
class HelloWorld {
    private var name = "leo"

    def sayHello() {
        println("Hello, " + name)
    }

    def getName: String = name

    @Test
    def Test1(): Unit = {
        // 创建类的对象,并调用其方法
        val helloWorld = new HelloWorld
        helloWorld.sayHello()
        //也可以不加括号,如果定义方法时不带括号,则调用方法时也不能带括号
        println(helloWorld.getName)
    }
}

//默认getter与setter
// 定义不带private的var field,此时scala生成的面向JVM的类时,会定义为private的name字段,并提供public的getter和setter方法
// 而如果使用private修饰field,则生成的getter和setter也是private的
// 如果定义val field,则只会生成getter方法
// 如果不希望生成setter和getter方法,则将field声明为private[this]
class Student {
    var name = "leo"

    @Test
    def Test2(): Unit = {
        // 调用getter和setter方法,分别叫做name和name_ =
        val leo = new Student
        print(leo.name)
        leo.name = "leo1"
    }
}

//自定义getter与setter
//如果只是希望拥有简单的getter和setter方法,那么就按照scala提供的语法规则
// 根据需求为field选择合适的修饰符就好:var、val、private、private[this]
// 但是如果希望能够自己对getter与setter进行控制,则可以自定义getter与setter方法
// 自定义setter方法的时候一定要注意scala的语法限制,签名、=、参数间不能有空格
class Student2 {
    private var myName = "leo"

    def name: String = "your name is " + myName

    def name_=(newValue: String) {
        print("you cannot edit your name!!!")
    }

    @Test
    def Test3(): Unit = {
        val leo = new Student2
        print(leo.name)
        println()
        leo.name = "leo1"
    }
}

// 仅暴露field的getter方法
// 如果你不希望field有setter方法,则可以定义为val,但是此时就再也不能更改field的值了
// 但是如果希望能够仅仅暴露出一个getter方法,并且还能通过某些方法更改field的值,那么需要综合使用private以及自定义getter方法
// 此时,由于field是private的,所以setter和getter都是private,对外界没有暴露;自己可以实现修改field值的方法;自己可以覆盖getter方法
class Student3 {
    private var myName = "leo"

    def updateName(newName: String) {
        if (newName == "leo1") myName = newName
        else print("not accept this new name!!!")
    }

    def name: String = "your name is " + myName
}

// private[this]的使用 对象私有
// 如果将field使用private来修饰,那么代表这个field是类私有的,在类的方法中,可以直接访问类的其他对象的private field
// 这种情况下,如果不希望field被其他对象访问到,那么可以使用private[this],意味着对象私有的field,只有本对象内可以访问到
class Student5 {
    private var myAge = 0

    def age_=(newValue: Int) {
        if (newValue > 0) myAge = newValue
        else print("illegal age!")
    }

    def age: Int = myAge

    def older(s: Student5): Boolean = {
        myAge > s.myAge
    }

    //辅助constructor
    // Scala中,可以给类定义多个辅助constructor,类似于java中的构造函数重载
    //辅助constructor之间可以互相调用,而且必须第一行调用主constructor
    class Student8 {
        private var name = ""
        private var age = 0
        def this(name: String) {
            this()
            this.name = name
        }
        def this(name: String, age: Int) {
            this(name)
            this.age = age
        }
    }

    //主constructor
    // Scala中,主constructor是与类名放在一起的,与java不同
    // 而且类中,没有定义在任何方法或者是代码块之中的代码,就是主constructor的代码,这点感觉没有java那么清晰
    class Student9(val name: String, val age: Int) {
        println("your name is " + name + ", your age is " + age)
    }

    // 主constructor中还可以通过使用默认参数,来给参数默认的值
    class Student10(val name: String = "leo", val age: Int = 30) {
        println("your name is " + name + ", your age is " + age)
    }

    // 如果主constrcutor传入的参数什么修饰都没有,比如name: String,那么如果类内部的方法使用到了,则会声明为private[this] name;
    // 否则没有该field,就只能被constructor代码使用而已

    // Scala中,同样可以在类中定义内部类;但是与java不同的是,每个外部类的对象的内部类,都是不同的类
    import scala.collection.mutable.ArrayBuffer
    class Class {
        class Student(val name: String) {}
        val students = new ArrayBuffer[Student]
        def getStudent(name: String): Student =  {
            new Student(name)
        }

        @Test
        def Test3(): Unit = {
            val c1 = new Class
            val s1 = c1.getStudent("leo")
            c1.students += s1

            val c2 = new Class
            val s2 = c2.getStudent("leo")
//            报错,每个外部类的对象的内部类,都是不同的类
//            c1.students += s2
        }
    }


}