Hướng dẫn python pass lambda as argument - python vượt qua lambda làm đối số

Hàm Lambda (hay chính xác hơn là biểu thức Lambda) chỉ đơn giản là một hàm bạn có thể xác định tại chỗ, ngay tại nơi bạn cần. Ví dụ,

Show
f = lambda x: x * 2

hoàn toàn giống như

def f(x):
    return x * 2

Và khi tôi nói chính xác, ý tôi là nó - họ tháo rời với cùng một mã byte. Sự khác biệt duy nhất giữa hai là cái được xác định trong ví dụ thứ hai có tên.

Biểu thức lambda trở nên hữu ích vì việc tạo ra một biểu thức không phải là một tuyên bố, điều đó có nghĩa là, như những người khác đã trả lời, bạn có thể làm

print iterator(lambda x: x / 4 + 12, 100, 5)

để có được chính xác những gì bạn muốn.

Tuy nhiên, sự khác biệt chính giữa các biểu thức Lambda và các hàm chính quy là Lambdas bị hạn chế hơn. Lambdas chỉ có thể chứa các biểu thức, không phải câu lệnh. Một biểu thức là bất cứ điều gì bạn có thể đặt ở phía bên phải của bài tập

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
2. .

Điều này có nghĩa là một biểu thức lambda không thể gán cho một biến (trên thực tế, nó không thể có các biến cục bộ, ngoài các tham số của nó). Nó không thể in (trừ khi nó gọi một chức năng khác). Nó không thể có một vòng lặp, vòng lặp trong thời gian, một bài kiểm tra (trừ toán tử ternary

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
3) hoặc thử/ngoại trừ khối.

Nếu bạn cần phải làm bất kỳ điều đó, chỉ cần xác định một chức năng thông thường. Trên thực tế, bất cứ khi nào bạn nghĩ rằng bạn muốn sử dụng một lambda, hãy nghĩ hai lần. Mã sẽ không dễ đọc hơn nếu bạn sử dụng chức năng thông thường? Đó không phải là biểu thức Lambda mà bạn muốn sử dụng lại ở một nơi khác trong mã của bạn?

Cuối cùng, luôn luôn làm những gì dẫn đến mã dễ đọc và có thể duy trì nhất. Không có sự khác biệt giữa Lambdas và các chức năng bình thường khi có liên quan đến hiệu suất.

Xem bây giờ hướng dẫn này có một khóa học video liên quan được tạo bởi nhóm Python thực sự. Xem nó cùng với hướng dẫn bằng văn bản để hiểu sâu hơn về sự hiểu biết của bạn: Cách sử dụng các chức năng của Python Lambda This tutorial has a related video course created by the Real Python team. Watch it together with the written tutorial to deepen your understanding: How to Use Python Lambda Functions

Tôi có thể chuyển một chức năng như một đối số trong Python không?

Python Lambdas là rất ít, các hàm ẩn danh, chịu một cú pháp hạn chế hơn nhưng ngắn gọn hơn so với các hàm trăn thông thường.

Đến cuối bài viết này, bạn sẽ biết:

  • Python Lambdas đã trở thành như thế nào
  • Cách lambdas so sánh với các đối tượng chức năng chính quy
  • Cách viết các chức năng Lambda
  • Chức năng nào trong Thư viện tiêu chuẩn Python Tận dụng Lambdas
  • Khi nào nên sử dụng hoặc tránh các chức năng của Python Lambda

Hướng dẫn này chủ yếu dành cho các lập trình viên Python trung bình đến có kinh nghiệm, nhưng nó có thể truy cập được cho bất kỳ tâm trí tò mò nào với sự quan tâm đến lập trình và tính toán Lambda.

Tất cả các ví dụ bao gồm trong hướng dẫn này đã được thử nghiệm với Python 3.7.

Tính toán Lambda

Các biểu thức Lambda trong Python và các ngôn ngữ lập trình khác có nguồn gốc từ Lambda Tính toán, một mô hình tính toán được phát minh bởi Alonzo Church. Bạn sẽ phát hiện ra khi tính toán Lambda được giới thiệu và tại sao nó lại là một khái niệm cơ bản kết thúc trong hệ sinh thái Python.

Lịch sử

Nhà thờ Alonzo chính thức hóa Tính toán Lambda, một ngôn ngữ dựa trên sự trừu tượng thuần túy, vào những năm 1930. Các chức năng của Lambda cũng được gọi là Trừu tượng Lambda, một tài liệu tham khảo trực tiếp đến mô hình trừu tượng của sáng tạo ban đầu của Alonzo Church.

Tính toán Lambda có thể mã hóa bất kỳ tính toán nào. Nó hoàn thành, nhưng trái với khái niệm về một cỗ máy Turing, nó là tinh khiết và không giữ bất kỳ trạng thái nào.

Các ngôn ngữ chức năng có nguồn gốc từ logic toán học và tính toán Lambda, trong khi các ngôn ngữ lập trình bắt buộc nắm lấy mô hình tính toán dựa trên trạng thái được phát minh bởi Alan Turing. Hai mô hình tính toán, máy tính Lambda và máy Turing, có thể được dịch sang nhau. Sự tương đương này được gọi là giả thuyết của nhà thờ.

Các ngôn ngữ chức năng trực tiếp thừa hưởng triết lý tính toán Lambda, áp dụng một cách tiếp cận tuyên bố của lập trình nhấn mạnh sự trừu tượng, chuyển đổi dữ liệu, thành phần và độ tinh khiết (không có trạng thái và không có tác dụng phụ). Ví dụ về các ngôn ngữ chức năng bao gồm Haskell, Lisp hoặc Erlang.

Ngược lại, máy Turing đã dẫn đến lập trình bắt buộc được tìm thấy bằng các ngôn ngữ như Fortran, C hoặc Python.

Phong cách bắt buộc bao gồm lập trình với các tuyên bố, thúc đẩy luồng của chương trình từng bước với các hướng dẫn chi tiết. Cách tiếp cận này thúc đẩy đột biến và yêu cầu quản lý trạng thái.

Sự tách biệt trong cả hai gia đình trình bày một số sắc thái, vì một số ngôn ngữ chức năng kết hợp các tính năng bắt buộc, như OCAML, trong khi các đặc điểm chức năng đã thấm vào gia đình ngôn ngữ bắt buộc nói riêng với việc giới thiệu các hàm Lambda trong Java hoặc Python.

Python vốn không phải là ngôn ngữ chức năng, nhưng nó đã áp dụng một số khái niệm chức năng từ rất sớm. Vào tháng 1 năm 1994,

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4,
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5,
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
6 và toán tử
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7 đã được thêm vào ngôn ngữ.

Ví dụ đầu tiên

Dưới đây là một vài ví dụ để cung cấp cho bạn sự thèm ăn cho một số mã python, kiểu chức năng.

Hàm nhận dạng, một hàm trả về đối số của nó, được biểu thị bằng định nghĩa hàm Python tiêu chuẩn bằng cách sử dụng từ khóa

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
8 như sau:

>>>

>>> def identity(x):
...     return x

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
9 lấy một đối số
def add_one(x):
    return x + 1
0 và trả lại khi gọi.

Ngược lại, nếu bạn sử dụng cấu trúc Python Lambda, bạn sẽ nhận được những điều sau:

