java函数式编程(java函数式编程性能)

本篇文章给大家谈谈java函数式编程,以及java函数式编程性能对应的知识点,希望对各位有所帮助,不要忘了收藏本站喔。

本文目录一览:

java学习有哪些阶段?

我想自学java是每一个想要进入这个行业的首选,可自学JAVA,对于初级水平的小白是难上加上。所以我建议首先学习下HTML相关的知识,其次逐渐深入java语言知识,最后系统重新学习一边。今天呢,IT培训就和大家简单聊聊JAVA基础以及学习的6个阶段,希望能余知宽够帮助到大家!

一.JAVA基础

1.Java编程语言(新版)

2.Java进阶之设计模式

3.JDK核心API

4.MySQL基础课程

5.正则表达式基础

6.JDBC入门教程

J2SESSH框架

7.Java函数式编程

8.J2SE网络通信实践

9.Struts框架教程

10.Hibernate框架教程

11.Spring框架入门教程(新版)

12.MyBatis框架基础入门

13.SpringMVC简易教程

二.6个阶段

第一阶段:HTML、DIV+CSS及相关的前端基础知识语言;

第二阶段:JavaScript及相关的JQuery、Ajax等交互相关的知识;

第三阶段:JAVA基础,面对对象思想以及IO流等JAVA基础知识;

第四阶段:JSP及Servlet,以及相关的JDBC连接数据库等相关的知识;

第五阶段:Java及JSP、Servlet等相关知识学完后,之后就是数据库Mysql、Oracle等相关知识;

第六阶段:三大主流框架Spring、Struts2及Hibernate,抑或是Mybatis等框架,学完框架基本上就OK了,就可以出去找工作了;

以上,就是小编为大家整理的自学JAVA知识,文字写起来看起来就简单的几个字,但是实践起来确实不是那么竖亮容易,当然了如果你很聪明的话除外。总之,自学的话,需要极强的自控能力,不能3天打鱼2天晒猛李网。只有坚持下来,才能事半功倍。

Java8的函数式编程怎么样?

使用函数式代码的好处:

减少了可变量(Immutable Variable)的声明

能够更好旅码晌档的利用并行(Parallelism)

代码更加简洁和可读

函数式接口

函数式接口就是仅声明了一个方法的接口,比如我们熟悉的Runnable,Callable,Comparable等都可以作拆谨哪为函数式接口。当然,在Java 8中,新添加了一类函数式接口,如Function,Predicate,Consumer,Supplier等。

java支持的编程范式

Java编程范式

1.命令式编程

核心内容就是:“用语句更改程序的状态”

大多数流行的编程语言都或多或少基于命令式编程发展而来,命令式语言最典型的实例就是C语言

2.面向空袭对象编程

面向对象编程经常与命令式编程联系在一起,在实践当中,两者是可以共存的。Java就是这种协作的生动证明

面向对象基于四个基本原则:封装、继承、多态、抽象

3.声明式编程

与命令式编程相反,声明式编程它指定程序应该做什么,而不具体说明怎么做。

纯粹的声明式语言包括数据库查询语言(如SQL和Xpath)以及正则表达式。

与命令式编程语言相比,声明式编程语言更加抽象,它们并不模拟硬件结构,因此不会改变程序状态,而是将它们转换为新巧睁状态,并且更接近数学逻辑

通常,非命令是的编程范式都被认为属于声明式类别。

4.函数式编程

函数式编程是声明式编程的子范式,与命令式编程相反,函数式变成不会改变程序的内部状态。

在函数式编程术孝亏岁语中,函数类似于数学函数,函数的输出仅依赖于其参数,而不管程序的状态如何,完全不受函数式是何时执行的影响

函数式语言受欢迎的原因之一是它们可以轻松的在并行环境中运行,这与多线程不太一样,函数式语言支持并行的关键在于它们的基本原理:函数仅依赖与输入参数而不依赖于程序的状态。它们可以在任何地方运行,然后将多个并行执行的结果连接起来并进一步使用

