Java小技巧:利用局部函数来隐藏和保护函数

有过一些开发经验的开发者应该都知道:尽量用局部变量,尽可能避免用全局变量。过于暴露的变量可能被滥用而导致安全问题,对于代码维护也造成一些麻烦(得仔细检查该变量是不是被很多地方用到,等等)。

那么对于函数/过程/方法呢?随着面向对象编程和一些模块化技术的普及,函数也可以隐藏起来,比如:

public class Demo {
    public static void main(String[] args) {





        fun1();
    }


    private static void fun1() {
        //...
    }



}



这里的fun1静态方法就只能在Demo类的内部可以使用。

但即便如此,实际开发中一个类的代码可能有几百上千行,那么一个private的方法的可见范围和C语言中的全局函数差不多了,但这个方法很可能只有一个地方用到,甚至可能是为这个地方定制的辅助方法,那么这个方法的可见范围是整个类就显得不合适了。

利用Lambda表达式在方法内部定义局部函数

在JavaScript中,我们可以利用function直接定义局部函数:

function main(){
    function fun1(){
        //...
    }

    fun1();
}

Java并不能在方法里直接定义方法,不过好在 Java 8 借鉴一些函数式语言引入了Lambda表达式和函数式接口,这样Java也可以定义局部函数。利用这些特性,上面的代码就变为了:

class Demo {




    public static void main(String[] args) {





        Runnable fun1 = () -> {
            //....




        };

        fun1.run();
    }

}

如果想定义带返回值的局部函数,可以用Supplier接口:

class Demo {




    public static void main(String[] args) {





        Supplier<Integer> fun1 = () -> {
            //....




            return 1;
        };


        Integer integer = fun1.get();
    }



}



如果是只接受参数无返回值的局部函数,则使用Consumer接口:

class Demo {




    public static void main(String[] args) {





        Consumer<Integer> fun1 = (Integer i) -> {
            //....




        };

        fun1.accept(1);
    }

}

而既要参数又要返回值的就使用Function接口:

class Demo {




    public static void main(String[] args) {





        Function<Integer, String> fun1 = (Integer i) -> {
            //....




            return "number:" + i;
        };


        String s = fun1.apply(1);
    }



}



如果出现多个参数的情况,可以定义自己的函数式接口:

interface Fun2<A, B, C> {
    C apply(A a, B b);
}

class Demo {
    public static void main(String[] args) {
        Fun2<Integer, String, String> fun1 = (Integer i, String s) -> {
            //....
            return s + i;
        };
        String s = fun1.apply(1, "number:");
    }
}

或者使用柯里化(Currying):

class Demo {




    public static void main(String[] args) {





        Function<Integer, Function<String, String>> fun1 = (Integer i) -> (String s) -> {
            //....




            return s + i;
        };


        String s = fun1.apply(1).apply("number:");
    }



}



(如果语言有过程类型支持,例如scala、kotlin,编写局部函数会方便很多)

© 版权声明
THE END
喜欢就支持一下吧
点赞0

Warning: mysqli_query(): (HY000/3): Error writing file '/tmp/MYa2L4GC' (Errcode: 28 - No space left on device) in /www/wwwroot/583.cn/wp-includes/class-wpdb.php on line 2345
admin的头像-五八三
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

图形验证码
取消
昵称代码图片