Trong ví dụ trên, biểu thức bao gồm:

  • Từ khóa:
    >>> add_one = lambda x: x + 1
    >>> add_one(2)
    3
    
    7
    >>> add_one = lambda x: x + 1
    >>> add_one(2)
    3
    
    7
  • Một biến ràng buộc:
    def add_one(x):
        return x + 1
    
    0
    def add_one(x):
        return x + 1
    
    0
  • Một cơ thể:
    def add_one(x):
        return x + 1
    
    0
    def add_one(x):
        return x + 1
    
    0

Bạn có thể viết một ví dụ được xây dựng hơn một chút, một hàm thêm

def add_one(x):
    return x + 1
4 vào một đối số, như sau:

Bạn có thể áp dụng chức năng ở trên cho một đối số bằng cách xung quanh hàm và đối số của nó với dấu ngoặc đơn:

>>>

>>> (lambda x: x + 1)(2)
3

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
9 lấy một đối số
def add_one(x):
    return x + 1
0 và trả lại khi gọi.

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3

Ngược lại, nếu bạn sử dụng cấu trúc Python Lambda, bạn sẽ nhận được những điều sau:

>>>

>>> add_one = lambda x: x + 1
>>> add_one(2)
3

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
9 lấy một đối số
def add_one(x):
    return x + 1
0 và trả lại khi gọi.

def add_one(x):
    return x + 1

Ngược lại, nếu bạn sử dụng cấu trúc Python Lambda, bạn sẽ nhận được những điều sau:

>>>

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
9 lấy một đối số
def add_one(x):
    return x + 1
0 và trả lại khi gọi.

Chức năng ẩn danh

Các thuật ngữ sau đây có thể được sử dụng thay thế cho nhau tùy thuộc vào loại ngôn ngữ lập trình và văn hóa:

  • Chức năng ẩn danh
  • Chức năng Lambda
  • Biểu thức Lambda
  • Lambda trừu tượng
  • Hình thức Lambda
  • Chức năng nghĩa đen

Đối với phần còn lại của bài viết này sau phần này, bạn sẽ thấy thuật ngữ chức năng Lambda.lambda function.

Theo nghĩa đen, một hàm ẩn danh là một hàm không có tên. Trong Python, một hàm ẩn danh được tạo bằng từ khóa

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7. Một cách lỏng lẻo hơn, nó có thể hoặc không được gán một tên. Hãy xem xét một hàm ẩn danh hai đối tượng được xác định với
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7 nhưng không bị ràng buộc với một biến. Lambda không được đặt tên:

>>>

>>> lambda x, y: x + y

Hàm ở trên xác định một biểu thức Lambda có hai đối số và trả về tổng của chúng.

Khác với việc cung cấp cho bạn phản hồi rằng Python hoàn toàn ổn với hình thức này, nó không dẫn đến bất kỳ việc sử dụng thực tế nào. Bạn có thể gọi chức năng trong trình thông dịch Python:

Ví dụ trên là tận dụng tính năng chỉ dành cho trình thông dịch tương tác được cung cấp thông qua dấu gạch dưới (

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3). Xem ghi chú dưới đây để biết thêm chi tiết.

Bạn không thể viết mã tương tự trong mô -đun Python. Hãy xem xét

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3 trong phiên dịch viên là một tác dụng phụ mà bạn đã tận dụng. Trong một mô -đun Python, bạn sẽ gán một tên cho Lambda hoặc bạn sẽ chuyển Lambda cho một chức năng. Bạn sẽ sử dụng hai cách tiếp cận đó sau trong bài viết này.

Một mẫu khác được sử dụng trong các ngôn ngữ khác như JavaScript là ngay lập tức thực hiện hàm Python Lambda. Điều này được gọi là một biểu thức chức năng được gọi ngay lập tức (iife, phát âm là iff iffy). Đây là một ví dụ:Immediately Invoked Function Expression (IIFE, pronounce “iffy”). Here’s an example:

>>>

def f(x):
    return x * 2
0

Hàm ở trên xác định một biểu thức Lambda có hai đối số và trả về tổng của chúng.

Khác với việc cung cấp cho bạn phản hồi rằng Python hoàn toàn ổn với hình thức này, nó không dẫn đến bất kỳ việc sử dụng thực tế nào. Bạn có thể gọi chức năng trong trình thông dịch Python:

Ví dụ trên là tận dụng tính năng chỉ dành cho trình thông dịch tương tác được cung cấp thông qua dấu gạch dưới (

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3). Xem ghi chú dưới đây để biết thêm chi tiết.

Bạn không thể viết mã tương tự trong mô -đun Python. Hãy xem xét

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3 trong phiên dịch viên là một tác dụng phụ mà bạn đã tận dụng. Trong một mô -đun Python, bạn sẽ gán một tên cho Lambda hoặc bạn sẽ chuyển Lambda cho một chức năng. Bạn sẽ sử dụng hai cách tiếp cận đó sau trong bài viết này.

Một mẫu khác được sử dụng trong các ngôn ngữ khác như JavaScript là ngay lập tức thực hiện hàm Python Lambda. Điều này được gọi là một biểu thức chức năng được gọi ngay lập tức (iife, phát âm là iff iffy). Đây là một ví dụ:

>>>

def f(x):
    return x * 2
1

Hàm ở trên xác định một biểu thức Lambda có hai đối số và trả về tổng của chúng.

Khác với việc cung cấp cho bạn phản hồi rằng Python hoàn toàn ổn với hình thức này, nó không dẫn đến bất kỳ việc sử dụng thực tế nào. Bạn có thể gọi chức năng trong trình thông dịch Python:

Ví dụ trên là tận dụng tính năng chỉ dành cho trình thông dịch tương tác được cung cấp thông qua dấu gạch dưới (

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3). Xem ghi chú dưới đây để biết thêm chi tiết.

Bạn không thể viết mã tương tự trong mô -đun Python. Hãy xem xét

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3 trong phiên dịch viên là một tác dụng phụ mà bạn đã tận dụng. Trong một mô -đun Python, bạn sẽ gán một tên cho Lambda hoặc bạn sẽ chuyển Lambda cho một chức năng. Bạn sẽ sử dụng hai cách tiếp cận đó sau trong bài viết này.

Một mẫu khác được sử dụng trong các ngôn ngữ khác như JavaScript là ngay lập tức thực hiện hàm Python Lambda. Điều này được gọi là một biểu thức chức năng được gọi ngay lập tức (iife, phát âm là iff iffy). Đây là một ví dụ:

Hàm Lambda ở trên được xác định và sau đó được gọi ngay lập tức với hai đối số (def add_one(x): return x + 1 6 và >>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}' >>> full_name('guido', 'van rossum') 'Full name: Guido Van Rossum' 6). Nó trả về giá trị >>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}' >>> full_name('guido', 'van rossum') 'Full name: Guido Van Rossum' 7, là tổng của các đối số.

Một số ví dụ trong hướng dẫn này sử dụng định dạng này để làm nổi bật khía cạnh ẩn danh của hàm Lambda và tránh tập trung vào

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7 trong Python như một cách ngắn hơn để xác định hàm.

Python không khuyến khích sử dụng các biểu thức Lambda được gọi ngay lập tức. Nó chỉ đơn giản là kết quả từ một biểu thức lambda có thể gọi được, không giống như cơ thể của một chức năng bình thường.

>>>

def f(x):
    return x * 2
2

Hàm ở trên xác định một biểu thức Lambda có hai đối số và trả về tổng của chúng.

