package shujuleiyumifenglei
/*
Kotlin 可以创建一个只包含数据的类,关键字为data
*/
/*
编译器会自动在主构造函数中根据所有声明的属性提取以下函数
equals() //hashCode()
toString() 格式: "User(name=John,age=42)"
componentN() functions 对应与属性,按声明顺序排列
copy函数
如果这些函数在类中已经明确定义或者从超类中继承而来,就不会再生成
为了保证生成代码的一致性,以及有意义数据类需要满足以下条件
1、主构造函数中至少包含一个参数
2、所有主构造函数的参数必须标识为val或者var
3、数据类不能声明为abstract , open , sealed 或者inner
4、数据类不能继承其他类,但可以实现接口
*/
data class User(val name:String,val age:Int)
//---------复制--------
//fun copy(name:String=this.name,age:Int=this.age)=User(name,age)
/*
fun main(args: Array<String>) {
val jack=User("jack",1)
val olderJack=jack.copy(age=2) //可以直接指定字段名创建
println(jack)
println(olderJack)
//数据类以及解构声明
//组件函数允许数据类在解构声明中使用
val (nam)=jack//将对应属性名(或对应位置的)的值赋值到组件函数中
println("$nam ")
//标准数据类
//标准库提供了Pair和Triple 命名数据类是更好的设计选择,
//因为这样的代码可读性更强且提供了有意义的名字和属性
}
*/
/*
密封类用来表示首先的泪继承结构:当一个值为有限集中的类型,
而不能有任何其他类型时,从某种意义上,他们是枚举类的扩展,
枚举类型的值集合也是受限的,但每个枚举常量值存在一个实例
而密封泪的一个子类可以有可包含状态的多个实例
声明一个密封类,使用sealed修饰类,密封类可以有子类
但是所有的子类都必须要内嵌在密封类中
sealed不能修饰interface abstract class 会报警告不会出现编译错误
*/
sealed class Expr
data class Const(val number:Double):Expr()
data class Sum(val e1:Expr,val e2:Expr):Expr()
object NotANumber:Expr()
fun eval (expr:Expr):Double=when(expr){
is Const->expr.number
is Sum->eval(expr.e1)+eval(expr.e2)
NotANumber->Double.NaN
}
//----------强化版的枚举 类似于简单工厂模式设计-----------
sealed class Operation {
class Add(val value: Int) : Operation()
class Substract(val value: Int) : Operation()
class Multiply(val value: Int) : Operation()
class Divide(val value: Int) : Operation()
}
fun execute(x: Int, op: Operation) = when (op) {
is Operation.Add -> x + op.value
is Operation.Substract -> x - op.value
is Operation.Multiply -> x * op.value
is Operation.Divide -> x / op.value
}
fun main(args: Array<String>) {
println(execute(10,Operation.Add(5)))
}