成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python函數式編程:不可變數據結構

開發 后端
在這個由兩篇文章構成的系列中,我將討論如何將函數式編程方法論中的思想引入至 Python 中,來充分發揮這兩個領域的優勢。

 [[249083]]

不可變性可以幫助我們更好地理解我們的代碼。下面我將講述如何在不犧牲性能的條件下來實現它。

在這個由兩篇文章構成的系列中,我將討論如何將函數式編程方法論中的思想引入至 Python 中,來充分發揮這兩個領域的優勢。

本文(也就是***篇文章)中,我們將探討不可變數據結構的優勢。第二部分會探討如何在 toolz 庫的幫助下,用 Python 實現高層次的函數式編程理念。

為什么要用函數式編程?因為變化的東西更難推理。如果你已經確信變化會帶來麻煩,那很棒。如果你還沒有被說服,在文章結束時,你會明白這一點的。

我們從思考正方形和矩形開始。如果我們拋開實現細節,單從接口的角度考慮,正方形是矩形的子類嗎?

子類的定義基于里氏替換原則。一個子類必須能夠完成超類所做的一切。

如何為矩形定義接口?

  1. from zope.interface import Interface
  2.  
  3. class IRectangle(Interface):
  4.     def get_length(self):
  5.         """正方形能做到"""
  6.     def get_width(self):
  7.         """正方形能做到"""
  8.     def set_dimensions(self, length, width):
  9.         """啊哦"""

如果我們這么定義,那正方形就不能成為矩形的子類:如果長度和寬度不等,它就無法對 set_dimensions 方法做出響應。

另一種方法,是選擇將矩形做成不可變對象。

  1. class IRectangle(Interface):
  2.     def get_length(self):
  3.         """正方形能做到"""
  4.     def get_width(self):
  5.         """正方形能做到"""
  6.     def with_dimensions(self, length, width):
  7.         """返回一個新矩形"""

現在,我們可以將正方形視為矩形了。在調用 with_dimensions 時,它可以返回一個新的矩形(它不一定是個正方形),但它本身并沒有變,依然是一個正方形。

這似乎像是個學術問題 —— 直到我們認為正方形和矩形可以在某種意義上看做一個容器的側面。在理解了這個例子以后,我們會處理更傳統的容器,以解決更現實的案例。比如,考慮一下隨機存取數組。

我們現在有 ISquareIRectangle,而且 ISequereIRectangle 的子類。

我們希望把矩形放進隨機存取數組中:

  1. class IArrayOfRectangles(Interface):
  2.     def get_element(self, i):
  3.         """返回一個矩形"""
  4.     def set_element(self, i, rectangle):
  5.         """'rectangle' 可以是任意 IRectangle 對象"""

我們同樣希望把正方形放進隨機存取數組:

  1. class IArrayOfSquare(Interface):
  2.     def get_element(self, i):
  3.         """返回一個正方形"""
  4.     def set_element(self, i, square):
  5.         """'square' 可以是任意 ISquare 對象"""

盡管 ISquareIRectangle 的子集,但沒有任何一個數組可以同時實現 IArrayOfSquareIArrayOfRectangle.

為什么不能呢?假設 bucket 實現了這兩個類的功能。

  1. >>> rectangle = make_rectangle(3, 4)
  2. >>> bucket.set_element(0, rectangle) # 這是 IArrayOfRectangle 中的合法操作
  3. >>> thing = bucket.get_element(0) # IArrayOfSquare 要求 thing 必須是一個正方形
  4. >>> assert thing.height == thing.width
  5. Traceback (most recent call last):
  6.   File "<stdin>", line 1, in <module>
  7. AssertionError

無法同時實現這兩類功能,意味著這兩個類無法構成繼承關系,即使 ISquareIRectangle 的子類。問題來自 set_element 方法:如果我們實現一個只讀的數組,那 IArrayOfSquare 就可以是 IArrayOfRectangle 的子類了。

在可變的 IRectangle 和可變的 IArrayOf* 接口中,可變性都會使得對類型和子類的思考變得更加困難 —— 放棄變換的能力,意味著我們的直覺所希望的類型間關系能夠成立了。

可變性還會帶來作用域方面的影響。當一個共享對象被兩個地方的代碼改變時,這種問題就會發生。一個經典的例子是兩個線程同時改變一個共享變量。不過在單線程程序中,即使在兩個相距很遠的地方共享一個變量,也是一件簡單的事情。從 Python 語言的角度來思考,大多數對象都可以從很多位置來訪問:比如在模塊全局變量,或在一個堆棧跟蹤中,或者以類屬性來訪問。

如果我們無法對共享做出約束,那我們可能要考慮對可變性來進行約束了。

這是一個不可變的矩形,它利用了 attr 庫:

  1. @attr.s(frozen=True)
  2. class Rectange(object):
  3.     length = attr.ib()
  4.     width = attr.ib()
  5.     @classmethod
  6.     def with_dimensions(cls, length, width):
  7.         return cls(length, width)