Khác với việc cung cấp cho bạn phản hồi rằng Python hoàn toàn ổn với hình thức này, nó không dẫn đến bất kỳ việc sử dụng thực tế nào. Bạn có thể gọi chức năng trong trình thông dịch Python:

>>>

def f(x):
    return x * 2
3

Hàm ở trên xác định một biểu thức Lambda có hai đối số và trả về tổng của chúng.

Khác với việc cung cấp cho bạn phản hồi rằng Python hoàn toàn ổn với hình thức này, nó không dẫn đến bất kỳ việc sử dụng thực tế nào. Bạn có thể gọi chức năng trong trình thông dịch Python:

Ví dụ trên là tận dụng tính năng chỉ dành cho trình thông dịch tương tác được cung cấp thông qua dấu gạch dưới (

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3). Xem ghi chú dưới đây để biết thêm chi tiết.

>>>

def f(x):
    return x * 2
4

TraceBack của một ngoại lệ được nêu ra trong khi hàm Lambda được thực thi chỉ xác định hàm gây ra ngoại lệ là

def f(x):
    return x * 2
04.

Ở đây, một ngoại lệ tương tự được nâng lên bởi một chức năng bình thường:

>>>

def f(x):
    return x * 2
5

Hàm thông thường gây ra lỗi tương tự nhưng dẫn đến một dấu vết chính xác hơn vì nó cho tên hàm,

def f(x):
    return x * 2
07.

Cú pháp

Như bạn đã thấy trong các phần trước, một biểu mẫu lambda thể hiện sự phân biệt cú pháp từ một hàm bình thường. Cụ thể, một hàm Lambda có các đặc điểm sau:

  • Nó chỉ có thể chứa các biểu thức và có thể bao gồm các câu trong cơ thể của nó.
  • Nó được viết như một dòng thực thi duy nhất.
  • Nó không hỗ trợ chú thích loại.
  • Nó có thể được gọi ngay lập tức (iife).

Không có tuyên bố

Một hàm Lambda có thể chứa bất kỳ câu lệnh nào. Trong hàm Lambda, các câu như

>>> lambda x, y: x + y
7,
def f(x):
    return x * 2
09,
def f(x):
    return x * 2
10 hoặc
def f(x):
    return x * 2
11 sẽ tăng ngoại lệ
def f(x):
    return x * 2
12. Ở đây, một ví dụ về việc thêm
def f(x):
    return x * 2
10 vào thân máy Lambda:

>>>

def f(x):
    return x * 2
6

Ví dụ giả định này dự định

def f(x):
    return x * 2
10 rằng tham số
def add_one(x):
    return x + 1
0 có giá trị
def add_one(x):
    return x + 1
6. Nhưng, trình thông dịch xác định
def f(x):
    return x * 2
12 trong khi phân tích mã liên quan đến câu lệnh
def f(x):
    return x * 2
10 trong phần thân của
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7.

Biểu thức đơn

Trái ngược với một hàm bình thường, hàm Python lambda là một biểu thức duy nhất. Mặc dù, trong phần thân của

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7, bạn có thể truyền biểu thức trên một số dòng bằng dấu ngoặc đơn hoặc chuỗi đa dòng, nó vẫn là một biểu thức duy nhất:

>>>

def f(x):
    return x * 2
7

Ví dụ trên trả về chuỗi

def f(x):
    return x * 2
21 khi đối số lambda là lẻ và
def f(x):
    return x * 2
22 khi đối số là chẵn. Nó trải trên hai dòng vì nó được chứa trong một tập hợp các dấu ngoặc đơn, nhưng nó vẫn là một biểu thức duy nhất.

Loại chú thích

Nếu bạn đã bắt đầu áp dụng gợi ý loại, hiện có sẵn trong Python, thì bạn có một lý do chính đáng khác để thích các chức năng bình thường hơn các chức năng của Python Lambda. Kiểm tra kiểm tra loại Python (hướng dẫn) để tìm hiểu thêm về gợi ý loại Python và kiểm tra loại. Trong hàm Lambda, không có tương đương cho các điều sau:

def f(x):
    return x * 2
8

Bất kỳ loại lỗi nào với

def f(x):
    return x * 2
23 đều có thể bị bắt bởi các công cụ như
def f(x):
    return x * 2
24 hoặc
def f(x):
    return x * 2
25, trong khi đó
def f(x):
    return x * 2
12 với hàm Lambda tương đương được nâng lên khi chạy: thời gian chạy:

>>>

def f(x):
    return x * 2
9

Giống như cố gắng bao gồm một tuyên bố trong Lambda, việc thêm chú thích loại ngay lập tức dẫn đến

def f(x):
    return x * 2
12 khi chạy.

Iife

Bạn đã thấy một số ví dụ về thực thi chức năng được gọi ngay lập tức:

>>>

print iterator(lambda x: x / 4 + 12, 100, 5)
0

Bên ngoài phiên dịch viên Python, tính năng này có thể không được sử dụng trong thực tế. Nó có một hậu quả trực tiếp của một chức năng Lambda được gọi là được xác định. Ví dụ: điều này cho phép bạn chuyển định nghĩa của biểu thức Lambda Python cho hàm bậc cao hơn như

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4,
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5 hoặc
>>> lambda x, y: x + y
1 hoặc chức năng chính.

Tranh luận

Giống như một đối tượng hàm bình thường được xác định với

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
8, các biểu thức Python Lambda hỗ trợ tất cả các cách khác nhau để truyền đối số. Điêu nay bao gôm:

  • Lập luận vị trí
  • Đối số được đặt tên (đôi khi được gọi là đối số từ khóa)
  • Danh sách biến các đối số (thường được gọi là Varargs)varargs)
  • Danh sách biến các đối số từ khóa
  • Đối số chỉ từ khóa

Các ví dụ sau đây minh họa các tùy chọn mở cho bạn để chuyển các đối số cho các biểu thức Lambda:

>>>

print iterator(lambda x: x / 4 + 12, 100, 5)
1

Trang trí

Trong Python, một người trang trí là việc thực hiện một mẫu cho phép thêm một hành vi vào một chức năng hoặc một lớp. Nó thường được biểu thị bằng tiền tố cú pháp

def f(x):
    return x * 2
32 một hàm. Đây là một ví dụ giả định:

print iterator(lambda x: x / 4 + 12, 100, 5)
2

Trong ví dụ trên,

def f(x):
    return x * 2
33 là một hàm thêm hành vi vào
def f(x):
    return x * 2
34, do đó gọi
def f(x):
    return x * 2
35 dẫn đến kết quả đầu ra sau:

print iterator(lambda x: x / 4 + 12, 100, 5)
3

def f(x):
    return x * 2
34 chỉ in
def f(x):
    return x * 2
37, nhưng người trang trí thêm một hành vi bổ sung cũng in
def f(x):
    return x * 2
38.

Một người trang trí có thể được áp dụng cho một lambda. Mặc dù nó không thể trang trí Lambda với cú pháp

def f(x):
    return x * 2
32, một người trang trí chỉ là một chức năng, vì vậy nó có thể gọi hàm Lambda:

print iterator(lambda x: x / 4 + 12, 100, 5)
4

def f(x):
    return x * 2
40, được trang trí với
def f(x):
    return x * 2
41 trên dòng 11, được gọi bằng đối số
>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
6 trên dòng 15. Ngược lại, trên dòng 18, một hàm Lambda ngay lập tức được liên quan và nhúng vào một cuộc gọi đến
def f(x):
    return x * 2