Java函数式编程语言是什么?

函数式编程语言的核心是它以处理数据的方式处理代码。这意味着函数应该是第一等级(First-class)的值,并且能够被赋值给变量,传递给函数等等。

事实上,很多函数式语言比这走得更远,将计算和算法看得比它们操作的数据更重要。其中有些语言想分离程序状态和函数(以一种看起来有点对立的方式,使用面向对象的语言,这通常会将它们联系得更紧密)。

Clojure编程语言就是一个这样的例子则早,尽管它运行于基于类的Java虚拟机,Clojure的本质是函数式语言,并且在高级语言源程并盯物序中不直绝液接公布类和对象(尽管提供了与Java良好的互操作性)。

[img]

函数式编程-Lambda与Stream

我们在创建线程并启动时可以使用匿名内部类的写法:

可以使用Lambda的格式对其进行修改。修改后如下:

现有方法定义如下,其中IntBinaryOperator是一个接口。先使用匿名内部类的写法调用该方法。

Lambda写法:

现有方法定义如下,其中IntPredicate是一个接口。先使用匿名内部类的写法调用该方法。

Lambda写法:

现有方法定义如下,其中Function是一个接口。先使用匿名内部类的写法调用卜行握该方法。

Lambda写法:

现有方法定义如下,其中IntConsumer是一个接口。先使用匿名内部类的写法调用该方法。

Lambda写法:

Java8的Stream使用的是函数式编程模式,如同它的名字一样,它可以被用来对集合或数组进行链状流式的操作。可以更方便的让我们对集合或数组操作。

我们可以调用getAuthors方法获取到作家的集合。现在需要打印所有年龄小于18的作家的名字,并且要注意去重。

单列集合: 集合对象.stream()

数组:Arrays.stream(数组)或者使用Stream.of来创建

双列集合:转换成单列集合后再创建

可以对流中的元素进行条件过滤,符合过滤条件的才能继续留在流中。

例如:

打印所有姓名长度大于1的作家的姓名

可以把对流中的元素进行计算或转换。

例如:

打印所有作家的姓名

可以去除流中的重复元素。

例如:

打印所有作家的姓名,并且要求其中不能有重复元素。

注意:distinct方法是依赖Object的equals方法来判断是否是相同对象的。所以需要注意重写equals方法。

可以对流中的元素进行排序。

例如:

对流中的元素按照年龄进行降序排序,并且要求不能有重复的元素。

注意:如果调用空参的sorted()方法,需要流中的元素是实现了Comparable。

可以设置流的最大长度,超出的部分将被抛弃。带昌

例如:

对流中的元素按照年龄进行降序排序,并且要求不能有重复的元素,然后打印其中年龄最大的两个作家的姓名。

跳过流中的前n个元素,返回剩下的元素

例如:

打印除了年龄最大的作家外的其他作家,要求不能有重复元素,并且按照年龄降序排序。

map只能把一个对象转换成另一个对象来作为流中的元素。而flatMap可以把一个对象转换成多个对象作为流中的元素。

例一:

打印所有书籍的名字。要求对重复的元素进行去重。

例二:

打印现有数据的所有分类。要求对分类进行去重。不能出现这种格式:哲学,爱情

对流中的元素进行遍历操作,我们通过传入的型庆参数去指定对遍历到的元素进行什么具体操作。

例子:

输出所有作家的名字

可以用来获取当前流中元素的个数。

例子:

打印这些作家的所出书籍的数目,注意删除重复元素。

可以用来或者流中的最值。

例子:

分别获取这些作家的所出书籍的最高分和最低分并打印。

把当前流转换成一个集合。

例子:

获取一个存放所有作者名字的List集合。

获取一个所有书名的Set集合。

获取一个Map集合,map的key为作者名,value为List

