restapi(7)- 谈谈函数式编程的思维模式和习惯

2024-04-09 04:38

本文主要是介绍restapi(7)- 谈谈函数式编程的思维模式和习惯,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  国庆前,参与了一个c# .net 项目,真正重新体验了一把搬砖感觉:在一个多月时间好像不加任何思考,不断敲键盘加代码。我想,这也许是行业内大部分中小型公司程序猿的真实写照:都是坐在电脑前的搬砖工人。不过也不是没有任何收获,在搬砖的过程中我似乎发现了一些现象和造成这些现象背后的原因及OOP思维、习惯模式。和大部分IT公司一样,这间公司在行业里存在了一定时间(不是初创)所以在产品和技术方面有一定的积累,通俗点就是一堆现成的c# .net 代码。然后就是项目截止日期压力。为了按时完成任务的我只能在原有代码基础上不断加功能,根本没有机会去考虑用什么样的代码模式、结构去达到更好的效果。在这个过程中有个有趣的现象引起了我的注意:基本上我只需按照某种流程(多数是业务需求)一个个增加环节就可以实现一项完整功能,当然我是不会计较这些环节对软件其它部分是否产生影响,又或者以后代码维护会不会很麻烦,只要能及时交货就行。想想这种做法恰恰是面向对象编程或所谓行令式编程的特点,即:通过逐行执行命令引导程序的状态改变,最终状态就是运行程序的结果了,或者就是功能的实现了。通过一行行增加代码最终总会到达预期的状态,不是吗。这正是OO编程的思维模式:因为程序状态体现在每行代码上,随时可以检查,验证思路,所以OOP比较容易上手(相对函数式编程而言)。

        回顾一下函数式编程:好像很难按照自然逻辑思维顺序来实现一个功能,这是因为函数式编程是一种嵌套式间接性的编程模式,即程序是在某种嵌套里运行的。函数式编程又被称为monatic-programming,即在monad里编程。monad就是我所说的嵌套,是一种类型结构,最常用的是Future类型。在现代编程里多线程编程非常普遍,实际上往往我们离不开各种各样的Future。举个形象的例子:如果实现把脏水从A点引到B点输出纯净水作为某种函数式程序,编程如同搭建管道网。必须首先准备好各式各样功能的喉管,实现每种喉管的特殊功能如过滤、消毒等,然后再连接组合形成送水管道。

      我在进行函数式编程时总是要把所以问题前前后后都考虑清楚了才能开始动手。首先会把一项功能的所有环节先总结出来,这些都是一些函数。然后尝试把这些函数的类型统一了,就像上面提到的喉管一样,因为不同规格的喉管是无法连接的。同样,不同类型的嵌套monad是无法实现函数组合的。然后先根据需求实现这些函数的输入输出,最后把这些函数组合起来形成完整功能。你看,在函数式编程里是无法做到随意想到那就写到那的,必须先进行整体的思量。所以,函数式编程在代码重用和维护上有先天的优势。这个例子也体现了函数式编程的思维模式。

   下面我想用一个实际的例子来示范函数式编程模式:前面几篇讨论的例子里有一个是把前端httpclient上传httpserver的图片存放入服务器端mongodb数据库的。现在发现客户端上传图片数据流有困难,希望上传一个图片下载网址,由httpserver自行下载图片并写入mongodb。单从这个功能来讲,应该由几个环节组成:

1、从上传的数据中抽出图片下载网址

2、下载图片,通过http的request请求,从response里获取图片数据流

3、通过mongodb的count功能获取图片系列序号

4、将图片写入mongodb

首先,我需要把这几个环节形成函数,然后统一函数类型。无可争议,最好选择Future[A]这样的函数返回类型:

假设数据是用json格式传上来的,那得有个类型作为数据结构:

  case class UpData (pid: String, url: String)

可以如下获取上传的数据:

 entity(as[String]) { json =>val upData: UpData = fromJson[UpData](json)...
}

获取图片系列序号:返回Future[Long]

repository.count(upData.pid).toFuture[Long]