43, người trang trí. Khi bạn thực thi mã ở trên, bạn có được những điều sau:

print iterator(lambda x: x / 4 + 12, 100, 5)
5

Xem cách, như bạn đã thấy, tên của hàm Lambda xuất hiện là

def f(x):
    return x * 2
04, trong khi
def f(x):
    return x * 2
45 được xác định rõ ràng cho hàm bình thường.

Trang trí chức năng Lambda theo cách này có thể hữu ích cho mục đích gỡ lỗi, có thể để gỡ lỗi hành vi của hàm Lambda được sử dụng trong bối cảnh chức năng bậc cao hoặc chức năng chính. Hãy cùng xem một ví dụ với

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4:

print iterator(lambda x: x / 4 + 12, 100, 5)
6

Đối số đầu tiên của

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 là một lambda nhân lên đối số của nó với
def add_one(x):
    return x + 1
6. Lambda này được trang trí với
def f(x):
    return x * 2
43. Khi được thực thi, ví dụ trên xuất ra sau:

print iterator(lambda x: x / 4 + 12, 100, 5)
7

Kết quả

def f(x):
    return x * 2
50 là một danh sách thu được từ việc nhân mỗi phần tử của
def f(x):
    return x * 2
51. Hiện tại, hãy xem xét
def f(x):
    return x * 2
51 tương đương với danh sách
def f(x):
    return x * 2
53.

Bạn sẽ được tiếp xúc với

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 chi tiết hơn trong bản đồ.

Một lambda cũng có thể là một người trang trí, nhưng nó không được khuyến khích. Nếu bạn thấy mình cần phải làm điều này, hãy tham khảo PEP 8, các khuyến nghị lập trình.

Để biết thêm về trang trí Python, hãy xem Primer trên trang trí Python.

Khép kín

Đóng là một hàm trong đó mọi biến miễn phí, mọi thứ ngoại trừ các tham số, được sử dụng trong hàm đó được liên kết với một giá trị cụ thể được xác định trong phạm vi kèm theo của hàm đó. Trong thực tế, đóng cửa xác định môi trường mà chúng chạy, và do đó có thể được gọi từ bất cứ đâu.

Các khái niệm về lambdas và đóng cửa không nhất thiết phải liên quan, mặc dù các hàm Lambda có thể đóng cửa giống như cách các chức năng bình thường cũng có thể là đóng cửa. Một số ngôn ngữ có các cấu trúc đặc biệt để đóng hoặc lambda (ví dụ, Groovy với một khối mã ẩn danh là đối tượng đóng) hoặc biểu thức lambda (ví dụ, biểu thức Java Lambda với tùy chọn giới hạn để đóng).

Ở đây, một đóng cửa được xây dựng với chức năng Python bình thường:

print iterator(lambda x: x / 4 + 12, 100, 5)
8

def f(x):
    return x * 2
55 Trả về
def f(x):
    return x * 2
56, một hàm lồng nhau tính tổng của ba đối số:

  • def add_one(x):
        return x + 1
    
    0 được thông qua như một đối số cho
    def f(x):
        return x * 2
    
    55.
    is passed as an argument to
    def f(x):
        return x * 2
    
    55.
  • def f(x):
        return x * 2
    
    59 là một biến cục bộ đến
    def f(x):
        return x * 2
    
    55.
    is a variable local to
    def f(x):
        return x * 2
    
    55.
  • def f(x):
        return x * 2
    
    61 là một đối số được chuyển đến
    def f(x):
        return x * 2
    
    56.
    is an argument passed to
    def f(x):
        return x * 2
    
    56.

Để kiểm tra hành vi của

def f(x):
    return x * 2
55 và
def f(x):
    return x * 2
56,
def f(x):
    return x * 2
55 được gọi ba lần trong vòng lặp
def f(x):
    return x * 2
66 in như sau:

print iterator(lambda x: x / 4 + 12, 100, 5)
9

Trên dòng 9 của mã,

def f(x):
    return x * 2
56 được trả về bởi lời mời của
def f(x):
    return x * 2
55 được liên kết với tên
def f(x):
    return x * 2
69. Trên dòng 5,
def f(x):
    return x * 2
56 nắm bắt
def add_one(x):
    return x + 1
0 và
def f(x):
    return x * 2
59 vì nó có quyền truy cập vào môi trường nhúng của nó, do đó khi gọi đóng cửa, nó có thể hoạt động trên hai biến miễn phí
def add_one(x):
    return x + 1
0 và
def f(x):
    return x * 2
59.

Tương tự, một

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7 cũng có thể là một đóng cửa. Ở đây, ví dụ tương tự với chức năng Python Lambda:

>>> def identity(x):
...     return x
0

Khi bạn thực thi mã ở trên, bạn có được đầu ra sau:

>>> def identity(x):
...     return x
1

Trên dòng 6,

def f(x):
    return x * 2
55 trả về một lambda và gán nó cho biến
def f(x):
    return x * 2
69. Trên dòng 3, phần thân của hàm Lambda tham chiếu
def add_one(x):
    return x + 1
0 và
def f(x):
    return x * 2
59. Biến
def f(x):
    return x * 2
59 có sẵn tại thời điểm định nghĩa, trong khi
def add_one(x):
    return x + 1
0 được xác định trong thời gian chạy khi
def f(x):
    return x * 2
55 được gọi.

Trong tình huống này, cả chức năng bình thường và Lambda đều hoạt động tương tự. Trong phần tiếp theo, bạn sẽ thấy một tình huống trong đó hành vi của Lambda có thể bị lừa dối do thời gian đánh giá của nó (thời gian định nghĩa so với thời gian chạy).

Thời gian đánh giá

Trong một số tình huống liên quan đến các vòng lặp, hành vi của một chức năng Python Lambda như một sự đóng cửa có thể phản trực giác. Nó đòi hỏi sự hiểu biết khi các biến miễn phí bị ràng buộc trong bối cảnh của một lambda. Các ví dụ sau đây cho thấy sự khác biệt khi sử dụng hàm thông thường so với sử dụng Lambda Python.

Kiểm tra kịch bản trước bằng cách sử dụng chức năng thông thường:

>>>

>>> def identity(x):
...     return x
2

Trong một hàm bình thường,

def f(x):
    return x * 2
83 được đánh giá tại thời điểm định nghĩa, trên dòng 9, khi hàm được thêm vào danh sách:
def f(x):
    return x * 2
84.

Bây giờ, với việc thực hiện cùng một logic với chức năng Lambda, hãy quan sát hành vi bất ngờ:

>>>

>>> def identity(x):
...     return x
3

Trong một hàm bình thường,

def f(x):
    return x * 2
83 được đánh giá tại thời điểm định nghĩa, trên dòng 9, khi hàm được thêm vào danh sách:
def f(x):
    return x * 2
84.

Bây giờ, với việc thực hiện cùng một logic với chức năng Lambda, hãy quan sát hành vi bất ngờ:

>>>

>>> def identity(x):
...     return x
4

Trong một hàm bình thường,

def f(x):
    return x * 2
83 được đánh giá tại thời điểm định nghĩa, trên dòng 9, khi hàm được thêm vào danh sách:
def f(x):
    return x * 2
84.

Bây giờ, với việc thực hiện cùng một logic với chức năng Lambda, hãy quan sát hành vi bất ngờ:

