泛函编程(5)-数据结构(Functional Data Structures)

     编程即是编制对数据进行运算的过程。特殊的运算必须用特定的数据结构来支持有效运算。如果没有数据结构的支持,我们就只能为每条数据申明一个内存地址了,然后使用这些地址来操作这些数据,也就是我们熟悉的申明变量再对变量进行读写这个过程了。试想想如果没有数据结构,那我们要申明多少个变量呢。所以说,数据结构是任何编程不可缺少的元素。

    泛函编程使用泛函数据结构(Functional Data Structure)来支持泛函程序。泛函数据结构的特点是”不可变特性“(Immutability), 是泛函编程中函数组合(composition)的必需。所以,与其它编程范畴不同,泛函编程的泛函数据结构必须具体一套特定的数据运算方式。

泛函数据结构及运算方法具备以下特征:

1、不可变特性(Immutable)

2、运算在数据结构内进行。尽量避免使用中间变量

3、运算返回新的数据结构作为结果

我们先看看熟悉的OOP数据运算风格:

1 scala> var arr = Array(1,2,3)
2 arr: Array[Int] = Array(1, 2, 3)
1 scala> var sum = arr(0)+arr(1)+arr(2)
2 sum: Int = 6

以上运算是需要中间变量的。而且是在结构外进行的:先把数据从地址读出再相加。

1 scala> arr(0) = sum
2 
3 scala> arr
4 res9: Array[Int] = Array(6, 2, 3)

直接赋值后arr内容变了。在这里arr是“可变的”(Mutable)数据结构。肯定的是如果下面需要再次使用arr时,我们是无法保证它内容一致性的。

再看看泛函风格:

 

1 scala> val arr = Array(1,2,3)
2 arr: Array[Int] = Array(1, 2, 3)
3 scala> val sum = arr.sum
4 sum: Int = 6

 

泛函运算直接在数据结构内进行,不需要中间变量。

1 scala> val arr1 = arr map { x => if(x == 1) sum else x }
2 arr1: Array[Int] = Array(6, 2, 3)
3 
4 scala> arr
5 res10: Array[Int] = Array(1, 2, 3)

arr1是赋值后新的数据结构。arr没有变化。这样我们可以放心使用arr来进行函数组合了。

可能这里会出现一些误解:arr1先复制了arr内的数据后再修改内容,所以arr没有变。这样理解有对也有不对:从效果来说arr1是复制了arr。但从具体做法上系统只是把arr(0)下面节点的指针指向了arr1(0),并没有进行实质的数据复制。

Python泛函编程是一种编程范式,它主要关注于使用高阶函数和不可变数据来构建程序。泛函编程的核心思想是将计算过程看作是函数之间的转换,通过组合和应用函数来解决问题。 在Python中,泛函编程可以通过以下几个方面来实现: 1. 高阶函数:Python中的函数是一等公民,可以作为参数传递给其他函数,也可以作为返回值。高阶函数可以接受一个或多个函数作为参数,并返回一个新的函数。常见的高阶函数包括map、filter和reduce等。 2. 匿名函数:Python中的lambda表达式可以用来创建匿名函数,这些函数通常用于简单的计算或作为其他函数的参数。 3. 不可变数据:在泛函编程中,数据被视为不可变的,即不能被修改。这样可以避免副作用,使得程序更加可靠和易于理解。 4. 列表推导式:列表推导式是一种简洁的语法,可以通过对一个列表进行转换或筛选来创建一个新的列表。它可以替代循环语句,使代码更加简洁和易读。 5. 函数组合:泛函编程鼓励将多个函数组合在一起,形成一个新的函数。这样可以将复杂的问题分解为多个简单的函数,提高代码的可读性和可维护性。 6. 惰性求值:泛函编程中的惰性求值指的是只在需要的时候才进行计算,可以提高程序的效率。Python中的生成器和迭代器就是惰性求值的一种实现方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值