可以用来判断是否有任意符合匹配条件的元素,结果为boolean类型。

例子:

判断是否有年龄在29以上的作家

可以用来判断是否都符合匹配条件,结果为boolean类型。如果都符合结果为true,否则结果为false。

例子:

判断是否所有的作家都是成年人

可以判断流中的元素是否都不符合匹配条件。如果都不符合结果为true,否则结果为false

例子:

判断作家是否都没有超过100岁的。

获取流中的任意一个元素。该方法没有办法保证获取的一定是流中的第一个元素。

例子:

获取任意一个年龄大于18的作家,如果存在就输出他的名字

获取流中的第一个元素。

例子:

获取一个年龄最小的作家,并输出他的姓名。

对流中的数据按照你指定的计算方式计算出一个结果。(缩减操作)

reduce的作用是把stream中的元素给组合起来,我们可以传入一个初始值,它会按照我们的计算方式依次拿流中的元素和初始化值进行计算,计算结果再和后面的元素计算。

reduce两个参数的重载形式内部的计算方式如下:

其中identity就是我们可以通过方法参数传入的初始值,accumulator的apply具体进行什么计算也是我们通过方法参数来确定的。

例子:

使用reduce求所有作者年龄的和

使用reduce求所有作者中年龄的最大值

使用reduce求所有作者中年龄的最小值

reduce一个参数的重载形式内部的计算

如果用一个参数的重载方法去求最小值代码如下:

我们在编写代码的时候出现最多的就是空指针异常。所以在很多情况下我们需要做各种非空的判断。

例如:

尤其是对象中的属性还是一个对象的情况下。这种判断会更多。

而过多的判断语句会让我们的代码显得臃肿不堪。

所以在JDK8中引入了Optional,养成使用Optional的习惯后你可以写出更优雅的代码来避免空指针异常。

并且在很多函数式编程相关的API中也都用到了Optional,如果不会使用Optional也会对函数式编程的学习造成影响。

Optional就好像是包装类,可以把我们的具体数据封装Optional对象内部。然后我们去使用Optional中封装好的方法操作封装进去的数据就可以非常优雅的避免空指针异常。

我们一般使用 Optional 的 静态方法ofNullable 来把数据封装成一个Optional对象。无论传入的参数是否为null都不会出现问题。

你可能会觉得还要加一行代码来封装数据比较麻烦。但是如果改造下getAuthor方法,让其的返回值就是封装好的Optional的话,我们在使用时就会方便很多。

而且在实际开发中我们的数据很多是从数据库获取的。Mybatis从3.5版本可以也已经支持Optional了。我们可以直接把dao方法的返回值类型定义成Optional类型,MyBastis会自己把数据封装成Optional对象返回。封装的过程也不需要我们自己操作。

如果你 确定一个对象不是空 的则可以使用 Optional 的 静态方法of 来把数据封装成Optional对象。

但是一定要注意,如果使用of的时候传入的参数必须不为null。(尝试下传入null会出现什么结果)

如果一个方法的返回值类型是Optional类型。而如果我们经判断发现某次计算得到的返回值为null,这个时候就需要把null封装成Optional对象返回。这时则可以使用 Optional 的 静态方法empty 来进行封装。

所以最后你觉得哪种方式会更方便呢? ofNullable

我们获取到一个Optional对象后肯定需要对其中的数据进行使用。这时候我们可以使用其 ifPresent 方法对来消费其中的值。

这个方法会判断其内封装的数据是否为空,不为空时才会执行具体的消费代码。这样使用起来就更加安全了。

例如,以下写法就优雅的避免了空指针异常。

如果我们想获取值自己进行处理可以使用get方法获取,但是不推荐。因为当Optional内部的数据为空的时候会出现异常。

如果我们期望安全的获取值。我们不推荐使用get方法,而是使用Optional提供的以下方法。

我们可以使用filter方法对数据进行过滤。如果原本是有数据的,但是不符合判断,也会变成一个无数据的Optional对象。