Kết quả bất ngờ xảy ra do biến miễn phí def f(x): return x * 2 83, như được triển khai, bị ràng buộc tại thời điểm thực hiện của biểu thức Lambda. Hàm Python Lambda trên dòng 4 là một đóng cửa thu thập def f(x): return x * 2 83, một biến miễn phí bị ràng buộc trong thời gian chạy. Khi chạy, trong khi gọi hàm def f(x): return x * 2 87 trên dòng 7, giá trị của def f(x): return x * 2 83 là def f(x): return x * 2 89.

Để khắc phục vấn đề này, bạn có thể chỉ định biến miễn phí theo thời gian định nghĩa như sau:

def f(x):
    return x * 2
94

Hàm Lambda Python hoạt động giống như một hàm bình thường liên quan đến các đối số. Do đó, tham số Lambda có thể được khởi tạo với giá trị mặc định: tham số

def f(x):
    return x * 2
83 lấy bên ngoài
def f(x):
    return x * 2
83 làm giá trị mặc định. Hàm Python Lambda có thể đã được viết là
def f(x):
    return x * 2
92 và có kết quả tương tự.

>>> def identity(x):
...     return x
5

def f(x):
    return x * 2
98 Xác định một trường hợp thử nghiệm với ba phương pháp thử nghiệm, mỗi phương pháp trong số họ thực hiện một kịch bản thử nghiệm cho
def f(x):
    return x * 2
99 được thực hiện như một hàm Lambda. Việc thực hiện tệp Python
print iterator(lambda x: x / 4 + 12, 100, 5)
00 có chứa
def f(x):
    return x * 2
98 tạo ra những điều sau:

>>> def identity(x):
...     return x
6

Đúng như dự đoán, chúng tôi có hai trường hợp thử nghiệm thành công và một lỗi cho

print iterator(lambda x: x / 4 + 12, 100, 5)
02: kết quả là
>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
7, nhưng kết quả dự kiến ​​là
print iterator(lambda x: x / 4 + 12, 100, 5)
04. Thất bại này là do một sai lầm có chủ ý trong trường hợp thử nghiệm. Thay đổi kết quả dự kiến ​​từ
print iterator(lambda x: x / 4 + 12, 100, 5)
04 thành
>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
7 sẽ đáp ứng tất cả các bài kiểm tra cho
def f(x):
    return x * 2
98.

def f(x):
    return x * 2
95

Mô -đun

def f(x):
    return x * 2
95 trích xuất mã Python tương tác từ
print iterator(lambda x: x / 4 + 12, 100, 5)
10 để thực hiện các thử nghiệm. Mặc dù cú pháp của các hàm Lambda Python không hỗ trợ một
print iterator(lambda x: x / 4 + 12, 100, 5)
10 điển hình, nhưng có thể gán một chuỗi cho phần tử
print iterator(lambda x: x / 4 + 12, 100, 5)
12 của Lambda có tên:

>>> def identity(x):
...     return x
7

def f(x):
    return x * 2
95 trong nhận xét DOC của Lambda
def f(x):
    return x * 2
99 mô tả các trường hợp thử nghiệm tương tự như trong phần trước.

Khi bạn thực hiện các bài kiểm tra qua

print iterator(lambda x: x / 4 + 12, 100, 5)
15, bạn sẽ nhận được như sau:

>>> def identity(x):
...     return x
8

Các kết quả kiểm tra thất bại từ cùng một lỗi được giải thích trong việc thực hiện các thử nghiệm đơn vị trong phần trước.

Bạn có thể thêm một

print iterator(lambda x: x / 4 + 12, 100, 5)
10 vào Python Lambda thông qua một bài tập cho
print iterator(lambda x: x / 4 + 12, 100, 5)
12 để ghi lại hàm Lambda. Mặc dù có thể, cú pháp Python tốt hơn
print iterator(lambda x: x / 4 + 12, 100, 5)
10 cho các chức năng bình thường so với các hàm Lambda.

Để biết tổng quan toàn diện về thử nghiệm đơn vị trong Python, bạn có thể muốn tham khảo bắt đầu với thử nghiệm trong Python.

Biểu hiện Lambda lạm dụng

Một số ví dụ trong bài viết này, nếu được viết trong bối cảnh mã Python chuyên nghiệp, sẽ đủ điều kiện là lạm dụng.

Nếu bạn thấy mình đang cố gắng vượt qua một cái gì đó mà một biểu thức Lambda không hỗ trợ, thì đây có lẽ là một dấu hiệu cho thấy một chức năng bình thường sẽ phù hợp hơn.

print iterator(lambda x: x / 4 + 12, 100, 5)
10 cho biểu thức lambda trong phần trước là một ví dụ tốt. Cố gắng khắc phục thực tế rằng chức năng Python Lambda không hỗ trợ các câu lệnh là một lá cờ đỏ khác.

Các phần tiếp theo minh họa một vài ví dụ về việc sử dụng Lambda nên tránh. Những ví dụ đó có thể là các tình huống, trong bối cảnh của Python Lambda, mã thể hiện mô hình sau:

  • Nó không theo hướng dẫn phong cách Python (PEP 8)
  • Nó cồng kềnh và khó đọc.
  • Nó thông minh không cần thiết với chi phí của khả năng đọc khó.

Nâng cao một ngoại lệ

Cố gắng nâng cao một ngoại lệ trong một chiếc Python Lambda sẽ khiến bạn suy nghĩ hai lần. Có một số cách thông minh để làm như vậy, nhưng ngay cả một cái gì đó như sau là tốt hơn để tránh:

>>>

>>> def identity(x):
...     return x
9

Bởi vì một tuyên bố không chính xác về mặt cú pháp trong cơ thể Python Lambda, cách giải quyết trong ví dụ trên bao gồm trừu tượng hóa cuộc gọi tuyên bố với chức năng chuyên dụng

print iterator(lambda x: x / 4 + 12, 100, 5)
20. Sử dụng loại cách giải quyết này nên tránh. Nếu bạn gặp loại mã này, bạn nên xem xét việc tái cấu trúc mã để sử dụng chức năng thông thường.

Phong cách mật mã

Như trong bất kỳ ngôn ngữ lập trình nào, bạn sẽ tìm thấy mã Python có thể khó đọc vì phong cách được sử dụng. Các chức năng của Lambda, do sự đồng nhất của chúng, có thể có lợi cho việc viết mã khó đọc.

Ví dụ Lambda sau đây chứa một số lựa chọn kiểu xấu:

>>>

>>> (lambda x: x + 1)(2)
3
0

Bởi vì một tuyên bố không chính xác về mặt cú pháp trong cơ thể Python Lambda, cách giải quyết trong ví dụ trên bao gồm trừu tượng hóa cuộc gọi tuyên bố với chức năng chuyên dụng

print iterator(lambda x: x / 4 + 12, 100, 5)
20. Sử dụng loại cách giải quyết này nên tránh. Nếu bạn gặp loại mã này, bạn nên xem xét việc tái cấu trúc mã để sử dụng chức năng thông thường.

>>>

>>> (lambda x: x + 1)(2)
3
1

Bởi vì một tuyên bố không chính xác về mặt cú pháp trong cơ thể Python Lambda, cách giải quyết trong ví dụ trên bao gồm trừu tượng hóa cuộc gọi tuyên bố với chức năng chuyên dụng

print iterator(lambda x: x / 4 + 12, 100, 5)
20. Sử dụng loại cách giải quyết này nên tránh. Nếu bạn gặp loại mã này, bạn nên xem xét việc tái cấu trúc mã để sử dụng chức năng thông thường.

