java函数式编程例子_用Java8来进行函数式编程

这个标题有点意思

Java8的明显特点是通过默认接口定义、Lambda表达式、方法引用、函数式接口带来了函数式编程,这些功能的出现也改变了java多年来的一些习惯

接口定义增强:这是一个极其毁三观的方式

java的接口一直是由全局常量和抽象方法组成,但是在Java8出现后,这一个形势就因此改变了…

场景:存在一个接口,而同时有2k个类实现了该接口,突然有一天需求更改,需在接口里添加一个方法,而所有实现该接口的子类该方法的实现是一样的,按照之前的方式,需在每一个子类复写该方法,so….你需要复制粘贴2k次为了解决这个问题,default就诞生了

default示例interface Formula {    double calculate(int a);    default double sqrt(int a) {        return Math.sqrt(a);

}    //static方式    static void get(){

system.out.println("...");

}

}

Formula formula = new Formula() {

@Override    public double calculate(int a) {        return sqrt(a * 100);

}

};

formula.calculate(100);     // 100.0formula.sqrt(16);           // 4.0//static方式Formula.get();

除了用default定义方法,一旦使用了static定义方法意味着这个方法只可以直接由类名称调用。

另外还有一个重要概念:内部类访问方法参数的时候可以不加上final关键字

Lambda表达式

lambda属于函数式编程的概念

传统的匿名内部类,Android中添加监听器的典型例子Button button = (Button) findViewById(R.id.button);

button.setOnClickListener(new OnClickListener() {    @Override

public void onClick(View view) {

Toast.makeText(MainActivity.this, "Button Clicked", Toast.LENGTH_SHORT).show();

}

});这段代码好繁琐

这个代码认真一看,其实主要运用到的代码仅仅只有Toast使用的这一句,但由于java的面向对象语法,不得不嵌套更多内容

做法太过严谨,于是java8引入了函数式编程简化语法怎么简化呢?

Lambda范例:Button button = (Button) findViewById(R.id.button);

button.setOnClickListener(v->Toast.makeText(MainActivity.this, "Button Clicked", Toast.LENGTH_SHORT).show());

长度减了一大半,使用了Lambda表达式大大简化了语法道理我都懂,怎么使用?

Lambda语法三种形式(参数)->单行语句;    () -> System.out.println(“hello”)

(参数)->{单行语句}; (String s) -> { System.out.println(s); }

(参数)->表达式     (int x, int y) -> x + y

范例:public void runnableTest() {        // 一个匿名的 Runnable

Runnable r1 = new Runnable() {

@Override            public void run() {

System.out.println("Hello world one!");

}

};        // Lambda Runnable

Runnable r2 = () -> System.out.println("Hello world two!");        // 执行两个 run 函数

r1.run();

r2.run();

}让我再举一个简化变得更短的例子List names = Arrays.asList("peter", "anna", "mike", "xenia");

Collections.sort(names, (String a, String b) -> b.compareTo(a));//让他再变得更短些

Collections.sort(names, (a, b) -> b.compareTo(a));嘿嘿,看明白了吗

当只有一个表达式时,那么会直接进行返回操作

方法引用

以前更多的是在对象上使用引用,而java8多出的是方法引用这是什么鬼?

待会再跟你解释.

方法引用需要使用 :: 关键字,这是java8才有的东东

接下来,让我们说说四种形式方法引用:引用静态方法:类名称 :: static 方法名称;

引用某个对象的方法:实例化对象 :: 普通方法;

引用特定类型的方法:特定类 :: 普通方法

引用构造方法:类名称 :: new

例子:引用静态方法:

在String类里面有一个valueOf()方法:public static String valueOf(int x);interface Inter

{

public R zhuanhuan(P p);

}

public class Test{

public static void main(String args[]){

Inter msg = String::valueOf;

String str = msg.zhuanhuan(3000);

System.out.println(str); // 3000

//原始方法

Inter msg = new Inter() {

public String zhuanhuan(Integer p) {            return  String.valueOf(p);

}

};

//lambda          Inter msg =(p)->String.valueOf(p);

}

}

通过Inter msg = String::valueOf;

让Inter的R方法拥有了valueOf的功能卧槽,这….不就是传说中的直接复制敌人绝招嘛,

将String.valueOf()方法变为了Inter接口里的R()方法,

再来另外三个例子看看?

例子:引用普通方法:@FunctionalInterface //此为函数式接口,只能定义一个方法interface Inter

{

public R upper();

//public void print();}public class Test{

public static void main(String args[]){

//要在实例化对象下使用          //hello是String的实例化对象          Inter msg = "hello"::toUpperCase;

String str = msg.Upper();

System.out.println(str); // HELLO     }

}

此时我们应该有了疑问:

通过两个代码演示,如果要实现函数引用,接口里必须只存在一个方法。如果再来一个方法,方法不是无法引用了吗?如划线语句//public void print();

所以为了保证被引用接口里面只有一个代码,需加上注解@FunctionalInterface 此为函数式接口

之前引用的方式中,都是静态方法,接下来我们试试引用普通方法需实例化

例子:引用特定类方法 ,比较方法public int compareTo(String anotherString);interface Inter

{

public int compare(P p1,P p2);

}public class Test{

public static void main(String args[]){

Inter msg = String::compareTo;

System.out.println(msg.compare("A","B")); // -1     }

}

与之前相比,方法引用前不再需要定义对象,而是可以理解为将对象定义在了参数上

例子:引用构造方法又一个毁三观的功能interface Inter{

public C create(String t,double p);

}class Book {

private String title;

private double price;

public Book(String title,double price){

this.title = title;

this.price = price;

}

public String toString(){

return "book name:"+this.title+",book price:"+this.price;

}

}public class Test{

public static void main(String args[]){

Inter msg = Book::new;

Book book = msg.create("java",20);

System.out.println(book);//book name:java,book price:20     }

}

那为啥java8不定义这些接口直接给我们使用呢?当然有啦

函数式接口

jdk8提供了一个函数式接口包java.util.function,里面有众多的函数式接口,而其中最基础最常操作的有以下四个核心接口:

功能型接口:public interface Function{public R apply(T t);}

接收一个参数 返回一个结果

例如String.valueOf()

消费型接口:public interface Consumer{public void accept(T t);}

接收参数 不返回结果

例如System.out.println

供给型接口:public interface Supplier{public void get(T t);}

不接收参数 返回结果

例如String的toUpperCase()

断言型接口:public interface Predicate{public boolean test(T t);}

判断操作使用

例如String的equalsIgnoreCase()说这些,来个例子?public class Test{

public static void main(String args[]){

//功能型接口          Function fun = "hello"::startsWith;

System.out.println(fun.apply("he"));          //true          //消费型接口          Consumer cons = System.out::println;

cons.accept("hello");                         //hello          //供给型接口          Supplier sup = "hello"::toUpperCase;

System.out.println(sup.get());                //HELLO          //断言型接口          Predicate pre ="hello"::equalsIgnoreCase;

System.out.println(pre.test("Hello"));        //true              }

}

这几个接口包含的各种引用,也是函数式接口的代表,那么存在其他的众多接口中,都是这四个接口的扩展提升

So,这些就是Java8带来的新特性啦多多实践有利掌握

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值