当前位置: 首页 > news >正文

scala函数式编程

目录

不同范式对比:

1.面向对象编程

2.函数式编程

2.1函数基本语法

2.2函数和方法的区别

核心概念:

 2.3函数定义

 2.4函数参数

2.5 函数至简原则

2.6.高阶函数

 三.偏函数

四.柯里化函数

五.递归函数

递归函数注意点:

 六.控制抽象

1.值调用,把计算后的值传递过去,

 2.名调用,把代码传递过去,

七.惰性加载


不同范式对比:

  • 面向过程:按照步骤解决问题。

  • 面向对象:分解对象、行为、属性,通过对象关系以及行为调用解决问题。耦合低,复用性高,可维护性强。

面向对象和面向过程都是命令式编程,但是函数式编程不关心具体运行过程,而是关心数据之间的映射。纯粹的函数式编程语言中没有变量,所有量都是常量,计算过程就是不停的表达式求值的过程,每一段程序都有返回值。不关心底层实现,对人来说更好理解,相对地编译器处理就比较复杂。

函数式编程优点:编程效率高,函数式编程的不可变性,对于函数特定输入输出是特定的,与环境上下文等无关。函数式编程无副作用,利于并行处理,所以Scala特别利于应用于大数据处理,比如Spark,Kafka框架。

1.面向对象编程

解决问题,分解对象,行为,属性,然后通过对象的关系以及行为的调用来解决问题

对象的本质:对数据和行为的一个封装

2.函数式编程

解决问题时,将问题分解成一个个步骤,将每个步骤进行封装(函数),通过调用这些封装好的步骤,解决问题。

函数的本质:函数可以当作一个值进行传递

2.1函数基本语法

实例:需求:定义一个函数,实现将传入的名称打印出来。

object TestFunction {def main(args: Array[String]): Unit = {// (1)函数定义def f(arg: String): Unit = {println(arg) }// (2)函数调用 // 函数名(参数)f("hello world") } }

2.2函数和方法的区别

核心概念:

                 1.为完成某一功能的程序语句的集合,称之为函数

                 2.类中的函数称之为方法

Scala 语言可以在任何的语法结构中声明任何的语法

函数没有重载和重写的概念;方法可以进行重载和重写

Scala 中函数可以嵌套定义

 2.3函数定义

函数 1:无参,无返回值

函数 2:无参,有返回值

函数 3:有参,无返回值

函数 4:有参,有返回值

函数 5:多参,无返回值

函数 6:多参,有返回值

实例:

 

 2.4函数参数

可变参数,类似于Java,使用数组包装。(可变参数可以当作数组来用)

如果参数列表中存在多个参数,那么可变参数一般放置在最后

参数默认值,一般将有默认值的参数放置在参数列表的后面

带名称传参

 

 

 不给名称的就是按顺序赋值

调用时带名参数必须位于实参列表末尾

和默认参数一起使用会很方便,比如有多个默认参数,但只想覆盖其中一个

2.5 函数至简原则

  • 能省则省。
  • 最后一行代码会作为返回值,可以省略return
  • 函数体只有一行代码的话,可以省略花括号。
  • 如果返回值类型能够自动推断那么可以省略。
  • 如果函数体中用return做返回,那么返回值类型必须指定。
  • 如果声明返回Unit,那么函数体中使用return返回的值也不起作用。
  • 如果期望是无返回值类型,那么可以省略=。这时候没有返回值,函数也可以叫做过程。【2.13.0已废弃,能编过不过会提示。】
  • 无参函数如果声明时没有加(),调用时可以省略()。【如果声明时有()调用也可以省略,不过2.13.3废弃了。】
  • 不关心函数名称时,函数名称和def也可以省略,去掉返回值类型,将=修改为=>定义为匿名函数。

 

 

 

2.6.高阶函数

三种形式:函数作为值传递、函数作为参数、函数作为返回值。

作为值传递:经过赋值之后在底层变成一个lambda对象。

实例1模拟 Map 映射、Filter 过滤、Reduce 聚合

 