>>>

>>> (lambda x: x + 1)(2)
3
2

Bởi vì một tuyên bố không chính xác về mặt cú pháp trong cơ thể Python Lambda, cách giải quyết trong ví dụ trên bao gồm trừu tượng hóa cuộc gọi tuyên bố với chức năng chuyên dụng

print iterator(lambda x: x / 4 + 12, 100, 5)
20. Sử dụng loại cách giải quyết này nên tránh. Nếu bạn gặp loại mã này, bạn nên xem xét việc tái cấu trúc mã để sử dụng chức năng thông thường.

Phong cách mật mã

Như trong bất kỳ ngôn ngữ lập trình nào, bạn sẽ tìm thấy mã Python có thể khó đọc vì phong cách được sử dụng. Các chức năng của Lambda, do sự đồng nhất của chúng, có thể có lợi cho việc viết mã khó đọc.

>>> (lambda x: x + 1)(2)
3
3

Ví dụ Lambda sau đây chứa một số lựa chọn kiểu xấu:

>>> (lambda x: x + 1)(2)
3
4

Mặc dù

print iterator(lambda x: x / 4 + 12, 100, 5)
31 không chỉ ra một vấn đề cho việc sử dụng các chức năng của Python Lambda trong các thuộc tính, nhưng chúng rất khó đọc và dễ bị lỗi vì sử dụng nhiều chuỗi như
print iterator(lambda x: x / 4 + 12, 100, 5)
35 và
print iterator(lambda x: x / 4 + 12, 100, 5)
36.

Việc thực hiện đúng

print iterator(lambda x: x / 4 + 12, 100, 5)
27 sẽ được dự kiến ​​như sau:

>>> (lambda x: x + 1)(2)
3
5

print iterator(lambda x: x / 4 + 12, 100, 5)
29 sẽ được viết như sau:

>>> (lambda x: x + 1)(2)
3
6

Theo nguyên tắc chung, trong bối cảnh mã được viết bằng Python, thích các chức năng thường xuyên hơn các biểu thức Lambda. Tuy nhiên, có những trường hợp được hưởng lợi từ cú pháp Lambda, như bạn sẽ thấy trong phần tiếp theo.

Việc sử dụng phù hợp của các biểu thức Lambda

Lambdas trong Python có xu hướng là chủ đề của những tranh cãi. Một số lập luận chống lại Lambdas trong Python là:

  • Các vấn đề với khả năng đọc
  • Việc áp đặt một cách suy nghĩ chức năng
  • Cú pháp nặng với từ khóa
    >>> add_one = lambda x: x + 1
    >>> add_one(2)
    3
    
    7

Mặc dù các cuộc tranh luận sôi nổi đặt câu hỏi về sự tồn tại đơn thuần của tính năng này trong Python, các chức năng của Lambda có các thuộc tính đôi khi cung cấp giá trị cho ngôn ngữ Python và cho các nhà phát triển.

Các ví dụ sau đây minh họa các kịch bản trong đó việc sử dụng các hàm Lambda không chỉ phù hợp mà còn được khuyến khích trong mã Python.

Cấu trúc chức năng cổ điển

Các hàm Lambda thường xuyên được sử dụng với các hàm tích hợp

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 và
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5, cũng như
>>> lambda x, y: x + y
1, được phơi bày trong mô-đun
print iterator(lambda x: x / 4 + 12, 100, 5)
43. Ba ví dụ sau đây là minh họa tương ứng về việc sử dụng các chức năng đó với các biểu thức Lambda làm bạn đồng hành:

>>>

>>> (lambda x: x + 1)(2)
3
7

Bạn có thể phải đọc mã giống với các ví dụ ở trên, mặc dù có dữ liệu có liên quan hơn. Vì lý do đó, điều quan trọng là phải nhận ra các cấu trúc đó. Tuy nhiên, những cấu trúc đó có các lựa chọn thay thế tương đương được coi là Pythonic hơn. Trong các lựa chọn thay thế cho Lambdas, bạn sẽ học cách chuyển đổi các chức năng bậc cao và Lambdas đi kèm của chúng thành các hình thức thành ngữ khác.

Chức năng chính

Các hàm chính trong Python là các hàm bậc cao hơn lấy tham số

print iterator(lambda x: x / 4 + 12, 100, 5)
44 làm đối số được đặt tên.
print iterator(lambda x: x / 4 + 12, 100, 5)
44 nhận được một hàm có thể là
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7. Hàm này ảnh hưởng trực tiếp đến thuật toán được điều khiển bởi chính hàm chính. Dưới đây là một số chức năng chính:

  • >>> lambda x, y: x + y
    
    2: Phương pháp liệt kê
    list method
  • >>> lambda x, y: x + y
    
    3,
    >>> lambda x, y: x + y
    
    4,
    >>> lambda x, y: x + y
    
    5: Các chức năng tích hợp
    built-in functions
  • print iterator(lambda x: x / 4 + 12, 100, 5)
    
    51 và
    print iterator(lambda x: x / 4 + 12, 100, 5)
    
    52: Trong mô -đun thuật toán hàng đợi heap
    print iterator(lambda x: x / 4 + 12, 100, 5)
    
    53
    in the Heap queue algorithm module
    print iterator(lambda x: x / 4 + 12, 100, 5)
    
    53

Hãy tưởng tượng rằng bạn muốn sắp xếp một danh sách các ID được thể hiện dưới dạng chuỗi. Mỗi ID là sự kết hợp của chuỗi

print iterator(lambda x: x / 4 + 12, 100, 5)
54 và một số. Theo mặc định, sắp xếp danh sách này với chức năng tích hợp
>>> lambda x, y: x + y
3, sử dụng thứ tự từ vựng vì các yếu tố trong danh sách là chuỗi.

Để ảnh hưởng đến việc thực thi sắp xếp, bạn có thể gán Lambda cho đối số được đặt tên

print iterator(lambda x: x / 4 + 12, 100, 5)
44, sao cho việc sắp xếp sẽ sử dụng số được liên kết với ID:

>>>

>>> (lambda x: x + 1)(2)
3
8

Bạn có thể phải đọc mã giống với các ví dụ ở trên, mặc dù có dữ liệu có liên quan hơn. Vì lý do đó, điều quan trọng là phải nhận ra các cấu trúc đó. Tuy nhiên, những cấu trúc đó có các lựa chọn thay thế tương đương được coi là Pythonic hơn. Trong các lựa chọn thay thế cho Lambdas, bạn sẽ học cách chuyển đổi các chức năng bậc cao và Lambdas đi kèm của chúng thành các hình thức thành ngữ khác.

Chức năng chính

Các hàm chính trong Python là các hàm bậc cao hơn lấy tham số

print iterator(lambda x: x / 4 + 12, 100, 5)
44 làm đối số được đặt tên.
print iterator(lambda x: x / 4 + 12, 100, 5)
44 nhận được một hàm có thể là
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7. Hàm này ảnh hưởng trực tiếp đến thuật toán được điều khiển bởi chính hàm chính. Dưới đây là một số chức năng chính:

>>> (lambda x: x + 1)(2)
3
9

>>> lambda x, y: x + y
2: Phương pháp liệt kê

Hướng dẫn python pass lambda as argument - python vượt qua lambda làm đối số

