Kotlin 定义函数的一些语法

fun main(args:Array<String>):Unit {

    val x:() -> Unit = { println("hello") } // 无参数函数,声明类型时用 ()-> 实现时不能要,{() -> println("hello")} 是错误的
    val odd:(Int) -> Boolean = { x -> x % 2 ==1 } // 类型声明参数要放到括号中(即使只有一个参数
    val add:(Int,Int) -> Int = { x,y -> x+y } //函数实现不能用括号即使有多个参数
    val sum = { x:Array<Int> -> var s = 0;for(i in x) s+=i;s }

}
fun main(args:Array<String>){

	val is_odd:(Int)->Boolean = { x -> x % 2 == 1 }//类型声明中参数类型列表的括号不能省略
	val is_even = { x:Int -> x % 2 == 0 }
	val add:(Int,Int)->Int = { x , y -> x +y }
	fun mul(x:Int,y:Int):Int = x * y // 这种定义的效率最高,其他的定义被编译成 extends Lambda implements Function2
	val sub = fun(x:Int,y:Int) = x - y;
	println(is_odd(2));
	println(is_even(2));
	println(add(2,3));
	println(mul(2,3));
	println(sub(2,3));
}

operator fun invoke

fun main(args:Array<String>):Unit {

    val x = A(3)//有参数构造函数不会与伴生对象定义的invoke()歧义,但是可能会和 invoke(x)歧义。
    x();//调用实例对象定义在类中的invoke方法
    A();//如果A有无参构造函数就会与 object invoke 的括号调用歧义,将无法直接用括号调用返回 world 的函数
    B();//没有伴生类的对象定义invoke方法不会产生直接括号调用与无参构造函数的歧义
    A("wengmj");//如果么有定义A.invoke(String)则定义的A.invoke函数有效
}

class A(val x:Int)
{
    init{
        println("hello")
    }

    operator fun invoke():Unit{
        println("abc")
    }

    companion object{
        operator fun invoke():Unit{
            println("world")
        }

        operator fun invoke(x:String):Unit{
            println(x)
        }
    }
}

object B
{
    operator fun invoke():Unit{
        println("def")
    }
}

实现一个可迭代的类

//传统的实现迭代接口方法
class A : Iterator<Int> {
    val x = arrayOf(1,2,3,4,5).iterator();
    override fun next(): Int = this.x.next();
    override fun hasNext(): Boolean = this.x.hasNext();
}
// 实现 operator fun iterator()
class A  {
    val x = arrayOf(1,2,3,4,5).iterator();
    operator fun iterator():Iterator<Int> = this.x.iterator();
}
//扩展一个已有的类使其可迭代
class A() {
    val x = arrayOf(1,2,3,4,5).iterator();
}
operator fun A.iterator():Iterator<Int> = this.x.iterator();
fun main(args:Array<String>):Unit {
    for(x in A())
        println(x)
}

类扩展函数和属性

val String.hasChar:(Char) -> Boolean
    get()= {c:Char -> this.any{ it == c}}

val String.leng:Int //类的属性扩展不能放在函数里:Local extension properties are not allowed get() = this.length; fun main(args:Array<String>):Unit { fun String.len():Int{ //函数里定义扩展函数是可以的 return this.length; } println("hello".hasChar('e')) println("hello".len()) println("hello".leng) }

构造一个递归结构的列表

//扩展实现 IntRange to Array
fun IntProgression.toArray() = Array<Int>(( last - first) / step + 1 , { this.elementAt(it) })
fun main(args:Array<String>):Unit {
    val p = (1..100 step 10).toArray();
    val list = L<Int>(*p).push(1000) // 数组作为 vararg 参数,变量前加 *
    print(sum(list))
}
//递归列表
class L<T> private constructor(val h:T,val t:L<T>?){
    fun push(h:T):L<T> = L<T>(h,this);
    companion object {
        operator fun <T> invoke(vararg args:T):L<T> {
            var x = L(args[0],null);
            for(i in 1..args.lastIndex)
                x = L(args[i],x);
            return x;
        }
    }
}
//用于校验的递归求和函数
fun sum(list:L<Int>):Int {
    with(list) {
        if(t == null)
            return h;
        return h + sum(t)
    }
}

泛型Demo

class A<T>(val name:T){
    companion object{
        fun <T> say(x:T){
            println(x);
        }
    }
    fun hello(){
        println("hello ${this.name}");
    }
}
fun main(args:Array<String>):Unit {
    A.say<String>("hello");
    A<Int>(123).hello();
}

匿名函数和λ表达式

fun main(args:Array<String>):Unit {

    val ins = arrayOf(-1,-2,1,2,3,4);
    /*
       如果一个函数用另一个函数作为它的最后一个参数, λ 表达式可以放在小括号外面的大括号中,
       如果最后一个函数参数同时是唯一的参数则小括号也可以省略. 
    */
    val x = ins.filter { x -> x > 0 } 
    println(x)
    //如果用匿名函数则只能放在小括号中
    val y = ins.filter (fun(x:Int):Boolean { return x>0 });
    println(y) 
}

  

  

posted on 2016-02-23 23:08  scala  阅读(948)  评论(0编辑  收藏  举报