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 中的 CAST 函数详解及常见用法

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

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

MySQL 中 ROW_NUMBER() 函数最佳实践

《MySQL中ROW_NUMBER()函数最佳实践》MySQL中ROW_NUMBER()函数,作为窗口函数为每行分配唯一连续序号,区别于RANK()和DENSE_RANK(),特别适合分页、去重... 目录mysql 中 ROW_NUMBER() 函数详解一、基础语法二、核心特点三、典型应用场景1. 数据分

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Python get()函数用法案例详解

《Pythonget()函数用法案例详解》在Python中,get()是字典(dict)类型的内置方法,用于安全地获取字典中指定键对应的值,它的核心作用是避免因访问不存在的键而引发KeyError错... 目录简介基本语法一、用法二、案例:安全访问未知键三、案例:配置参数默认值简介python是一种高级编