>>> lambda x, y: x + y
3,
>>> lambda x, y: x + y
4,
>>> lambda x, y: x + y
5: Các chức năng tích hợp

print iterator(lambda x: x / 4 + 12, 100, 5)
51 và
print iterator(lambda x: x / 4 + 12, 100, 5)
52: Trong mô -đun thuật toán hàng đợi heap
print iterator(lambda x: x / 4 + 12, 100, 5)
53

Hãy tưởng tượng rằng bạn muốn sắp xếp một danh sách các ID được thể hiện dưới dạng chuỗi. Mỗi ID là sự kết hợp của chuỗi print iterator(lambda x: x / 4 + 12, 100, 5) 54 và một số. Theo mặc định, sắp xếp danh sách này với chức năng tích hợp >>> lambda x, y: x + y 3, sử dụng thứ tự từ vựng vì các yếu tố trong danh sách là chuỗi.

Để ảnh hưởng đến việc thực thi sắp xếp, bạn có thể gán Lambda cho đối số được đặt tên

print iterator(lambda x: x / 4 + 12, 100, 5)
44, sao cho việc sắp xếp sẽ sử dụng số được liên kết với ID:

Khung UI

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
0

Bạn có thể phải đọc mã giống với các ví dụ ở trên, mặc dù có dữ liệu có liên quan hơn. Vì lý do đó, điều quan trọng là phải nhận ra các cấu trúc đó. Tuy nhiên, những cấu trúc đó có các lựa chọn thay thế tương đương được coi là Pythonic hơn. Trong các lựa chọn thay thế cho Lambdas, bạn sẽ học cách chuyển đổi các chức năng bậc cao và Lambdas đi kèm của chúng thành các hình thức thành ngữ khác.

Một cách tiếp cận khác là sử dụng

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7:

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
1

Giải pháp này là sạch hơn, dễ đọc hơn và nhanh hơn để gõ trình thông dịch. Mặc dù thời gian thực hiện ít hơn một chút đối với phiên bản

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7, việc thực thi các chức năng một lần nữa có thể hiển thị một lợi thế nhỏ cho phiên bản
print iterator(lambda x: x / 4 + 12, 100, 5)
67. Thời gian thực hiện của
print iterator(lambda x: x / 4 + 12, 100, 5)
68 được loại trừ khỏi thời gian thực hiện chung và không nên có bất kỳ tác động nào đến kết quả.

Khỉ vá

Để thử nghiệm, đôi khi nó cần phải dựa vào kết quả lặp lại, ngay cả khi trong quá trình thực hiện bình thường một phần mềm nhất định, các kết quả tương ứng dự kiến ​​sẽ khác nhau hoặc thậm chí là hoàn toàn ngẫu nhiên.

Hãy nói rằng bạn muốn kiểm tra một chức năng, trong thời gian chạy, xử lý các giá trị ngẫu nhiên. Nhưng, trong quá trình thực hiện thử nghiệm, bạn cần khẳng định chống lại các giá trị có thể dự đoán theo cách lặp lại. Ví dụ sau đây cho thấy làm thế nào, với chức năng

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7, việc vá khỉ có thể giúp bạn:

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
2

Trình quản lý bối cảnh giúp cách điện cách cách điện cho khỉ vá một hàm từ thư viện tiêu chuẩn (

print iterator(lambda x: x / 4 + 12, 100, 5)
70, trong ví dụ này). Hàm Lambda được gán cho
print iterator(lambda x: x / 4 + 12, 100, 5)
71 thay thế hành vi mặc định bằng cách trả về giá trị tĩnh.

Điều này cho phép kiểm tra bất kỳ chức năng nào tùy thuộc vào

print iterator(lambda x: x / 4 + 12, 100, 5)
72 theo kiểu có thể dự đoán được. Trước khi thoát khỏi Trình quản lý bối cảnh, hành vi mặc định của
print iterator(lambda x: x / 4 + 12, 100, 5)
72 được thiết lập lại để loại bỏ bất kỳ tác dụng phụ bất ngờ nào có ảnh hưởng đến các lĩnh vực khác của thử nghiệm có thể phụ thuộc vào hành vi mặc định của
print iterator(lambda x: x / 4 + 12, 100, 5)
72.

Các khung kiểm tra đơn vị như

def f(x):
    return x * 2
94 và
print iterator(lambda x: x / 4 + 12, 100, 5)
76 Đưa khái niệm này đến mức độ tinh vi cao hơn.

Với

print iterator(lambda x: x / 4 + 12, 100, 5)
76, vẫn sử dụng hàm
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7, cùng một ví dụ trở nên thanh lịch và súc tích hơn:

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
3

Với trận đấu pytest

print iterator(lambda x: x / 4 + 12, 100, 5)
79,
print iterator(lambda x: x / 4 + 12, 100, 5)
71 được ghi đè bằng một lambda sẽ trả về giá trị xác định,
print iterator(lambda x: x / 4 + 12, 100, 5)
81, cho phép xác thực thử nghiệm. Trật trình pytest
print iterator(lambda x: x / 4 + 12, 100, 5)
79 cho phép bạn kiểm soát phạm vi ghi đè. Trong ví dụ trên, việc gọi
print iterator(lambda x: x / 4 + 12, 100, 5)
71 trong các thử nghiệm tiếp theo, mà không sử dụng việc vá khỉ, sẽ thực hiện việc thực hiện bình thường của chức năng này.

Thực hiện kiểm tra

print iterator(lambda x: x / 4 + 12, 100, 5)
76 cho kết quả sau:

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
4

Bài kiểm tra vượt qua khi chúng tôi xác nhận rằng

print iterator(lambda x: x / 4 + 12, 100, 5)
85 đã được thực hiện và kết quả là những kết quả được mong đợi trong bối cảnh thử nghiệm.

Giải pháp thay thế cho Lambdas

Mặc dù có những lý do tuyệt vời để sử dụng

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
7, nhưng có những trường hợp sử dụng nó được cau mày. Vậy các lựa chọn thay thế là gì?

Các chức năng bậc cao như

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4,
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5 và
>>> lambda x, y: x + y
1 có thể được chuyển đổi thành các hình thức thanh lịch hơn với sự xoắn ốc nhẹ của sự sáng tạo, đặc biệt là với sự hiểu biết danh sách hoặc biểu thức máy phát.

Để tìm hiểu thêm về toàn bộ danh sách, hãy xem khi nào nên sử dụng danh sách hiểu trong Python. Để tìm hiểu thêm về các biểu thức máy phát, hãy xem cách sử dụng trình tạo và mang lại trong Python.

Bản đồ

Hàm tích hợp

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 có chức năng như một đối số đầu tiên và áp dụng nó cho từng phần tử của đối số thứ hai, một điều không thể sử dụng được. Ví dụ về các vòng lặp là chuỗi, danh sách và bộ dữ liệu. Để biết thêm thông tin về Iterables và iterators, hãy xem Iterables và Iterators.iterable. Examples of iterables are strings, lists, and tuples. For more information on iterables and iterators, check out Iterables and Iterators.

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 trả về một trình lặp tương ứng với bộ sưu tập được chuyển đổi. Ví dụ, nếu bạn muốn chuyển đổi danh sách các chuỗi thành một danh sách mới với mỗi chuỗi được viết hoa, bạn có thể sử dụng
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4, như sau:

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
5

Bạn cần gọi

print iterator(lambda x: x / 4 + 12, 100, 5)
93 để chuyển đổi trình lặp được trả về bởi
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 thành một danh sách mở rộng có thể được hiển thị trong trình thông dịch shell python.

