Giới thiệu Python Descriptor

Các Python descriptors đã được giới thiệu trong Python 2.2, cùng với các lớp kiểu dáng mới, nhưng chúng vẫn chưa được sử dụng rộng rãi. Các Python descriptors là cách để tạo ra các thuộc tính được quản lý. Trong số rất nhiều lợi thế của chúng, các thuộc tính được quản lý đó được sử dụng để bảo vệ một thuộc tính khỏi những thay đổi hoặc để tự động cập nhật các giá trị của một thuộc tính phụ thuộc. Các descriptors làm tăng sự hiểu biết về Python và cải thiện các kỹ năng mã hóa. Bài này giới thi ệu giao thức descriptors và trình bày cách tạo và sử dụng các descriptors.

pdf7 trang | Chia sẻ: lylyngoc | Lượt xem: 1745 | Lượt tải: 2download
Bạn đang xem nội dung tài liệu Giới thiệu Python Descriptor, để tải tài liệu về máy bạn click vào nút DOWNLOAD ở trên
Giới thiệu Python Descriptor Giới thiệu Các Python descriptors đã được giới thiệu trong Python 2.2, cùng với các lớp kiểu dáng mới, nhưng chúng vẫn chưa được sử dụng rộng rãi. Các Python descriptors là cách để tạo ra các thuộc tính được quản lý. Trong số rất nhiều lợi thế của chúng, các thuộc tính được quản lý đó được sử dụng để bảo vệ một thuộc tính khỏi những thay đổi hoặc để tự động cập nhật các giá trị của một thuộc tính phụ thuộc. Các descriptors làm tăng sự hiểu biết về Python và cải thiện các kỹ năng mã hóa. Bài này giới thiệu giao thức descriptors và trình bày cách tạo và sử dụng các descriptors. Giao thức descriptors Giao thức descriptors trong Python chỉ đơn giản là một cách để xác định những gì sẽ xảy ra khi tham chiếu một thuộc tính trong một mô hình. Nó cho phép một lập trình viên quản lý truy cập thuộc tính dễ dàng và hiệu quả:  set  get  delete Trong các ngôn ngữ lập trình khác, các descriptors được gọi là setter và getter, ở đây các hàm public được sử dụng để Get (nhận giá trị) và Set (đặt giá trị ) cho một biến private. Python không có một khái niệm về các biến private và có thể coi giao thức descriptors như là một cách của Python để đạt được điều tương tự. Nhìn chung, một descriptor là một thuộc tính của đối tượng với một hành vi kết buộc, một hành vi mà việc truy cập thuộc tính của nó bị ghi đè bằng các phương thức trong giao thức descriptors. Các phương thức đó là __get__, __set__ và __delete__. Nếu định nghĩa một phương thức bất kỳ trong số này cho một đối tượng, người ta nói rằng đó là một descriptor. Hãy xem xét kỹ hơn các phương thức này trong Liệt kê 1. Liệt kê 1. Các phương thức descriptors __get__(self, instance, owner) __set__(self, instance, value) __delete__(self, instance) Ở đây: __get__ truy cập thuộc tính. Nó trả về giá trị của thuộc tính hoặc đưa ra ngoại lệ AttributeError nếu không tồn tại thuộc tính được yêu cầu. __set__ được gọi là một phép gán thuộc tính. Không trả về cái gì cả. __delete__ kiểm soát hoạt động xóa. Không trả về cái gì cả. Điều quan trọng cần lưu ý là các descriptors được gán cho một lớp, chứ không gán cho một cá thể của lớp. Việc sửa đổi lớp này sẽ ghi đè lên hoặc xóa chính descriptor đó, chứ không phải kích hoạt mã của nó. Về đầu trang Khi nào cần các descriptors Hãy xem xét một thuộc tính email. Việc xác minh định dạng email đúng là cần thiết trước khi gán một giá trị cho thuộc tính đó. Descriptor này cho phép xử lý địa chỉ email thông qua một biểu thức chính quy và định dạng của nó được xác nhận hợp lệ trước khi gán nó cho một thuộc tính. Trong nhiều trường hợp khác, các giao thức descriptors trong Python kiểm soát truy cập đến các thuộc tính, chẳng hạn như bảo vệ thuộc tính name. Về đầu trang Tạo các descriptors Bạn có thể tạo ra một descriptor theo một số cách sau:  Tạo một lớp và ghi đè lên bất kỳ các phương thức nào của descriptor: __set__, __ get__ và __delete__. Phương thức này được sử dụng khi cần descriptor giống nhau qua nhiều lớp và các thuộc tính khác nhau, ví dụ, để xác nhận hợp lệ cho kiểu.  Sử dụng một kiểu property (đặc tính) là một cách đơn giản hơn và linh hoạt hơn để tạo ra một descriptor.  Sử dụng sức mạnh của các bộ trang trí đặc tính, là một tổ hợp của phương thức về kiểu đặc tính và các bộ trang trí của Python. Tất cả các ví dụ dưới đây là giống nhau theo quan điểm hoạt động. Sự khác biệt nằm ở việc thực hiện. Về đầu trang Tạo các descriptors bằng cách sử dụng các phương thức lớp Liệt kê 2 cho thấy tính đơn giản của việc kiểm soát gán thuộc tính trong Python. Liệt kê 2. Tạo các descriptors bằng cách sử dụng các phương thức lớp class Descriptor(object): def __init__(self): self._name = '' def __get__(self, instance, owner): print "Getting: %s" % self._name return self._name def __set__(self, instance, name): print "Setting: %s" % name self._name = name.title() def __delete__(self, instance): print "Deleting: %s" %self._name del self._name class Person(object): name = Descriptor() Hãy sử dụng mã này và xem kết quả ban đầu: >>> user = Person() >>> user.name = 'john smith' Setting: john smith >>> user.name Getting: John Smith 'John Smith' >>> del user.name Deleting: John Smith Một lớp descriptor được tạo ra bằng cách ghi đè các phương thức __set__(), __get__() và __delete__() của lớp cha mẹ sao cho  get sẽ in ra Getting  delete sẽ in ra Deleting  set sẽ in ra Setting và thay đổi giá trị thuộc tính thành tiêu đề (viết hoa chữ cái đầu tiên, viết thường các chữ khác) trước khi gán. Việc này rất tiện dụng, chẳng hạn khi lưu và in các tên. Việc chuyển đổi thành chữ hoa cũng có thể được chuyển sang cho phương thức __get__(). _value sẽ có giá trị ban đầu và sẽ được chuyển đổi thành tiêu đề khi có yêu cầu get. Về đầu trang Tạo các descriptors bằng cách sử dụng kiểu property Trong khi bộ mô tả được xác định trong Liệt kê 2 là hợp lệ và hoạt động được, một phương thức khác là thông qua kiểu đặc tính. Với hàm property(), dễ dàng tạo ra một bộ mô tả có thể sử dụng cho bất kỳ thuộc tính nào. Cú pháp để tạo property() là property(fget=None, fset=None, fdel=None, doc=None) ở đây:  fget – phương thức get (nhận) thuộc tính  fset – phương thức set (thiết lập) thuộc tính  fdel – phương thức delete (xóa) thuộc tính  doc – docstring (chuỗi ký tự tài liệu) Viết lại ví dụ bằng cách sử dụng property, như trongLiệt kê 3. Liệt kê 3. Tạo descriptor bằng kiểu property class Person(object): def __init__(self): self._name = '' def fget(self): print "Getting: %s" % self._name return self._name def fset(self, value): print "Setting: %s" % value self._name = value.title() def fdel(self): print "Deleting: %s" %self._name del self._name name = property(fget, fset, fdel, "I'm the property.") Hãy sử dụng mã này và xem kết quả: >>> user = Person() >>> user.name = 'john smith' Setting: john smith >>> user.name Getting: John Smith 'John Smith' >>> del user.name Deleting: John Smith Rõ ràng, kết quả như nhau. Lưu ý ở đây là các phương thức fget, fset và fdel đều là tùy chọn, nhưng nếu một phương thức không được xác định rõ, sẽ xảy ra một ngoại lệ AttributeError khi cố thực hiện hoạt động tương ứng. Ví dụ, đặc tính name được khai báo là None như với fset và sau đó nhà phát triển cố gắng gán giá trị cho thuộc tính name thì một ngoại lệ AttributeError được đưa ra. Có thể sử dụng điều này để định nghĩa thuộc tính chỉ đọc trong hệ thống. name = property(fget, None, fdel, "I'm the property") user.name = 'john smith' Kết quả: Traceback (most recent call last): File stdin, line 21, in mоdule user.name = 'john smith' AttributeError: can't set attribute Về đầu trang Tạo các descriptors bằng cách sử dụng các bộ trang trí đặc tính Các bộ mô tả có thể được tạo ra bằng các bộ trang trí của Python, như trong Liệt kê 4. Bộ trang trí Python là một sự thay đổi đặc trưng cho cú pháp Python, cho phép sửa lại thuận tiện hơn các hàm và các phương thức. Trong trường hợp này, các phương thức quản lý thuộc tính bị sửa đổi. Hãy tìm thêm thông tin về việc áp dụng các bộ trang trí Python trong bài viết, Các bộ trang trí làm mọi thứ trở nên dễ dàng hơn trên developerWorks. Liệt kê 4. Tạo các descriptors với các bộ trang trí đặc tính class Person(object): def __init__(self): self._name = '' @property def name(self): print "Getting: %s" % self._name return self._name @name.setter def name(self, value): print "Setting: %s" % value self._name = value.title() @name.deleter def name(self): print ">Deleting: %s" % self._name del self._name Về đầu trang Tạo các descriptors tại thời điểm chạy Tất cả các ví dụ trước hoạt động với thuộc tính name. Hạn chế của cách tiếp cận này là sự cần thiết phải ghi đè __set__(), __get__() và __delete__() riêng biệt cho từng thuộc tính. Liệt kê 5 cung cấp một giải pháp khả thi khi một nhà phát triển muốn thêm các thuộc tính đặc tính tại thời gian chạy. Liệt kê này sử dụng kiểu đặc tính để xây dựng một bộ mô tả dữ liệu (data descriptor). Liệt kê 5. Tạo các descriptors tại thời gian chạy class Person(object): def addProperty(self, attribute): # create local setter and getter with a particular attribute name getter = lambda self: self._getProperty(attribute) setter = lambda self, value: self._setProperty(attribute, value) # construct property attribute and add it to the class setattr(self.__class__, attribute, property(fget=getter, \ fset=setter, \ doc="Auto-generated method")) def _setProperty(self, attribute, value): print "Setting: %s = %s" %(attribute, value) setattr(self, '_' + attribute, value.title()) def _getProperty(self, attribute): print "Getting: %s" %attribute return getattr(self, '_' + attribute) Hãy chạy thử mã này: >>> user = Person() >>> user.addProperty('name') >>> user.addProperty('phone') >>> user.name = 'john smith' Setting: name = john smith >>> user.phone = '12345' Setting: phone = 12345 >>> user.name Getting: name 'John Smith' >>> user.__dict__ {'_phone': '12345', '_name': 'John Smith'} Mã này tạo ra các thuộc tính name và phone tại thời gian chạy. Có thể truy cập tới chúng bằng tên tương ứng, nhưng chúng được lưu trữ trong từ điển namespace đối tượng là _name và _phone, như được quy định trong phương thức _setProperty. Về cơ bản, name và phone là các hàm truy cập vào các thuộc tính nội bộ _name và _phone. Bạn có thể có một câu hỏi liên quan đến một thuộc tính _name trong hệ thống khi nhà phát triển cố gắng thêm thuộc tính đặc tính name. Câu trả lời là nó sẽ ghi đè lên thuộc tính _name hiện có bằng thuộc tính đặc tính mới. Mã này cho phép kiểm soát các thuộc tính được xử lý bên trong một lớp ra sao. Về đầu trang Kết luận Các descriptors trong Python cho phép quản lý thuộc tính mạnh mẽ và linh hoạt với các lớp kiểu dáng mới. Được kết hợp với các bộ trang trí, chúng làm cho việc lập trình dễ dàng, cho phép tạo ra các Setter và các Getter, cũng như các thuộc tính chỉ đọc (read-only). Nó cũng cho phép bạn chạy xác nhận hợp lệ thuộc tính theo yêu cầu theo giá trị hoặc kiểu. Bạn có thể áp dụng các descriptors trong nhiều lĩnh vực, nhưng hãy sử dụng chúng hết sức thận trọng để tránh làm phức tạp mã không cần thiết xuất phát từ việc ghi đè lên các hành vi bình thường của một đối tượng.