函数式编程的思维转变从数据参数化到行为参数化
为什么要了解函数式编程;函数式编程对哪些设计模式产生了影响;如何从数据参数化过渡到行为参数化;策略模式的三种实现方式。
极客架构师专注架构师成长。
大家好,我是码农老吴。
今天是我的《Java极客》系列分享的第一期。
今天也是2022年的最后一天,经过多天的病毒折磨,刀片桑,咳嗽,浑身无力,都体验了一遍,终于在兔年到来之前,满血复活了。感谢大家的关心,提前一天,祝大家新年快乐,身体健康,职场顺利,万事大吉。
兔年快乐
古朗月行
【作者】李白
小时不识月,呼作白玉盘。
又疑瑶台镜,飞在青云端。
仙人垂两足,桂树何团团。
白兔捣药成,问言与谁餐。
蟾蜍蚀圆影,大明夜已残。
羿昔落九乌,天人清且安。
阴精此沦惑,去去不足观。
忧来其如何,凄怆摧心肝。分享思路
《Java极客》分享内容
为什么要分享函数式编程
函数式编程包含哪些内容
案例说明电商后台的商品过滤功能
版本1硬编码
版本2数据参数化
版本3方案1串行化的多条件过滤
版本3方案2并行化的多条件过滤
走出思维死角行为参数化
版本4基于策略模式
版本5基于匿名类的策略模式
版本6基于Lambda表达式的策略模式
下期预告
参考书籍《Java极客》分享内容
《Java极客》这个系列,我将分享Java开发相关的几个比较重要的,有一定难度的专题内容,内容主要涉及:函数式编程,泛型编程,并发编程,JVM等。为什么要分享函数式编程
一个最直接的原因,就是因为函数式编程,对我们目前正在分享的设计模式,其中的好几个设计模式,都产生了比较大的影响,比如策略模式,模板方法模式,观察者模式,责任链模式,工厂模式等等。为了方便大家,更好的学习函数式编程下的设计模式,我决定把Java中的函数式编程功能,系统的分享一下。函数式编程包含哪些内容
众所周知,Java语言并不是天生的函数式编程语言,只是在新的时代需求下(大数据分析和CPU多核时代),Java语言,特别是Java8,发展出来的,支持函数式编程的新功能。
内容主要包括:行为参数化,Lambda表达式,方法引用,流操作等等。
今天,我们就从行为参数化开始。案例说明电商后台的商品过滤功能
电商后台,难免对商品进行各种查询,统计,大多数情况下,通过数据库就可以完成,如果数据量非常大的情况下,还会用上大数据系统。当然,如果具体到店铺级别,一般商品的数量是可控的,量级不会太大(10万级别,不超过百万),直接使用Java集合框架,在内存中直接对商品进行查询,过滤,统计等操作,也是比较简洁,高效的。由于函数式编程的流操作,主要是通过集合框架进行数据操作的。所以我们主要以集合框架对数据进行分析。
商品实体类类图及代码如下:商品(SKU)类图
packagecom。geekarchitect。javageek。module001。demo01;importlombok。Data;author极客架构师吴念createTime20221229DatapublicclassSKU{privateLongid;privateStringname;privateLongcategoryId;privateStringcategoryName;privateDoubleprice;privateLongshopId;privateStringshopName;privateintsales;销售量privateintstock;库存量publicSKU(){}publicSKU(Longid,Stringname,LongcategoryId,StringcategoryName,Doubleprice,LongshopId,StringshopName,intsales,intstock){this。idid;this。namename;this。categoryIdcategoryId;this。categoryNamecategoryName;this。priceprice;this。shopIdshopId;this。shopNameshopName;this。salessales;this。stockstock;}}
模拟数据如下:packagecom。geekarchitect。javageek。module001。demo01;importorg。assertj。core。util。Lists;importjava。util。List;author极客架构师吴念createTime20221229publicabstractclassBaseTest{ListSKUgeneratorSKU(){SKUsku01newSKU(1L,华为MateBookX,1L,笔记本,10000D,1L,华为旗舰专卖店,1000,1000);SKUsku02newSKU(2L,华为MateBookpro,1L,笔记本,11000D,1L,华为旗舰专卖店,1000,1000);SKUsku03newSKU(3L,华为MateBookD,1L,笔记本,12000D,1L,华为旗舰专卖店,1000,1000);SKUsku04newSKU(4L,HUAWEIMate50,2L,手机,12000D,1L,华为旗舰专卖店,1000,1000);SKUsku05newSKU(5L,HUAWEIMate50PRO,2L,手机,12000D,1L,华为旗舰专卖店,1000,1000);SKUsku06newSKU(6L,HUAWEIMate50E,2L,手机,12000D,1L,华为旗舰专卖店,1000,1000);returnLists。newArrayList(sku01,sku02,sku03,sku04,sku05,sku06);}}
需求1查询商品分类属于笔记本的商品
也就是商品的categoryName属性,属性值为笔记本的商品。版本1硬编码SKUService
这版代码,是需求的最直接的体现,整个方法,入参sourceSkuList中包含的是待查询的商品信息,返回值filteredSkuList集合中,包含了符合查询条件的商品。
如果商品的CategoryNameequals笔记本,就把它添加到结果集合中。
很简单吗,但是除了初学者,大家一般不会这么编码,至少大家能想到,这次要查询笔记本,下次可能查询的就是别的了,所以笔记本不能硬编码,应该提取为一个参数。
这就要看下版代码了。
测试代码SKUServiceTest
运行结果
版本2数据参数化SKUService
这版代码,比较符合实际工作中的常见编程习惯。把要查询的条件,设置为参数,至少能适应一定的需求变化,但是如果需求查询的不是商品的分类(categoryName),而是别是什么呢,比如价格,我们该如何应对呢。
测试代码SKUServiceTest
运行结果
略需求2查询商品分类属于笔记本,价格大于10000的商品
这次我们的查询条件有两个,既要是笔记本,又要是价格大于1万。如何实现这个需求呢,对于两个查询条件。我们该如何处理呢,至少可以有两个思路。版本3方案1串行化的多条件过滤SKUService
这版本代码,是大多数人能想到的解决方案,两个查询条件,就定义两个参数,过滤的时候,两个条件先后执行。在大多数的情况下,这个方案没有太大问题,但是在多核时代下,这种方案的局限性就很明显,因为不论有多少个条件,每个商品,都必须串行的执行所有的查询条件。如果要发挥多核时代,CPU的性能,就需要尝试下面的思路了(注意,这种解决方案,如果非要支持并发执行也不是不可能,只是相对下面的解决方案,不太方便罢了)
测试代码SKUServiceTest
版本3方案2并行化的多条件过滤SKUService
多个查询条件,如果要实现并行化的处理,每个查询条件可以独立进行数据过滤。所以,我们实现了两个数据过滤的方法。
测试代码SKUServiceTest
在我们的测试代码中,我并没有真的并行执行这两个查询条件,而是一个执行完之后,再接着执行另外一个,从性能上看,应该还不如方案1,但是它提供了一种可能性,一种并发处理的可能性。我们自己如果要实现并行化的处理,要么直接编写多线程代码,要么使用线程池,代码都简单不了(我在并发编程的系列分享中,会有相关分享)。后面我们要分享的并行化的流操作,会直接帮助我们完成这个工作。
走出思维死角
前面我们对于两个需求,三个版本的解决方案,里面都存在一个思维定势,那就是数据参数化,我们要根据商品分类和价格查询商品,我们的参数,传递的就是商品分类的名称和价格这两个参数。我们能不能走出这个思维死角,换个思路呢。
我们的查询条件是,商品分类的名称是否等于笔记本,价格是否大于1万,前面我们的目光都聚焦在笔记本和1万这些数据上,而忽略了是否等于,是否大于这两个词,它们代表的是动作,是行为。我们能不能更直接一些,把行为参数化呢。
数据参数化,大家比较好理解,而行为参数化,在面向对象的Java语言中,不太直接,如果我们要传递一个行为(或者说传递一个方法),就只能通过传递对象,来间接的传递方法。
我们看下面的方案。版本4基于策略模式
接口及类
SKUFilterStrategy:商品过滤接口
CategoryNameFilterStrategy:根据商品分类过滤商品
PriceGreaterThanFilterStrategy:根据价格过滤商品
SKUServiceV2:商品过滤服务类
SKUFilterStrategy接口:商品过滤策略接口,返回值为boolean,表示商品是否符合我们的过滤条件。
CategoryNameFilterStrategy:商品过滤策略,判断商品分类是否属于笔记本。
PriceGreaterThanFilterStrategy:商品过滤策略,判断商品价格是否大于1万。
SKUServiceV2:商品过滤服务类
filterSKUByStrategy()方法,根据商品过滤策略,对集合中的商品进行过滤。这里使用了策略模式,使得过滤商品的策略,可以无限的扩展,而这个方法,不需要进行任何修改,真正的符合开闭原则,对扩展开放,对修改关闭。
还有一点,很重要的就是,我们传递的参数SKUFilterStrategyskuFilterStrategy,它是一个对象,而我们在方法中使用的,是这个对象的filter方法。这就是在Java这种面向对象的语言中,需要传递方法(行为)的一种间接的方案,通过传递不同的对象,来实现传递不同的方法。
有没有更直接的方案呢,我们前面定义的CategoryNameFilterStrategy和PriceGreaterThanFilterStrategy,看起来有些大材小用,或者说有些繁琐。
能不能更简单一些,在Java8以前,函数式编程还没有出来之前,我们还有一种选择,那就是匿名类了。author极客架构师吴念createTime20221229publicclassSKUServiceV2{privatestaticfinalLoggerLOGLoggerFactory。getLogger(SKUServiceV2。class);publicListSKUfilterSKUByStrategy(ListSKUsourceSkuList,SKUFilterStrategyskuFilterStrategy){ListSKUfilteredSkuListnewArrayList();for(SKUsku:sourceSkuList){if(skuFilterStrategy。filter(sku)){filteredSkuList。add(sku);}}returnfilteredSkuList;}}测试代码
运行结果
版本5基于匿名类的策略模式
这次我们要使用基于匿名类的策略模式,所以只定义了策略接口,策略实现类一个都没有。测试代码
由于商品的过滤条件,一般都是临时性的,我们专门定义一个类,有些累赘,而直接使用匿名类,则更直接一些。
但是,大家有没有发现,即使使用了匿名类,代码好像也没简化多少,只不过少了个类名罢了。能不能更简单一些,下面我们的主角就要上场了。
运行结果
版本6基于Lambda表达式的策略模式
这次我们要使用Java函数式编程中的lambda表达式,来实现策略模式,策略接口也是不能少的,策略实现类当然,也不需要。测试代码
我们的主角登场了,使用了Lambda表达式,代码精简了不少,所有的官样代码都没有了。而且最为重要的一点,就是我们实现了真正的行为参数化,我们在参数中,直接传递的就是方法中的代码片段。而没有传递对象。这就是函数式编程的一个重要特点。那么这里的代码,为什么要这么写,语法是什么样的,还可以怎么写,这就是我们下一期要详细讲解的内容了。
运行结果
下期预告
本期我们从数据参数化到行为参数化,从普通的策略模式,到匿名类的策略模式,再到基于Lambda表达式的策略模式,带领大家走进了Java函数式编程的大门,下期,我们就详细的聊聊Lambda表达式的语法规则,应用技巧以及它的底层原理。参考书籍
《Java8inAction》
《OnJava8》
1300万卢布奖金!谢尔巴科娃拼到鼻子出血,18岁生日拿冠军1300万卢布奖金!谢尔巴科娃18岁生日拿冠军,千金美艳不可方物北京时间2022年3月28日星期一消息。近日,谢尔巴科娃、瓦利耶娃参加了1TV杯,结果在跳跃赛和短节目自由……
欧洲小镇系列希腊卡斯特拉基镇,欧洲修行界的少林寺15世纪的欧洲是大航海时代,也是经济腾飞的重要时间节点,同时也是奥斯曼帝国的高光时期,疆域一度拓展至欧洲腹地,使得欧洲各个王室贵族人人自危,为避免财富被掠夺瓜分,他们在欧洲各地……
我给周总理当翻译作者:张正纲来源:《外交官说事儿》一、和陈老总换座位1964年春节,我从工作了四年的驻尼泊尔使馆回国。我做梦也没想到,回国后竟能给周总理做翻译。1965年8月,尼泊……
苏秦游说六国当初,洛阳人苏秦曾向秦王进献兼并天下的计划,却没有得到秦王的采纳,苏秦只好离去。后来,苏秦又到燕国,他对燕文公说:燕国之所以不遭受侵犯和掠夺,是因为南面有赵国做屏障。秦国要想攻……
这事办好了800元旅游订金已退华声在线全媒体记者欧金玉两年多以前交的800元旅游订金迟迟拿不回来,3月15日,消费者小古将此情况向新湖南客户端湘问频道投诉后,订金很快被退还。小古一直渴望前往张家……
胡雪岩勿近白虎是什么意思?1885年,曾富可敌国的胡雪岩,强撑着用尽最后一口气说出了白虎可怕,勿近白虎的遗言后,就含恨闭上了双眼。胡雪岩的一生波澜壮阔,他被称为一代商业奇才,但没想到在晚年却屡遭磨……
就夸解放军!你来跟我杠?每每谈到解放军的战绩的时候,总有人站出来说:还有金门呢?看过解放军军史的都知道,金门一战损失九千多,可惜的很。金门战役是解放军组织的第一次渡海登岛战役,但是从上到下没有特……
皮肤衰老的5大元凶,及时改正,或可延缓衰老,人显年轻与时间和地心引力抗衡,实现冻龄,应该是每位爱美女性的梦想。谁都不想成为黄脸婆,可是我们经常会出现各种各样的皮肤问题,爬上眼角的第一根皱纹、日益明显的法令线、逐年粗大的毛孔,这些……
毛主席转战陕北之六十四毛主席发布新的动员令一九四七年八月的月底,佳县朱官寨天气特别好。蓝天白云下面,漫山遍野的糜子谷子等即将成熟,阳畖四周的树木根深叶茂。金风送来丰收的喜讯,毛主席带来了前所未有的祥瑞。全国各个战……
中国首创大型民俗旅游区,被欧阳修等名人赞誉,就在世界水电之都说起千湖之省的湖北,那可谓是钟灵毓秀的江湖宝地。这里历史悠久、风光奇秀,山水相依,名胜遍野,还有灿烂辉煌的荆楚文化,以及琳琅满目的美食小吃。作为蜚声中外的旅游大省,到湖北旅行很……
老蒋的女婿,双面特工,暮年谢绝接受岳母巨额遗产如果说有这么一位传奇人物,在抗日战争时期和日本人打得火热,后来解放战争时期在国民党内部算得上是位高权重,最后还娶了蒋介石的女儿,这样的人却是一位坚定不移的共产主义支持者,你会相……
冬季手足冰冷,试试羊肉药膳来源:【大河健康报】天寒地冻,人体阳气潜藏于体内,容易出现手足冰冷,气血循环不畅的情况。羊肉甘热,具有补肾壮阳、暖中祛寒、温补气血、开胃健脾的作用。冬天吃羊肉……