Sử dụng danh sách hiểu loại bỏ sự cần thiết phải xác định và gọi chức năng Lambda:

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
6

Lọc

Hàm tích hợp

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5, một cấu trúc chức năng cổ điển khác, có thể được chuyển đổi thành một sự hiểu biết danh sách. Nó lấy một vị từ như một đối số đầu tiên và một điều có thể là một đối số thứ hai. Nó xây dựng một trình lặp có chứa tất cả các yếu tố của bộ sưu tập ban đầu thỏa mãn chức năng vị ngữ. Dưới đây, một ví dụ đã lọc tất cả các số chẵn trong một danh sách các số nguyên nhất định:

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
7

Lưu ý rằng

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5 trả về một trình lặp, do đó cần phải gọi loại
print iterator(lambda x: x / 4 + 12, 100, 5)
97 tích hợp xây dựng một danh sách được đưa ra một trình lặp.

Việc thực hiện thúc đẩy cấu trúc hiểu danh sách đưa ra như sau:

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
8

Giảm

Kể từ Python 3,

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
6 đã chuyển từ hàm tích hợp sang hàm mô-đun
print iterator(lambda x: x / 4 + 12, 100, 5)
43. Như
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
4 và
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
5, hai đối số đầu tiên của nó lần lượt là một hàm và có thể lặp lại. Nó cũng có thể lấy một bộ khởi tạo làm đối số thứ ba được sử dụng làm giá trị ban đầu của bộ tích lũy kết quả. Đối với mỗi yếu tố của ITEBLE,
>>> add_one = lambda x: x + 1
>>> add_one(2)
3
6 áp dụng chức năng và tích lũy kết quả được trả về khi có thể cạn kiệt.

Để áp dụng

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
6 vào danh sách các cặp và tính tổng của mục đầu tiên của mỗi cặp, bạn có thể viết bài này:

>>>

(lambda x: x + 1)(2) = lambda 2: 2 + 1
                     = 2 + 1
                     = 3
9

Một cách tiếp cận thành ngữ hơn khi sử dụng biểu thức trình tạo, làm đối số với

>>> def identity(x):
...     return x
04 trong ví dụ, là như sau:

>>>

Một cách tiếp cận thành ngữ hơn khi sử dụng biểu thức trình tạo, làm đối số với
>>> def identity(x):
...     return x
04 trong ví dụ, là như sau:

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
0

>>>

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
1

Một cách tiếp cận thành ngữ hơn khi sử dụng biểu thức trình tạo, làm đối số với

>>> def identity(x):
...     return x
04 trong ví dụ, là như sau:

>>> add_one = lambda x: x + 1
>>> add_one(2)
3
0

Một giải pháp hơi khác và có thể làm sạch sẽ loại bỏ sự cần thiết phải truy cập rõ ràng phần tử đầu tiên của cặp và thay vào đó sử dụng giải nén:

Việc sử dụng dấu gạch dưới (

>>> full_name = lambda first, last: f'Full name: {first.title()} {last.title()}'
>>> full_name('guido', 'van rossum')
'Full name: Guido Van Rossum'
3) là một quy ước Python chỉ ra rằng bạn có thể bỏ qua giá trị thứ hai của cặp.

>>> def identity(x):
...     return x
04 có một đối số duy nhất, vì vậy biểu thức máy phát không cần phải ở trong ngoặc đơn.

Lambdas Pythonic có hay không?

PEP 8, là hướng dẫn kiểu cho mã Python, có nội dung:

Luôn sử dụng câu lệnh DEF thay vì câu lệnh gán liên kết biểu thức Lambda trực tiếp với định danh. (Nguồn)

Điều này không khuyến khích sử dụng Lambda ràng buộc với một định danh, chủ yếu là nơi các chức năng nên được sử dụng và có nhiều lợi ích hơn. PEP 8 không đề cập đến các cách sử dụng khác của >>> add_one = lambda x: x + 1 >>> add_one(2) 3 7. Như bạn đã thấy trong các phần trước, các chức năng của Lambda chắc chắn có thể sử dụng tốt, mặc dù chúng bị hạn chế.

Một cách có thể để trả lời câu hỏi là các chức năng Lambda là hoàn toàn pythonic nếu không có gì có sẵn thêm pythonic. Tôi tránh xa việc xác định ý nghĩa của Pythonic, để lại cho bạn định nghĩa phù hợp nhất với suy nghĩ của bạn, cũng như phong cách mã hóa cá nhân hoặc nhóm của bạn.

  • Ngoài phạm vi hẹp của Python
    >>> add_one = lambda x: x + 1
    >>> add_one(2)
    3
    
    7, cách viết mã Python tuyệt đẹp với PEP 8 là một tài nguyên tuyệt vời mà bạn có thể muốn kiểm tra về phong cách mã trong Python.
  • Sự kết luận
  • Bây giờ bạn biết cách sử dụng các chức năng Python
    >>> add_one = lambda x: x + 1
    >>> add_one(2)
    3
    
    7 và có thể:
  • Viết Python Lambdas và sử dụng các chức năng ẩn danh

Chọn một cách khôn ngoan giữa các chức năng của Lambdas hoặc Python bình thường

Tránh sử dụng quá nhiều lambdas

Sử dụng lambdas với các chức năng bậc cao hoặc các chức năng khóa Python This tutorial has a related video course created by the Real Python team. Watch it together with the written tutorial to deepen your understanding: How to Use Python Lambda Functions

Tôi có thể chuyển một chức năng như một đối số trong Python không?

Bởi vì các chức năng là các đối tượng, chúng ta có thể truyền chúng như các đối số cho các chức năng khác. Các chức năng có thể chấp nhận các chức năng khác vì các đối số còn được gọi là các hàm bậc cao hơn. Trong ví dụ dưới đây, một lời chào hàm được tạo có chức năng như một đối số.. Functions that can accept other functions as arguments are also called higher-order functions. In the example below, a function greet is created which takes a function as an argument.

Có bao nhiêu đối số có thể được truyền cho chức năng Lambda trong Python?

Bạn có thể sử dụng nhiều đối số như bạn muốn trong hàm Lambda, nhưng nó chỉ có thể có một biểu thức. Biểu thức này được đánh giá và trả về kết quả.as many arguments as you want in a lambda function, but it can have only one expression. This expression is evaluated and returned as a result.

Hàm Lambda () được gọi như thế nào?

Hàm Lambda, còn được gọi là 'Hàm ẩn danh' giống như hàm Python thông thường nhưng có thể được xác định mà không có tên.Mặc dù các hàm bình thường được xác định bằng cách sử dụng từ khóa DEF, các hàm ẩn danh được xác định bằng cách sử dụng từ khóa Lambda.Tuy nhiên, chúng bị giới hạn trong một dòng biểu thức duy nhất.Anonymous function' is same as a regular python function but can be defined without a name. While normal functions are defined using the def keyword, anonymous functions are defined using the lambda keyword. However,they are restricted to single line of expression.

Bạn có thể chuyển một chức năng ẩn danh như một đối số cho một hàm khác không?

Các hàm ẩn danh là các chức năng không có tên.Các hàm ẩn danh có thể được sử dụng như một đối số cho các chức năng khác hoặc như một thực thi chức năng được gọi ngay lập tức.Anonymous functions can be used as an argument to other functions or as an immediately invoked function execution.