 三.偏函数

1.在对符合某个条件,而不是所有情况进行逻辑操作时,使用偏函数是一 个不错的选择

2.将包在大括号内的一组case语句封装为函数,我们称之为偏函数,它只 对会作用于指定类型的参数或指定范围值的参数实施计算,超出范围的 值会忽略(未必会忽略,这取决于你打算怎样处理)

3.偏函数在Scala中是一个特质PartialFunction

 

偏函数总结:

使用构建特质的实现类(使用的方式是PartialFunction的匿名子类)

PartialFunction 是个特质(看源码)

构建偏函数时,参数形式 [Any, Int]是泛型,第一个表示参数类型,第二个表示 返回参数

当使用偏函数时,会遍历集合的所有元素,编译器执行流程时先执行isDefinedAt()如果为

true ,就会执行 apply, 构建一个新的Int 对象返回

执行isDefinedAt() 为false 就过滤掉这个元素,即不构建新的Int对象.

map函数不支持偏函数,因为map底层的机制就是所有循环遍历,无法过滤处理 原来集合

的元素

collect函数支持偏函数

实例1:对male,female,男,女,hello进行转义,string转换为int

四.柯里化函数

柯里化(Currying)指的是将原来接受多个参数的函数变成新的接受一个参数的函数,的多个参数列表的过程。

基本介绍:

函数编程中,接受多个参数的函数都可以转化为接受单个参数的函数,这个转 化过程就叫

柯里化

柯里化就是证明了函数只需要一个参数而已。其实我们刚才的学习过程中,已 经涉及到了

柯里化操作。

不用设立柯里化存在的意义这样的命题。柯里化就是以函数为主体这种思想发展的必然产

生的结果。(即:柯里化是面向函数思想的必然产生结果)

五.递归函数

一个函数/方法在函数/方法体内又调用了本身,我们称之为递归调用

递归函数注意点:

  • 方法调用自身。
  • 递归要有结束逻辑。
  • 调用自身时,传递参数要有规律。
  • scala中递归定义函数必须声明返回值类型,因为无法通过推导获得。
  • 纯函数式语言比如Haskell,连循环都没有,很多操作都需要通过递归来做,性能比较依赖尾递归优化。

实例:

 

 六.控制抽象

1.值调用,把计算后的值传递过去,

按值传递参数,计算值后再传递。多数语言中一般函数调用都是这个方式,C++还存在引用传递。

 

 2.名调用,把代码传递过去,

按名称传递参数,直接用实参替换函数中使用形参的地方。能想到的只有C语言中的带参宏函数,其实并不是函数调用,预处理时直接替换。

 

 java只有值调用,scala既有值调用,又有名调用

七.惰性加载

当函数返回值被声明为 lazy 时,函数的执行将被推迟,直到我们首次对此取值,该函 数才会执行。这种函数我们称之为惰性函数。

 

http://www.lryc.cn/news/45585.html

相关文章:

  • 网吧2023:热闹回来了,电竞战歌起
  • 代码随想录算法训练营第五十九天|503.下一个更大元素II、42. 接雨水
  • 9、简单功能分析
  • 如何发送和接收参数?五种参数传递方法
  • 蓝桥杯C/C++VIP试题每日一练之矩形面积交
  • Spark大数据处理讲课笔记2.4 IDEA开发词频统计项目
  • 【ChatGPT 】国内无需注册 openai 即可访问 ChatGPT:ChatGPT Sidebar 浏览器扩展程序的安装与使用
  • 使用fetch()异步请求API数据实现汇率转换器
  • GPT-4“王炸”,10秒钟开发一套Web + APP 系统
  • Disjoint 集合数据结构或 Union-Find 算法简介
  • uniapp中nvue与vue的区别?
  • 带头双向循环链表的实现
  • 大屏使用dv-digital-flop定时刷新显示总人数
  • Java面向对象部分 个人学习记录
  • MySQL数据库——对Linux MySQL软件包的一些说明
  • 【JavaEE进阶】——第二节.Spring核心和设计思想
  • twitter开源算法(1)For You推荐系统架构
  • A General Framework for Uncertainty Estimation in Deep Learning源码阅读(二)
  • 串行通信协议---HART协议
  • 【独家】华为OD机试 - 寻找密码(C 语言解题)
  • FPGA有哪些优质的带源码的IP开源网站?
  • 基于模型预测控制(MPC)的微电网调度优化的研究(Matlab代码实现)
  • Postman接口测试之Mock快速入门
  • 分享一个国内可用的免费ChatGPT网站
  • 15. 三数之和(Java)
  • Navicat Premium 16安装教程
  • 蓝桥杯刷题冲刺 | 倒计时8天
  • 四.JAVA基础面试题:重要知识
  • 某面试官分享经验:看求职者第一眼,开口说第一句话,面试结果就差不多定了,准确率高达90%以上...
  • Java开发 - 消息队列之RabbitMQ初体验