我们可以使用isPresent方法进行是否存在数据的判断。如果为空返回值为false,如果不为空,返回值为true。但是这种方式并不能体现Optional的好处, 更推荐使用ifPresent方法 。

Optional还提供了map可以让我们的对数据进行转换,并且转换得到的数据也还是被Optional包装好的,保证了我们的使用安全。

例如我们想获取作家的书籍集合。

只有一个抽象方法 的接口我们称之为函数接口。

JDK的函数式接口都加上了 @FunctionalInterface 注解进行标识。但是无论是否加上该注解只要接口中只有一个抽象方法,都是函数式接口。

我们在使用lambda时,如果方法体中只有一个方法的调用的话(包括构造方法),我们可以用方法引用进一步简化代码。

我们在使用lambda时不需要考虑什么时候用方法引用,用哪种方法引用,方法引用的格式是什么。我们只需要在写完lambda方法发现方法体只有一行代码,并且是方法的调用时使用快捷键尝试是否能够转换成方法引用即可。

当我们方法引用使用的多了慢慢的也可以直接写出方法引用。

类名或者对象名::方法名

其实就是引用类的静态方法

如果我们在重写方法的时候,方法体中 只有一行代码 ,并且这行代码是 调用了某个类的静态方法 ,并且我们把要重写的 抽象方法中所有的参数都按照顺序传入了这个静态方法中 ,这个时候我们就可以引用类的静态方法。

例如:

如下代码就可以用方法引用进行简化

注意,如果我们所重写的方法是没有参数的,调用的方法也是没有参数的也相当于符合以上规则。

优化后如下:

如果我们在重写方法的时候,方法体中 只有一行代码 ,并且这行代码是 调用了某个对象的成员方法 ,并且我们把要重写的 抽象方法中所有的参数都按照顺序传入了这个成员方法中 ,这个时候我们就可以引用对象的实例方法

例如:

优化后:

如果我们在重写方法的时候,方法体中 只有一行代码 ,并且这行代码是 调用了第一个参数的成员方法 ,并且我们把要 重写的抽象方法中剩余的所有的参数都按照顺序传入了这个成员方法中 ,这个时候我们就可以引用类的实例方法。

例如:

优化后如下:

如果方法体中的一行代码是构造器的话就可以使用构造器引用。

如果我们在重写方法的时候,方法体中 只有一行代码 ,并且这行代码是 调用了某个类的构造方法 ,并且我们把 要重写的抽象方法中的所有的参数都按照顺序传入了这个构造方法中 ,这个时候我们就可以引用构造器。

例如:

优化后:

我们之前用到的很多Stream的方法由于都使用了泛型。所以涉及到的参数和返回值都是引用数据类型。

即使我们操作的是整数小数,但是实际用的都是他们的包装类。JDK5中引入的自动装箱和自动拆箱让我们在使用对应的包装类时就好像使用基本数据类型一样方便。但是你一定要知道装箱和拆箱肯定是要消耗时间的。虽然这个时间消耗很下。但是在大量的数据不断的重复装箱拆箱的时候,你就不能无视这个时间损耗了。

所以为了让我们能够对这部分的时间消耗进行优化。Stream还提供了很多专门针对基本数据类型的方法。

例如:mapToInt,mapToLong,mapToDouble,flatMapToInt,flatMapToDouble等。

当流中有大量元素时,我们可以使用并行流去提高操作的效率。其实并行流就是把任务分配给多个线程去完全。如果我们自己去用代码实现的话其实会非常的复杂,并且要求你对并发编程有足够的理解和认识。而如果我们使用Stream的话,我们只需要修改一个方法的调用就可以使用并行流来帮我们实现,从而提高效率。

parallel方法可以把串行流转换成并行流。

也可以通过parallelStream直接获取并行流对象。

关于java函数式编程和java函数式编程性能的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。

标签列表