这篇文章主要介绍了汇编语言的函数式编程实例分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇汇编语言的函数式编程实例分析文章都会有所收获,下面我们一起来看看吧。
要了解函数式编程,我们需要首先了解函数。 这听起来可能很无聊,但总而言之,它很有见地。
简单地说,函数是将输入转换为输出的东西。只是事情并没有那么简单。思考一下,在Python
中的下面这个函数的意义:
def square(x): return x*x
这个函数很简单。 它需要一个变量x
,可能是一个int
,或者是一个 float
或 double
,然后输出该变量的平方。
再思考一下下面的这个函数:
global_list = [] def append_to_list(x): global_list.append(x)
乍一看,这个函数接受了一个变量 x
,无论是哪种类型,由于没有 return
语句,它什么也不返回。事实真的是这样吗?
如果事先没有定义 global_list
,那么这个函数就不能工作,它的输出是相同的列表,尽管经过了修改。虽然 global_list
没有声明输入,但当我们使用该函数时,它就会发生变化:
append_to_list(1) append_to_list(2) global_list
它返回了 [1,2]
,而不是空列表。这可能就是问题所在,列表确实是函数的一个输入,虽然我们没有明确说明。
这些隐含的输入,或者其他情况下的输出,有一个官方名称:副作用。虽然我们只列举了一个简单的例子,但在更复杂的程序中,这些可能会让我们面临真正的困难。
大家可以思考一下该如何测试 append_to_list
:我们不仅需要阅读第一行并使用任何 x
来测试函数,还需要阅读整个定义,了解其作用,定义 global_list
并以这种方式进行测试。这个例子告诉我们,当你在处理有数千行代码的程序时,简单的东西很快就会变得乏味。
好消息是,有一个简单的解决方法:对函数作为输入的内容诚实。这样更好:
newlist = [] def append_to_list2(x, some_list): some_list.append(x) append_to_list2(1,newlist) append_to_list2(2,newlist) newlist
我们并没有作太大的改变,输出结果仍然是 [1,2]
,其他所有内容也保持不变。
但是,我们已经更改了一件事情:该代码现在没有副作用。
现在,当我们查看函数声明时,能确切知道发生了什么。如果程序运行不正常,我们也可以轻松地单独测试每个功能并查明哪个功能有问题。
具有明确声明的输入和输出的函数是没有副作用的函数,而没有副作用的函数就是纯函数。
函数编程的一个非常简单的定义是:仅用纯函数编写程序。纯函数永远不会修改变量,只会创建新的变量作为输出。
此外,对于给定输入的纯函数,我们可以得到特定的输出。相反,不纯函数可能依赖于某些全局变量。因此,如果全局变量不同,则相同的输入变量可能导致不同的输出。后者会让调试和代码维护变得更加困难。
这里有一个容易发现副作用的简单规则:由于每个函数必须具有某种输入和输出,因此没有任何输入或输出的函数声明必须是不纯的。如果采用函数式编程,这是你可能想要更改的第一个声明。
循环不是函数式编程中的东西。首先,我们先来思考以下的Python
循环:
integers = [1,2,3,4,5,6] odd_ints = [] squared_odds = [] total = 0 for i in integers: if i%2 ==1 odd_ints.append(i) for i in odd_ints: squared_odds.append(i*i) for i in squared_odds: total += i
相较于我们要执行的简单操作,以上代码明显过长。而且也没有起到作用,因为我们正在修改全局变量。
相反,我们可以用以下代码替代:
from functools import reduce integers = [1,2,3,4,5,6] odd_ints = filter(lambda n: n % 2 == 1, integers) squared_odds = map(lambda n: n * n, odd_ints) total = reduce(lambda acc, n: acc + n, squared_odds)
这是完整的函数式。它比较短,也更快,因为我们不需要迭代太多的数组元素。如果你理解 filter
, map
和 reduce
如何工作,代码也就不难理解了。
这并不意味着所有的函数代码都使用 map
、reduce
等。这也不意味着你需要函数式编程来理解 map
和 reduce
。只是当你抽象循环时,这些函数会弹出很多。
在谈到函数式编程的历史时,许多人都是从lambda
函数的发明开始的。 尽管 lambda
是函数式编程毫无疑问的基石,但它们并不是根本原因。
Lambda
函数是可用于使程序起作用的工具。 但是,我们也可以在面向对象的编程中使用lambda
。
上面的示例虽然不是静态类型的,但是它依然是函数式的。
即使静态类型为我们的代码增加了一层额外的安全保护,但是其函数正常也并非必不可少。 不过,这可能是一个不错的补充。
Perl
对副作用的处理方法与大多数编程语言截然不同。它包含了一个神奇的参数 $\
。Perl
确实有它的优点,但我不会用它进行函数式编程。
如果你在用 Java
进行函数式编程,那我只能祝你好运了。因为你的程序有一半是由静态关键字组成的,而且其他 Java
开发人员也会把你的程序视为耻辱。
这并不是说 Java
有多糟糕,而是因为它并不是为那些用函数式编程解决问题而设计的,比如数据库管理或机器学习应用程序。
有趣的是:Scala
的目标是统一面向对象和函数式编程。如果你觉得这有点奇怪,那你不是一个人,因为所有人都这么觉得:函数式编程的目标是完全消除副作用,而面向对象编程是把副作用保留在对象内部。
尽管如此,很多开发人员认为 Scala
是一种帮助他们从面向对象编程过渡到函数式编程的语言。或许在未来几年里,它们会更容易全面发挥作用。
Python
鼓励函数式编程。一个事实就能看到这一点:每个函数在默认情况下至少有一个输入self
。这很像Python
的禅:显式比隐式好!
据它的创建者说,Clojure
大约有 80% 是函数式编程。默认情况下,所有值都是不可变的,就像在函数式编程中需要它们一样。但是,我们可以通过在这些不可变的值周围使用可变值包装器来解决这个问题。当你打开这样一个包装,你得到的东西又是不变的。
这是为数不多的纯函数式和静态类型的语言之一。虽然在开发过程中这看起来像是一个时间消耗器,但在调试程序时,Haskell
会付出巨大的代价。它不像其他语言那么容易学,但绝对值得投资!
关于“汇编语言的函数式编程实例分析”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“汇编语言的函数式编程实例分析”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注亿速云行业资讯频道。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。