這是一個正方形:

  1. @attr.s(frozen=True)
  2. class Square(object):
  3.     side = attr.ib()
  4.     @classmethod
  5.     def with_dimensions(cls, length, width):
  6.         return Rectangle(length, width)

使用 frozen 參數,我們可以輕易地使 attrs 創建的類成為不可變類型。正確實現 __setitem__ 方法的工作都交給別人完成了,對我們是不可見的。

修改對象仍然很容易;但是我們不可能改變它的本質。

  1. too_long = Rectangle(100, 4)
  2. reasonable = attr.evolve(too_long, length=10)

Pyrsistent 能讓我們擁有不可變的容器。

  1. # 由整數構成的向量
  2. a = pyrsistent.v(1, 2, 3)
  3. # 并非由整數構成的向量
  4. b = a.set(1, "hello")

盡管 b 不是一個由整數構成的向量,但沒有什么能夠改變 a 只由整數構成的性質。

如果 a 有一百萬個元素呢?b 會將其中的 999999 個元素復制一遍嗎?Pyrsistent 具有“大 O”性能保證:所有操作的時間復雜度都是 O(log n). 它還帶有一個可選的 C 語言擴展,以在“大 O”性能之上進行提升。

修改嵌套對象時,會涉及到“變換器”的概念:

  1. blog = pyrsistent.m(
  2.     title="My blog",
  3.     links=pyrsistent.v("github", "twitter"),
  4.     posts=pyrsistent.v(
  5.         pyrsistent.m(title="no updates",
  6.                      content="I'm busy"),
  7.         pyrsistent.m(title="still no updates",
  8.                      content="still busy")))
  9. new_blog = blog.transform(["posts", 1, "content"],
  10.                           "pretty busy")

new_blog 現在將是如下對象的不可變等價物:

  1. {'links': ['github', 'twitter'],
  2.  'posts': [{'content': "I'm busy",
  3.             'title': 'no updates'},
  4.            {'content': 'pretty busy',
  5.             'title': 'still no updates'}],
  6.  'title': 'My blog'}

不過 blog 依然不變。這意味著任何擁有舊對象引用的人都沒有受到影響:轉換只會有局部效果。

當共享行為猖獗時,這會很有用。例如,函數的默認參數:

  1. def silly_sum(a, b, extra=v(1, 2)):
  2.     extra = extra.extend([a, b])
  3.     return sum(extra)

在本文中,我們了解了為什么不可變性有助于我們來思考我們的代碼,以及如何在不帶來過大性能負擔的條件下實現它。下一篇,我們將學習如何借助不可變對象來實現強大的程序結構。

責任編輯:龐桂玉 來源: Linux中國
相關推薦

2022-07-13 16:38:32

Python可變數據類型不可變數據類型

2017-10-26 08:53:38

前端JavaScript函數式編程

2025-02-08 13:23:27

JavaScript開發喚醒鎖

2018-09-21 11:19:30

Lambda架構函數數據系統

2023-09-21 16:13:20

Python數據結構

2024-02-26 10:47:09

Python數據類型開發

2009-09-27 15:23:00

Scala講座函數式編程Scala

2024-04-02 08:00:00

函數式編程開發

2023-10-20 08:18:17

Python數據類型

2023-12-14 15:31:43

函數式編程python編程

2020-01-21 08:27:50

代碼開發Python

2024-04-08 07:58:11

Python數據類型字符串

2024-01-15 07:15:05

函數式編程代碼

2024-03-08 13:06:31

Array數組JavaScrip

2021-07-16 07:57:34

Python數據結構

2021-03-09 06:30:32

JAVA數據結構算法

2021-03-18 08:44:20

Java數據結構算法

2021-04-13 09:37:41

Java數據結構算法

2019-04-14 22:22:28

Python數據結構算法

2021-06-14 09:34:23

對象存儲存儲
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 四虎成人免费视频 | 午夜精品91 | 免费亚洲婷婷 | 免费黄色的视频 | 在线一区二区国产 | 99成人| 91精品在线观看入口 | 亚洲一区二区免费视频 | 日韩在线视频免费观看 | 美国一级毛片a | 久久四虎| 欧美成人激情 | 欧美日韩一 | 国产精品a级 | 91成人在线 | 天堂免费 | 免费在线看a | 色综合色综合网色综合 | 日本欧美在线 | 久久91精品 | 精品乱码一区二区三四区 | 北条麻妃99精品青青久久主播 | 99免费视频 | 成人精品一区二区三区四区 | 99精品久久久久久久 | 久久这里只有精品首页 | 污视频在线免费观看 | 日韩欧美中文 | 国产精品a久久久久 | 欧美日韩91| 日本在线你懂的 | 国产精品一区二区久久久久 | 在线一区二区三区 | 亚洲精品www| 成人夜晚看av | 99久久精品一区二区成人 | 久久99精品久久久久久 | 黄色一级免费观看 | 亚洲国产视频一区二区 | 欧美一级做性受免费大片免费 | 欧美一区二区三区在线观看 |