下载图片:这个返回Future[ByteString]

    import akka.actor.ActorSystemimport akka.http.scaladsl.model._import akka.http.scaladsl.Httpdef downloadPicture(url: String)(implicit sys: ActorSystem): Future[ByteString] = {val dlRequest = HttpRequest(HttpMethods.GET, uri = url)Http(sys).singleRequest(dlRequest).flatMap {case HttpResponse(StatusCodes.OK, _, entity, _) =>entity.dataBytes.runFold(ByteString()) { case (hd, bs) =>hd ++ bs}case _ => Future.failed(new RuntimeException("failed  getting picture!"))}}

写入mongodb:这个函数也返回Future[?]

    def addPicuture(pid: String,seqno: Int, optDesc: Option[String],optWid:Option[Int],optHgh:Option[Int],bytes: Array[Byte]):Future[Completed] ={var doc = Document("pid" -> pid,"seqno" -> seqno,"pic" -> bytes)if (optDesc != None)doc = doc + ("desc" -> optDesc.get)if (optWid != None)doc = doc + ("width" -> optWid.get)if (optHgh != None)doc = doc + ("height" -> optHgh.get)repository.insert(doc).toFuture[Completed]}

好了,现在这几个函数都是Future类型的,可以进行组合了:

            val futSeqno: Future[Long] = for {cnt <- repository.count(upData.pid).toFuture[Long]barr <- downloadPicture(upData.url)_ <- addPicuture(upData.pid, cnt.toInt, None, None, None, barr.toArray)} yield cnt

futSeqNo是个组合的运算流程。注意它的类型还是future:意味这我们无法预测这个运算什么时候会完成,特别如果下载一张超大图片又或者网速缓慢的话,很可能在下载完成之前就执行了complete()。所以我们必须保证图片下载完成后才向终端httpclient返回response,就用onComplete来实现:

            onComplete(futSeqno) {case Success(lv) => complete(lv.toString())case _ => complete("error saving picture!")}

所以整段宏观代码如下:

        post {entity(as[String]) { json =>val upData: UpData = fromJson[UpData](json)val futSeqno: Future[Long] = for {cnt <- repository.count(upData.pid).toFuture[Long]barr <- downloadPicture(upData.url)_ <- addPicuture(upData.pid, cnt.toInt, None, None, None, barr.toArray)} yield cntonComplete(futSeqno) {case Success(lv) => complete(lv.toString())case _ => complete("error saving picture!")}}}~

是不是很容易读懂理解?实际上我们把复杂的细节函数藏在背后。而这些函数是高度可重复利用的,这也是我们在动手之前通盘考虑的成果。

这篇关于restapi(7)- 谈谈函数式编程的思维模式和习惯的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/887193

相关文章

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

MySQL字符串常用函数详解

《MySQL字符串常用函数详解》本文给大家介绍MySQL字符串常用函数,本文结合实例代码给大家介绍的非常详细,对大家学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql字符串常用函数一、获取二、大小写转换三、拼接四、截取五、比较、反转、替换六、去空白、填充MySQL字符串常用函数一、

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

MySql基本查询之表的增删查改+聚合函数案例详解

《MySql基本查询之表的增删查改+聚合函数案例详解》本文详解SQL的CURD操作INSERT用于数据插入(单行/多行及冲突处理),SELECT实现数据检索(列选择、条件过滤、排序分页),UPDATE... 目录一、Create1.1 单行数据 + 全列插入1.2 多行数据 + 指定列插入1.3 插入否则更

PostgreSQL中rank()窗口函数实用指南与示例

《PostgreSQL中rank()窗口函数实用指南与示例》在数据分析和数据库管理中,经常需要对数据进行排名操作,PostgreSQL提供了强大的窗口函数rank(),可以方便地对结果集中的行进行排名... 目录一、rank()函数简介二、基础示例:部门内员工薪资排名示例数据排名查询三、高级应用示例1. 每

全面掌握 SQL 中的 DATEDIFF函数及用法最佳实践

《全面掌握SQL中的DATEDIFF函数及用法最佳实践》本文解析DATEDIFF在不同数据库中的差异,强调其边界计算原理,探讨应用场景及陷阱,推荐根据需求选择TIMESTAMPDIFF或inte... 目录1. 核心概念:DATEDIFF 究竟在计算什么?2. 主流数据库中的 DATEDIFF 实现2.1

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字