Hướng dẫn python-import all from file - python-nhập tất cả từ tệp
Mục lục bài viết:
Nội dung chính Show
Nội dung chính Python cơ bản import Giới thiệu importlib.resources Sử dụng importlib
Python cơ bản importGiới thiệu importlib.resources Sử dụng importlib Mô-đunCác bạn có thể download code ví dụ tại ĐÂY để tiện trong quá trình tìm hiểu bài viết này.
Hướng dẫn này sẽ cung cấp một cái nhìn tổng quan toàn diện về 35câu lệnh của Python và cách nó hoạt động. Hệ thống nhập rất mạnh và bạn sẽ học cách khai thác sức mạnh này. Trong khi bạn sẽ bao gồm nhiều khái niệm đằng sau hệ thống nhập của Python, hướng dẫn này chủ yếu là hướng dẫn ví dụ. Bạn sẽ học từ một số ví dụ mã trong suốt.Trong hướng dẫn này, bạn sẽ học cách: Sử dụng mô-đun , gói và gói không gian tên
Xử lý tài nguyên và tệp dữ liệu bên trong gói của bạn Nhập mô-đun động trong thời gian chạy
Mã Python được tổ chức thành cả mô-đun và gói . Phần này sẽ giải thích chúng khác nhau như thế nào và bạn có thể làm việc với chúng như thế nào. >>>
Sử dụng 46mà không có bất kỳ đối số nào cho thấy những gì trong không gian tên chung. Để xem nội dung của 39không gian tên, bạn sử dụng 49.Bạn đã thấy cách sử dụng đơn giản nhất của 35. Tuy nhiên, có những cách khác để sử dụng nó cho phép bạn nhập các phần cụ thể của mô-đun và đổi tên mô-đun khi bạn nhập.Đoạn mã sau chỉ nhập 40biến từ 39mô-đun:>>>
Lưu ý rằng điều này 40nằm trong không gian tên chung chứ không phải trong 39không gian tên.Bạn cũng có thể đổi tên các mô-đun và thuộc tính khi chúng được nhập: >>>
Để biết thêm chi tiết về cú pháp nhập mô-đun, hãy xem Mô-đun và gói Python - Giới thiệu . Các góiBạn có thể sử dụng một gói để tổ chức thêm các mô-đun của mình. Bảng chú giải thuật ngữ Python.org định nghĩa gói như sau:
Lưu ý rằng một gói vẫn là một mô-đun. Là một người dùng, bạn thường không cần phải lo lắng về việc bạn đang nhập một mô-đun hay một gói. Trong thực tế, một gói thường tương ứng với một thư mục tệp chứa tệp Python và các thư mục khác. Để tự tạo một gói Python, bạn tạo một thư mục và một tệp có tên 56 bên trong nó. Các 56tập tin có chứa các nội dung của gói khi nó được coi là một mô-đun. Nó có thể được để trống.Lưu ý: Các thư mục không có 56tệp vẫn được Python coi là gói. Tuy nhiên, đây không phải là các gói thông thường, mà là một thứ được gọi là các gói không gian tên . Bạn sẽ tìm hiểu thêm về chúng sau này .Nói chung, các mô-đun con và gói con không được nhập khi bạn nhập một gói. Tuy nhiên, bạn có thể sử dụng 56để bao gồm bất kỳ hoặc tất cả các mô-đun con và gói con nếu bạn muốn. Để hiển thị một vài ví dụ về hành vi này, bạn sẽ tạo một gói để nói 60bằng một vài ngôn ngữ khác nhau. Gói này sẽ bao gồm các thư mục và tệp sau:
Mỗi tệp quốc gia sẽ in ra một lời chào, trong khi các 56tệp nhập một số gói con và mô-đun con một cách chọn lọc. Nội dung chính xác của các tệp như sau:
Lưu ý rằng 62chỉ nhập khẩu 63và không nhập khẩu 64. Tương tự, 65không nhập bất cứ thứ gì, trong khi 66nhập 67và 68nhưng không 69. Mỗi mô-đun quốc gia sẽ in lời chào khi được nhập.Hãy chơi với 70gói tại dấu nhắc tương tác để hiểu rõ hơn về cách các gói con và mô-đun con hoạt động:>>>
Khi 64được nhập, các mô-đun 72và 73mô-đun cũng được nhập. Bạn có thể thấy điều này vì các mô-đun quốc gia in lời chào khi chúng được nhập:>>>
Các 65tập tin trống. Điều này có nghĩa là việc nhập 75gói sẽ tạo ra không gian tên nhưng không có tác dụng nào khác:>>>
Hãy nhớ rằng, nhập một mô-đun vừa tải nội dung vừa tạo không gian tên chứa nội dung. Một vài ví dụ cuối cùng cho thấy rằng có thể cùng một mô-đun là một phần của các không gian tên khác nhau. Chi tiết kỹ thuật: Không gian tên mô-đun được triển khai dưới dạng từ điển Python và có sẵn tại 76thuộc tính:>>>
Bạn hiếm khi cần phải tương tác 76trực tiếp với .Tương tự, không gian tên toàn cục của Python cũng là một từ điển. Bạn có thể truy cập nó thông qua 78.Việc nhập các gói con và mô-đun con trong một 56tệp tin để làm cho chúng dễ sử dụng hơn cho người dùng của bạn là điều khá phổ biến . Bạn có thể xem một ví dụ về điều này trong 80gói phổ biến .Nhập khẩu tuyệt đối và tương đốiNhớ lại mã nguồn của 62trong ví dụ trước đó:Bạn đã từng thấy những 82câu như vậy 83, nhưng dấu chấm ( 84) 85có nghĩa là gì?Dấu chấm đề cập đến gói hiện tại và câu lệnh là một ví dụ về nhập tương đối . Bạn có thể đọc nó là “Từ gói hiện tại, hãy nhập gói con 63”.Có một câu lệnh nhập tuyệt đối tương đương trong đó bạn đặt tên rõ ràng cho gói hiện tại: Trên thực tế, tất cả các hoạt động nhập khẩu vào 70có thể đã được thực hiện một cách rõ ràng với các nhập khẩu tuyệt đối tương tự.Nhập tương đối phải ở dạng 82và vị trí bạn đang nhập phải bắt đầu bằng dấu chấm.Các phong cách dẫn PEP 8 khuyến cáo sử dụng nhập khẩu tuyệt đối nói chung. Tuy nhiên, nhập khẩu tương đối là một giải pháp thay thế để tổ chức phân cấp gói. Để biết thêm thông tin, hãy xem Nhập tuyệt đối so với Nhập tương đối bằng Python . Đường dẫn nhập của PythonLàm cách nào để Python tìm thấy các mô-đun và các gói mà nó nhập vào? Bạn sẽ thấy thêm chi tiết về cơ chế của hệ thống nhập Python sau . Hiện tại, chỉ cần biết rằng Python tìm kiếm các mô-đun và gói trong đường dẫn nhập của nó . Đây là danh sách các vị trí được tìm kiếm các mô-đun để nhập. Lưu ý: Khi bạn nhập 89, Python sẽ tìm kiếm 90một vài vị trí khác nhau trước khi tìm kiếm đường dẫn nhập.Đặc biệt, nó sẽ tìm kiếm trong bộ nhớ cache của mô-đun để xem liệu 90đã được nhập chưa và nó sẽ tìm kiếm trong số các mô-đun được tích hợp sẵn.Bạn sẽ tìm hiểu thêm về máy móc nhập Python đầy đủ trong phần sau . Bạn có thể kiểm tra đường dẫn nhập của Python bằng cách in 92. Nói chung, danh sách này sẽ chứa ba loại địa điểm khác nhau:
Thông thường, Python sẽ bắt đầu ở đầu danh sách các vị trí và tìm kiếm một mô-đun nhất định trong mỗi vị trí cho đến khi khớp đầu tiên. Vì thư mục tập lệnh hoặc thư mục hiện tại luôn nằm đầu tiên trong danh sách này, bạn có thể đảm bảo rằng các tập lệnh của mình tìm thấy các mô-đun và gói tự tạo của bạn bằng cách tổ chức các thư mục của bạn và cẩn thận về việc bạn chạy Python từ thư mục nào. Tuy nhiên, bạn cũng nên cẩn thận rằng bạn không tạo các mô-đun làm bóng hoặc ẩn các mô-đun quan trọng khác. Ví dụ, giả sử rằng bạn xác định 39mô-đun sau : 0Sử dụng mô-đun này hoạt động như mong đợi: >>> 1Nhưng mô-đun này cũng che bóng 39mô-đun được bao gồm trong thư viện tiêu chuẩn. Thật không may, điều đó có nghĩa là ví dụ trước đó của chúng tôi về việc tìm kiếm giá trị của π không còn hoạt động nữa:>>> 2Nhưng mô-đun này cũng che bóng 39mô-đun được bao gồm trong thư viện tiêu chuẩn. Thật không may, điều đó có nghĩa là ví dụ trước đó của chúng tôi về việc tìm kiếm giá trị của π không còn hoạt động nữa:Vấn đề là Python hiện tìm kiếm 39mô-đun mới của bạn 40thay vì tìm kiếm 39mô-đun trong thư viện chuẩn.Để tránh những loại vấn đề này, bạn nên cẩn thận với tên của các mô-đun và gói của mình. Đặc biệt, tên gói và mô-đun cấp cao nhất của bạn phải là duy nhất. Nếu >>> import math >>> dir() ['__annotations__', '__builtins__', ..., 'math'] >>> dir(math) ['__doc__', ..., 'nan', 'pi', 'pow', ...] 39được định nghĩa là một mô-đun con trong một gói, thì nó sẽ không phủ bóng mô-đun tích hợp sẵn.Ví dụ: Cấu trúc nhập khẩu của bạn 3Mặc dù bạn có thể sắp xếp việc nhập bằng cách sử dụng thư mục hiện tại cũng như bằng cách thao tác 93và thậm chí 92, quá trình này thường không cầu kỳ và dễ xảy ra lỗi. Để xem một ví dụ điển hình, hãy xem xét ứng dụng sau: 4
Ứng dụng sẽ tạo lại cấu trúc tệp nhất định bằng cách tạo thư mục và tệp trống. Các 02tập tin có chứa các kịch bản chính, và 03là một module thư viện với một vài chức năng để đối phó với các tập tin. Sau đây là một ví dụ về kết xuất từ ứng dụng, trong trường hợp này là chạy nó trong 04thư mục:Hai tệp mã nguồn cũng như 05tệp được tạo tự động được tạo lại bên trong một thư mục mới có tên 06. 5Bây giờ hãy xem mã nguồn. Chức năng chính của ứng dụng được xác định trong 02:Trong các dòng từ 12 đến 16 , bạn đọc một đường dẫn gốc từ dòng lệnh. Trong ví dụ trên, bạn sử dụng dấu chấm, có nghĩa là thư mục hiện tại. Đường dẫn này sẽ được sử dụng làm 08phân cấp tệp mà bạn sẽ tạo lại.Công việc thực tế xảy ra từ dòng 19 đến dòng 23 . Đầu tiên, bạn tạo một đường dẫn duy nhất 09, đó sẽ là đường dẫn gốc của hệ thống phân cấp tệp mới của bạn. Sau đó, bạn lặp qua tất cả các đường dẫn bên dưới bản gốc 08và tạo lại chúng dưới dạng tệp trống bên trong hệ thống phân cấp tệp mới.Đối với thao tác với các đường dẫn như thế này, 11trong thư viện tiêu chuẩn là khá hữu ích. Để biết thêm chi tiết về cách nó được sử dụng, hãy xem Mô-đun của Python 3 11: Điều chỉnh hệ thống tệp .
Lưu ý rằng bạn nhập 18trên dòng 8 . Mô-đun thư viện này chứa hai chức năng tiện ích: 6 19 sử dụng bộ đếm để tìm đường dẫn chưa tồn tại. Trong ứng dụng, bạn sử dụng nó để tìm một thư mục con duy nhất để sử dụng làm 09hệ thống phân cấp tệp được tạo lại. Tiếp theo, 21hãy đảm bảo rằng tất cả các thư mục cần thiết đã được tạo trước khi tạo một tệp trống bằng cách sử dụng 22.Hãy xem xét việc nhập 18lại: 7Nó trông khá ngây thơ. Tuy nhiên, khi dự án phát triển, dòng này sẽ khiến bạn đau đầu. Ngay cả khi bạn nhập 18từ 04dự án, quá trình nhập là tuyệt đối : nó không bắt đầu bằng dấu chấm. Điều này có nghĩa là 18phải được tìm thấy trong đường dẫn nhập để nhập hoạt động.May mắn thay, thư mục chứa tập lệnh hiện tại luôn nằm trong đường dẫn nhập của Python, vì vậy điều này hiện hoạt động tốt. Tuy nhiên, nếu dự án của bạn đạt được một số lực kéo, thì nó có thể được sử dụng theo những cách khác. Ví dụ: một người nào đó có thể muốn nhập tập lệnh vào một Máy tính xách tay Jupyter và chạy nó từ đó. Hoặc họ có thể muốn sử dụng lại 18thư viện trong một dự án khác. Họ thậm chí có thể tạo tệp thực thi bằng PyInstaller để dễ dàng phân phối tệp đó hơn. Thật không may, bất kỳ trường hợp nào trong số này đều có thể tạo ra sự cố với việc nhập 18. Để xem ví dụ, bạn có thể làm theo hướng dẫn PyInstaller và tạo một điểm vào ứng dụng của mình . Thêm một thư mục bổ sung bên ngoài thư mục ứng dụng của bạn: 8Trong thư mục bên ngoài, tạo ra các kịch bản điểm nhập cảnh, 29: 9Tập lệnh này sẽ nhập 13từ tập lệnh gốc của bạn và chạy nó. Lưu ý rằng 13không chạy khi 04được nhập vì 14thử nghiệm trên dòng 25 in 02. Điều đó có nghĩa là bạn cần chạy 13một cách rõ ràng.Về lý thuyết, điều này sẽ hoạt động tương tự như chạy ứng dụng trực tiếp: 0Tại sao điều đó không hoạt động? Đột nhiên, quá trình nhập 18phát sinh lỗi.Vấn đề là bằng cách khởi động ứng dụng bằng 29, bạn đã thay đổi vị trí của tập lệnh hiện tại, từ đó thay đổi đường dẫn nhập. 18không còn trên đường dẫn nhập nữa, vì vậy không thể nhập tuyệt đối.Một giải pháp khả thi là thay đổi đường dẫn nhập của Python: 1Điều này hoạt động vì đường dẫn nhập bao gồm thư mục chứa 02và 03. Vấn đề với cách tiếp cận này là đường dẫn nhập của bạn có thể rất lộn xộn và khó hiểu.Trong thực tế, bạn đang tạo lại một tính năng của các phiên bản Python đầu tiên được gọi là nhập tương đối ngầm định . Những điều này đã bị loại bỏ khỏi ngôn ngữ bởi PEP 328 với lý do sau:
Một giải pháp khác là sử dụng nhập tương đối để thay thế. Thay đổi nhập 02như sau: 2Bây giờ bạn có thể khởi động ứng dụng của mình thông qua tập lệnh điểm nhập: 3Rất tiếc, bạn không thể gọi ứng dụng trực tiếp được nữa: 4Vấn đề là việc nhập tương đối được giải quyết trong các tập lệnh khác với các mô-đun được nhập. Tất nhiên, bạn có thể quay lại và khôi phục quá trình nhập tuyệt đối trước khi chạy trực tiếp tập lệnh hoặc thậm chí bạn có thể thực hiện một số động tác 43nhào lộn để nhập tệp hoàn toàn hoặc tương đối tùy thuộc vào những gì hoạt động.Thậm chí còn có một vụ hack được chính thức xử phạt để làm cho việc nhập tương đối hoạt động trong các tập lệnh. Thật không may, điều này cũng buộc bạn phải thay đổi 92trong hầu hết các trường hợp. Trích lời Raymond Hettinger :
Thật vậy, một giải pháp tốt hơn — và ổn định hơn — là chơi cùng với hệ thống nhập và đóng gói của Python và cài đặt dự án của bạn dưới dạng gói cục bộ bằng cách sử dụng 45 .Tạo và cài đặt gói cục bộKhi bạn cài đặt một gói từ PyPI , gói đó có sẵn cho tất cả các tập lệnh trong môi trường của bạn. Tuy nhiên, bạn cũng có thể cài đặt các gói từ máy tính cục bộ của mình và chúng cũng sẽ được cung cấp theo cách tương tự. Tạo một gói cục bộ không liên quan nhiều đến chi phí. Đầu tiên, tạo tối thiểu 46và 47các tệp trong 04thư mục bên ngoài : 5Về lý thuyết, cái 49và 50có thể là bất cứ thứ gì bạn thích. Tuy nhiên, chúng sẽ được sử dụng 45khi tham chiếu đến gói của bạn, vì vậy bạn nên chọn các giá trị dễ nhận biết và không đụng chạm với các gói khác mà bạn sử dụng.
Một mẹo là cung cấp cho tất cả các gói cục bộ như vậy một tiền tố chung như 52hoặc tên người dùng của bạn. 53nên liệt kê thư mục hoặc các thư mục chứa mã nguồn của bạn. Sau đó, bạn có thể cài đặt gói cục bộ bằng cách sử dụng 45: 6Lệnh này sẽ cài đặt gói vào hệ thống của bạn. 04sau đó sẽ được tìm thấy trên đường dẫn nhập của Python, có nghĩa là bạn có thể sử dụng nó ở bất cứ đâu mà không phải lo lắng về thư mục tập lệnh, nhập tương đối hoặc các biến chứng khác. Các 56tùy chọn là viết tắt của có thể chỉnh sửa , đó là quan trọng vì nó cho phép bạn thay đổi mã nguồn của gói của bạn mà không cần cài đặt lại nó.Lưu ý: Loại tệp thiết lập này hoạt động tốt khi bạn đang làm việc với các dự án của riêng mình. Tuy nhiên, nếu bạn định chia sẻ mã với người khác, thì bạn nên thêm một số thông tin vào tệp thiết lập của mình. Để biết thêm chi tiết về các tệp thiết lập, hãy xem Cách xuất bản Gói Python nguồn mở lên PyPI . Bây giờ nó đã 04được cài đặt trên hệ thống của bạn, bạn có thể sử dụng câu lệnh nhập sau: 7Điều này sẽ hoạt động bất kể bạn kết thúc cuộc gọi ứng dụng của mình như thế nào. Mẹo: Trong mã của riêng bạn, bạn nên tách các tập lệnh và thư viện một cách có ý thức. Đây là một nguyên tắc nhỏ:
Bạn có thể có mã mà bạn muốn vừa chạy riêng vừa nhập từ các tập lệnh khác. Trong trường hợp đó, thường nên cấu trúc lại mã của bạn để bạn chia phần chung thành một mô-đun thư viện. Mặc dù tách các tập lệnh và thư viện là một ý tưởng hay, nhưng tất cả các tệp Python đều có thể được thực thi và nhập. Trong phần sau , bạn sẽ tìm hiểu thêm về cách tạo các mô-đun xử lý tốt cả hai. Gói không gian tênCác mô-đun và gói Python có liên quan rất chặt chẽ đến các tệp và thư mục. Điều này làm cho Python khác biệt với nhiều ngôn ngữ lập trình khác, trong đó các gói chỉ hoạt động như không gian tên mà không thực thi cách tổ chức mã nguồn. Xem thảo luận trong PEP 402 để biết các ví dụ. Các gói không gian tên đã có sẵn bằng Python kể từ phiên bản 3.3. Chúng ít phụ thuộc hơn vào hệ thống phân cấp tệp cơ bản. Đặc biệt, các gói không gian tên có thể được chia thành nhiều thư mục. Gói không gian tên được tạo tự động nếu bạn có thư mục chứa 37tệp nhưng không có 56. Xem PEP 420 để biết giải thích chi tiết.Lưu ý: Nói một cách chính xác, các gói không gian tên ngầm định đã được giới thiệu trong Python 3.3. Trong các phiên bản Python trước, bạn có thể tạo các gói vùng tên theo cách thủ công theo một số cách không tương thích khác nhau . PEP 420 thống nhất và đơn giản hóa các phương pháp tiếp cận trước đó. Để hiểu rõ hơn về lý do tại sao các gói không gian tên có thể hữu ích, chúng ta hãy thử triển khai một gói. Như một ví dụ thúc đẩy, bạn sẽ có một bước đi khác về vấn đề được giải quyết trong Mô hình phương pháp nhà máy và Triển khai của nó bằng Python : với một 60đối tượng, bạn muốn chuyển đổi nó thành một trong số các biểu diễn chuỗi. Nói cách khác, bạn muốn tuần tự hóa 60 các đối tượng.Để cụ thể hơn, bạn muốn triển khai mã hoạt động như sau: >>> 8Hãy giả sử rằng bạn may mắn và bắt gặp việc triển khai của bên thứ ba một số định dạng mà bạn cần tuần tự hóa tới và nó được tổ chức như một gói không gian tên: 9Tệp 62chứa mã có thể tuần tự hóa một đối tượng sang định dạng JSON : 0Giao diện serializer này có một chút hạn chế, nhưng nó sẽ đủ để chứng minh cách hoạt động của các gói không gian tên. Tệp 63chứa một tệp tương tự 64có thể chuyển đổi một đối tượng sang XML : 1Lưu ý rằng cả hai lớp thực hiện cùng một giao diện với 65, 66và 67phương pháp.Sau đó, bạn tạo một 60lớp có thể sử dụng các trình tuần tự này: 2A 60được xác định bởi ID, chức danh và nghệ sĩ của nó. Lưu ý rằng 70không cần biết nó chuyển đổi sang định dạng nào vì nó sử dụng giao diện chung được xác định trước đó.Giả sử rằng bạn đã cài đặt 71gói của bên thứ ba , bạn có thể sử dụng nó như sau:>>> 3Bằng cách cung cấp các đối tượng serializer khác nhau 70, bạn sẽ có được các bản trình bày khác nhau cho bài hát của mình.Lưu ý: Bạn có thể nhận được một 73hoặc một 74khi tự chạy mã. Điều này là do 71không có trong đường dẫn nhập Python của bạn . Bạn sẽ sớm thấy cách giải quyết vấn đề đó.Càng xa càng tốt. Tuy nhiên, bây giờ bạn nhận ra rằng bạn cũng cần phải chuyển đổi các bài hát của mình thành biểu diễn YAML , không được hỗ trợ trong thư viện của bên thứ ba. Nhập sự kỳ diệu của các gói không gian tên: bạn có thể thêm gói của riêng mình 76vào 71gói mà không cần chạm vào thư viện của bên thứ ba.Đầu tiên, tạo một thư mục trên hệ thống tệp cục bộ của bạn có tên 71. Điều quan trọng là tên của thư mục phải khớp với tên của gói không gian tên mà bạn đang tùy chỉnh: 4Trong 79tệp, bạn xác định của riêng bạn 76. Bạn căn cứ vào 81gói này , gói phải được cài đặt từ PyPI: 5Vì YAML và JSON là các định dạng khá giống nhau, bạn có thể sử dụng lại hầu hết việc triển khai 82: 6Lưu ý rằng dấu 76dựa trên 82, được nhập từ 71chính nó. Kể từ khi cả hai 86và 87là một phần của gói namespace cùng, thậm chí bạn có thể sử dụng một khẩu tương đối: 88.Tiếp tục ví dụ trên, bây giờ bạn cũng có thể chuyển đổi bài hát sang YAML: >>> 7Bằng cách cung cấp các đối tượng serializer khác nhau 70, bạn sẽ có được các bản trình bày khác nhau cho bài hát của mình.Lưu ý: Bạn có thể nhận được một 73hoặc một 74khi tự chạy mã. Điều này là do 71không có trong đường dẫn nhập Python của bạn . Bạn sẽ sớm thấy cách giải quyết vấn đề đó.Càng xa càng tốt. Tuy nhiên, bây giờ bạn nhận ra rằng bạn cũng cần phải chuyển đổi các bài hát của mình thành biểu diễn YAML , không được hỗ trợ trong thư viện của bên thứ ba. Nhập sự kỳ diệu của các gói không gian tên: bạn có thể thêm gói của riêng mình 76vào 71gói mà không cần chạm vào thư viện của bên thứ ba.Đầu tiên, tạo một thư mục trên hệ thống tệp cục bộ của bạn có tên 71. Điều quan trọng là tên của thư mục phải khớp với tên của gói không gian tên mà bạn đang tùy chỉnh:Trong 79tệp, bạn xác định của riêng bạn 76. Bạn căn cứ vào 81gói này , gói phải được cài đặt từ PyPI:>>> 8Bằng cách cung cấp các đối tượng serializer khác nhau 70, bạn sẽ có được các bản trình bày khác nhau cho bài hát của mình.Lưu ý: Bạn có thể nhận được một >>> from math import pi >>> pi 3.141592653589793 >>> math.pi NameError: name 'math' is not defined 73hoặc một >>> from math import pi >>> pi 3.141592653589793 >>> math.pi NameError: name 'math' is not defined 74khi tự chạy mã. Điều này là do >>> from math import pi >>> pi 3.141592653589793 >>> math.pi NameError: name 'math' is not defined 71không có trong đường dẫn nhập Python của bạn . Bạn sẽ sớm thấy cách giải quyết vấn đề đó.Càng xa càng tốt. Tuy nhiên, bây giờ bạn nhận ra rằng bạn cũng cần phải chuyển đổi các bài hát của mình thành biểu diễn YAML , không được hỗ trợ trong thư viện của bên thứ ba. Nhập sự kỳ diệu của các gói không gian tên: bạn có thể thêm gói của riêng mình 76vào 71gói mà không cần chạm vào thư viện của bên thứ ba.
99và 00là những công cụ tuyệt vời để thực thi một phong cách nhất quán cho các mục nhập của bạn.Dưới đây là ví dụ về phần nhập bên trong gói trình đọc nguồn cấp dữ liệu Python thực : 9Lưu ý cách nhóm này làm cho các phụ thuộc của mô-đun này rõ ràng: 01và 02cần được cài đặt trên hệ thống. Nhìn chung, bạn có thể cho rằng thư viện tiêu chuẩn có sẵn. Việc tách các mục nhập từ bên trong gói của bạn cung cấp cho bạn một số tổng quan về các phần phụ thuộc nội bộ của mã của bạn.Có những trường hợp nên uốn cong những quy tắc này một chút. Bạn đã thấy rằng nhập khẩu tương đối có thể là một giải pháp thay thế cho việc tổ chức phân cấp gói. Sau đó , bạn sẽ thấy cách trong một số trường hợp, bạn có thể chuyển nhập vào một định nghĩa hàm để phá vỡ các chu kỳ nhập. Nhập tài nguyênĐôi khi bạn sẽ có mã phụ thuộc vào tệp dữ liệu hoặc các tài nguyên khác. Trong các tập lệnh nhỏ, đây không phải là vấn đề — bạn có thể chỉ định đường dẫn đến tệp dữ liệu của mình và tiếp tục! Tuy nhiên, nếu tệp tài nguyên quan trọng đối với gói của bạn và bạn muốn phân phối gói của mình cho những người dùng khác, thì một số thách thức sẽ phát sinh:
Đã có một số nỗ lực để giải quyết những thách thức này, bao gồm cả 06. Tuy nhiên, với việc đưa 07vào thư viện tiêu chuẩn trong Python 3.7 , bây giờ có một cách tiêu chuẩn để xử lý các tệp tài nguyên.Giới thiệu importlib.resources 07cấp quyền truy cập vào các tài nguyên trong các gói. Trong ngữ cảnh này, tài nguyên là bất kỳ tệp nào nằm trong một gói có thể nhập. Tệp có thể có hoặc không tương ứng với tệp vật lý trên hệ thống tệp.Điều này có một vài lợi thế. Bằng cách sử dụng lại hệ thống nhập, bạn sẽ có cách xử lý các tệp bên trong các gói của mình một cách nhất quán hơn. Nó cũng cho phép bạn truy cập dễ dàng hơn vào các tệp tài nguyên trong các gói khác. Tài liệu tóm tắt nó một cách độc đáo:
07đã trở thành một phần của thư viện chuẩn trong Python 3.7. Tuy nhiên, trên các phiên bản cũ hơn của Python, một cổng hỗ trợ có sẵn dưới dạng 10 . Để sử dụng backport, hãy cài đặt nó từ PyPI : 0Backport tương thích với Python 2.7 cũng như Python 3.4 và các phiên bản mới hơn. Có một yêu cầu khi sử dụng 07: các tệp tài nguyên của bạn phải có sẵn bên trong một gói thông thường. Gói không gian tên không được hỗ trợ. Trong thực tế, điều này có nghĩa là tệp phải nằm trong thư mục chứa 56tệp.Ví dụ đầu tiên, giả sử bạn có tài nguyên bên trong một gói như thế này: 1 56chỉ là một tệp trống cần thiết để chỉ định 14là một gói thông thường.Sau đó, bạn có thể sử dụng 15và 16để mở các tệp văn bản và tệp nhị phân, tương ứng:>>> 2 15và 16tương đương với built-in 19với các 20tham số thiết lập để 21và 22, tương ứng. Các chức năng thuận tiện để đọc trực tiếp văn bản hoặc tệp nhị phân cũng có sẵn như 23và 24. Xem tài liệu chính thức để biết thêm thông tin.Lưu ý: Để sử dụng backport một cách liền mạch trên các phiên bản Python cũ hơn, bạn có thể nhập 07như sau: 3Xem phần mẹo và thủ thuật của hướng dẫn này để biết thêm thông tin. Phần còn lại của phần này sẽ trình bày một vài ví dụ phức tạp về việc sử dụng tệp tài nguyên trong thực tế. Ví dụ: Sử dụng tệp dữ liệuLà một ví dụ đầy đủ hơn về việc sử dụng tệp dữ liệu, bạn sẽ thấy cách triển khai chương trình đố vui dựa trên dữ liệu dân số của Liên hợp quốc . Đầu tiên, hãy tạo một 26gói và tải xuống 27từ trang web của Liên hợp quốc : 4Mở tệp CSV và xem dữ liệu: 5Mỗi dòng chứa dân số của một quốc gia trong một năm nhất định và một biến thể nhất định, cho biết loại kịch bản nào được sử dụng cho dự báo. Tệp này chứa các dự báo về dân số cho đến năm 2100. Hàm sau đọc tệp này và chọn ra tổng dân số của mỗi quốc gia cho một 28và 29: 6Các dòng được đánh dấu hiển thị cách 07được sử dụng để mở tệp dữ liệu. Để biết thêm thông tin về cách làm việc với tệp CSV, hãy xem Đọc và Viết tệp CSV bằng Python .Hàm trên trả về một từ điển với các số dân số: >>> 7Bạn có thể làm bất kỳ điều gì thú vị với từ điển dân số này, bao gồm cả phân tích và hình dung. Tại đây, bạn sẽ tạo một trò chơi đố yêu cầu người dùng xác định quốc gia nào trong tập hợp là đông dân nhất. Chơi trò chơi sẽ trông giống như sau: 8Các chi tiết của việc thực hiện nằm quá xa chủ đề của hướng dẫn này, vì vậy chúng sẽ không được thảo luận ở đây. Tuy nhiên, bạn có thể mở rộng phần bên dưới để xem mã nguồn hoàn chỉnh. Ví dụ: Thêm biểu tượng vào Tkinter GUIsKhi xây dựng giao diện người dùng đồ họa (GUI), bạn thường cần bao gồm các tệp tài nguyên như biểu tượng. Ví dụ sau đây cho thấy cách bạn có thể làm điều đó bằng cách sử dụng 07. Ứng dụng cuối cùng sẽ trông khá cơ bản, nhưng nó sẽ có biểu tượng tùy chỉnh cũng như hình minh họa trên nút Tạm biệt :Ví dụ sử dụng Tkinter , là một gói GUI có sẵn trong thư viện tiêu chuẩn. Nó dựa trên hệ thống cửa sổ Tk , ban đầu được phát triển cho ngôn ngữ lập trình Tcl. Có nhiều gói GUI khác có sẵn cho Python. Nếu bạn đang sử dụng một biểu tượng khác, thì bạn sẽ có thể thêm các biểu tượng vào ứng dụng của mình bằng cách sử dụng các ý tưởng tương tự như những biểu tượng được trình bày ở đây. Trong Tkinter, hình ảnh được xử lý bởi 32lớp . Để tạo một 32, bạn chuyển một đường dẫn đến một tệp hình ảnh.Hãy nhớ rằng, khi phân phối gói của bạn, bạn thậm chí không được đảm bảo rằng các tệp tài nguyên sẽ tồn tại dưới dạng tệp vật lý trên hệ thống tệp. 07giải quyết điều này bằng cách cung cấp 35. Hàm này sẽ trả về một đường dẫn đến tệp tài nguyên, tạo một tệp tạm thời nếu cần.Để đảm bảo mọi tệp tạm thời được dọn dẹp đúng cách, bạn nên sử dụng 35làm trình quản lý ngữ cảnh bằng từ khóa 37:>>> 9Bạn có thể làm bất kỳ điều gì thú vị với từ điển dân số này, bao gồm cả phân tích và hình dung. Tại đây, bạn sẽ tạo một trò chơi đố yêu cầu người dùng xác định quốc gia nào trong tập hợp là đông dân nhất. Chơi trò chơi sẽ trông giống như sau: 0Các chi tiết của việc thực hiện nằm quá xa chủ đề của hướng dẫn này, vì vậy chúng sẽ không được thảo luận ở đây. Tuy nhiên, bạn có thể mở rộng phần bên dưới để xem mã nguồn hoàn chỉnh. Ví dụ: Thêm biểu tượng vào Tkinter GUIs Khi xây dựng giao diện người dùng đồ họa (GUI), bạn thường cần bao gồm các tệp tài nguyên như biểu tượng. Ví dụ sau đây cho thấy cách bạn có thể làm điều đó bằng cách sử dụng 07. Ứng dụng cuối cùng sẽ trông khá cơ bản, nhưng nó sẽ có biểu tượng tùy chỉnh cũng như hình minh họa trên nút Tạm biệt :Ví dụ sử dụng Tkinter , là một gói GUI có sẵn trong thư viện tiêu chuẩn. Nó dựa trên hệ thống cửa sổ Tk , ban đầu được phát triển cho ngôn ngữ lập trình Tcl. Có nhiều gói GUI khác có sẵn cho Python. Nếu bạn đang sử dụng một biểu tượng khác, thì bạn sẽ có thể thêm các biểu tượng vào ứng dụng của mình bằng cách sử dụng các ý tưởng tương tự như những biểu tượng được trình bày ở đây. 1Trong Tkinter, hình ảnh được xử lý bởi 32lớp . Để tạo một 32, bạn chuyển một đường dẫn đến một tệp hình ảnh.Hãy nhớ rằng, khi phân phối gói của bạn, bạn thậm chí không được đảm bảo rằng các tệp tài nguyên sẽ tồn tại dưới dạng tệp vật lý trên hệ thống tệp. 07giải quyết điều này bằng cách cung cấp 35. Hàm này sẽ trả về một đường dẫn đến tệp tài nguyên, tạo một tệp tạm thời nếu cần.Để đảm bảo mọi tệp tạm thời được dọn dẹp đúng cách, bạn nên sử dụng 35làm trình quản lý ngữ cảnh bằng từ khóa 37:Đối với ví dụ đầy đủ, giả sử bạn có cấu trúc phân cấp tệp sau:Nếu bạn muốn tự mình thử ví dụ, thì bạn có thể tải xuống các tệp này cùng với phần còn lại của mã nguồn được sử dụng trong hướng dẫn này bằng cách nhấp vào liên kết bên dưới: >>> 2
Bạn có thể làm bất kỳ điều gì thú vị với từ điển dân số này, bao gồm cả phân tích và hình dung. Tại đây, bạn sẽ tạo một trò chơi đố yêu cầu người dùng xác định quốc gia nào trong tập hợp là đông dân nhất. Chơi trò chơi sẽ trông giống như sau: Lưu ý: Trong ví dụ trên, bạn xác định lại 46bằng cách sử dụng một hàm lambda. Bạn cũng có thể sử dụng một định nghĩa hàm bình thường:>>> 3Để tìm hiểu thêm về các hàm lambda, hãy xem Cách sử dụng các hàm Lambda trong Python . Trong phần này, bạn sẽ học cách nhập động bằng Python. Với chúng, bạn sẽ không phải quyết định nhập những gì cho đến khi chương trình của bạn đang chạy. Sử dụng importlibCho đến nay, bạn đã sử dụng 35từ khóa của Python để nhập các mô-đun và gói một cách rõ ràng. Tuy nhiên, toàn bộ máy móc nhập khẩu có sẵn trong 53gói và điều này cho phép bạn thực hiện việc nhập khẩu của mình một cách linh hoạt hơn. Tập lệnh sau yêu cầu người dùng nhập tên mô-đun, nhập mô-đun đó và in chuỗi tài liệu của nó: 4 54trả về một đối tượng mô-đun mà bạn có thể liên kết với bất kỳ biến nào. Sau đó, bạn có thể coi biến đó như một mô-đun được nhập thường xuyên. Bạn có thể sử dụng tập lệnh như sau: 5Trong mỗi trường hợp, mô-đun được nhập động bằng 54.Ví dụ: Phương pháp ban đầu với các gói không gian tênHãy nghĩ lại ví dụ về serializers trước đó. Với 71việc triển khai dưới dạng gói không gian tên, bạn có khả năng thêm bộ tuần tự tùy chỉnh. Trong ví dụ ban đầu từ một hướng dẫn trước, các bộ nối tiếp được tạo sẵn thông qua một nhà máy sản xuất bộ nối tiếp. Sử dụng 53, bạn có thể làm điều gì đó tương tự.Thêm mã sau vào 71gói không gian tên cục bộ của bạn : 6Nhà 59máy có thể tạo bộ tuần tự động dựa trên 60tham số và 61sau đó có thể áp dụng bộ tuần tự cho bất kỳ đối tượng nào triển khai một 70phương thức.Nhà máy đưa ra một số giả định mạnh mẽ về việc đặt tên cho cả mô-đun và lớp chứa các bộ tuần tự riêng lẻ. Trong phần tiếp theo , bạn sẽ tìm hiểu về kiến trúc plugin cho phép linh hoạt hơn. Bây giờ bạn có thể tạo lại ví dụ trước đó như sau: >>> 7Trong trường hợp này, bạn không cần nhập từng bộ nối tiếp một cách rõ ràng nữa. Thay vào đó, bạn chỉ định tên của bộ nối tiếp bằng một chuỗi. Chuỗi thậm chí có thể được chọn bởi người dùng của bạn trong thời gian chạy. Lưu ý: Trong một gói thông thường, bạn có thể đã triển khai 59và 61trong một 56tệp. Điều đó sẽ cho phép bạn chỉ cần nhập 71và sau đó gọi 67.Tuy nhiên, các gói không gian tên không được phép sử dụng 56, vì vậy bạn cần phải triển khai các chức năng này trong một mô-đun riêng biệt.Ví dụ cuối cùng cho thấy rằng bạn cũng nhận được một thông báo lỗi khá nếu bạn cố gắng tuần tự hóa sang một định dạng chưa được triển khai. Ví dụ: Một gói các pluginHãy xem xét một ví dụ khác về việc sử dụng nhập động. Bạn có thể sử dụng mô-đun sau để thiết lập kiến trúc plugin linh hoạt trong mã của mình. Điều này tương tự như ví dụ trước, trong đó bạn có thể cắm bộ tuần tự cho các định dạng khác nhau bằng cách thêm các mô-đun mới. Một ứng dụng sử dụng plugin hiệu quả là công cụ trực quan hóa khám phá Keo . Keo có thể đọc nhiều định dạng dữ liệu khác nhau ra khỏi hộp. Tuy nhiên, nếu định dạng dữ liệu của bạn không được hỗ trợ, thì bạn có thể viết trình tải dữ liệu tùy chỉnh của riêng mình . Bạn làm điều này bằng cách thêm một chức năng mà bạn trang trí và đặt ở một vị trí đặc biệt để Keo dễ dàng tìm thấy. Bạn không cần phải thay đổi bất kỳ phần nào của mã nguồn Keo. Xem tài liệu để biết tất cả các chi tiết. Bạn có thể thiết lập một kiến trúc plugin tương tự mà bạn có thể sử dụng trong các dự án của riêng mình. Trong kiến trúc, có hai cấp độ:
Các 69mô-đun đó cho thấy nhiều kiến trúc plugin có các chức năng sau: 8Các chức năng gốc được sử dụng để thêm chức năng vào các gói plugin một cách thuận tiện. Bạn sẽ thấy một số ví dụ về cách chúng được sử dụng trong thời gian ngắn. Xem xét tất cả các chi tiết của mã này nằm ngoài phạm vi của hướng dẫn này. Nếu bạn quan tâm, thì bạn có thể xem cách triển khai bằng cách mở rộng phần bên dưới. Hãy xem một số ví dụ về cách sử dụng plugin. Ví dụ đầu tiên là một 70gói mà bạn có thể sử dụng để thêm nhiều lời chào khác nhau vào ứng dụng của mình. Một kiến trúc plugin đầy đủ chắc chắn là quá mức cần thiết đối với ví dụ này, nhưng nó cho thấy cách các plugin hoạt động.
Giả sử bạn có 70gói sau : 9Mỗi 70mô-đun xác định một hàm nhận một 49đối số. Lưu ý cách tất cả chúng được đăng ký làm plugin bằng trình 74trang trí: 0Để tìm hiểu thêm về trình trang trí và cách chúng được sử dụng, hãy xem Primer trên Trình trang trí Python . Lưu ý: Để đơn giản hóa việc khám phá và nhập các plugin, tên của mỗi plugin dựa trên tên của mô-đun chứa nó thay vì tên chức năng. Điều này hạn chế bạn chỉ có một plugin cho mỗi tệp. Để hoàn tất thiết lập 70dưới dạng gói plugin, bạn có thể sử dụng các chức năng của nhà máy 69để thêm chức năng vào 70chính gói: 1Bây giờ bạn có thể sử dụng 78và 79như sau:>>> 2Lưu ý rằng 78tự động phát hiện tất cả các plugin có sẵn trong gói.Bạn cũng có thể chọn một cách linh hoạt hơn plugin nào để gọi. Trong ví dụ sau, bạn chọn một plugin một cách ngẫu nhiên. Tuy nhiên, bạn cũng có thể chọn một plugin dựa trên tệp cấu hình hoặc thông tin người dùng nhập: >>> 3Lưu ý rằng 78tự động phát hiện tất cả các plugin có sẵn trong gói. 4Bạn cũng có thể chọn một cách linh hoạt hơn plugin nào để gọi. Trong ví dụ sau, bạn chọn một plugin một cách ngẫu nhiên. Tuy nhiên, bạn cũng có thể chọn một plugin dựa trên tệp cấu hình hoặc thông tin người dùng nhập:
74 decorator được xác định bên trong mỗi mô-đun plugin đăng ký mỗi plugin đã nhập.
Mỗi tệp Python không bắt đầu bằng dấu gạch dưới được nhập. Các plugin trong bất kỳ tệp nào được phát hiện và đăng ký. 5Hãy kết thúc phần này với phiên bản cuối cùng của gói không gian tên serializers . Một vấn đề nổi bật là 59nhà máy đã đưa ra những giả định mạnh mẽ về việc đặt tên cho các lớp serializer. Bạn có thể làm cho điều này linh hoạt hơn bằng cách sử dụng các plugin. 6Đầu tiên, thêm một dòng đăng ký từng bộ tuần tự. Đây là một ví dụ về cách nó được thực hiện trong bộ 87tuần tự:Tiếp theo, cập nhật 91để sử dụng 69:Bạn thực hiện >>> import math as m >>> m.pi 3.141592653589793 >>> from math import pi as PI >>> PI 3.141592653589793 59bằng cách sử dụng >>> import math as m >>> m.pi 3.141592653589793 >>> from math import pi as PI >>> PI 3.141592653589793 94vì điều đó sẽ tự động khởi tạo từng bộ nối tiếp. Với việc tái cấu trúc này, bộ tuần tự hoạt động giống như trước đó. Tuy nhiên, bạn có thể linh hoạt hơn trong việc đặt tên cho các lớp serializer của mình.Để biết thêm thông tin về cách sử dụng plugin, hãy xem PyPlugs trên PyPI và Plugin: Thêm tính linh hoạt cho bản trình bày Ứng dụng của bạn từ PyCon 2019 . Hệ thống nhập Python Bạn đã thấy nhiều cách để tận dụng hệ thống nhập của Python. Trong phần này, bạn sẽ tìm hiểu thêm một chút về những gì xảy ra đằng sau khi các mô-đun và gói được nhập.Như với hầu hết các phần của Python, hệ thống nhập có thể được tùy chỉnh. Bạn sẽ thấy một số cách mà bạn có thể thay đổi hệ thống nhập, bao gồm tự động tải xuống các gói bị thiếu từ PyPI và nhập tệp dữ liệu như thể chúng là mô-đun.
Nạp vào Bị ràng buộc với một không gian tên >>> 7Lưu ý rằng 78tự động phát hiện tất cả các plugin có sẵn trong gói.Bạn cũng có thể chọn một cách linh hoạt hơn plugin nào để gọi. Trong ví dụ sau, bạn chọn một plugin một cách ngẫu nhiên. Tuy nhiên, bạn cũng có thể chọn một plugin dựa trên tệp cấu hình hoặc thông tin người dùng nhập: >>> 8Lưu ý rằng 78tự động phát hiện tất cả các plugin có sẵn trong gói.Bộ đệm ẩn mô-đun đóng một vai trò rất quan trọng trong hệ thống nhập Python. Nơi đầu tiên Python tìm kiếm các mô-đun khi thực hiện nhập là ở trong 98. Nếu một mô-đun đã có sẵn, thì mô-đun đó sẽ không được tải lại.Đây là một sự tối ưu hóa tuyệt vời, nhưng nó cũng là một điều cần thiết. Nếu các mô-đun được tải lại mỗi khi chúng được nhập, thì bạn có thể gặp phải sự mâu thuẫn trong một số trường hợp nhất định, chẳng hạn như khi mã nguồn cơ bản thay đổi trong khi tập lệnh đang chạy. Nhớ lại đường dẫn nhập bạn đã thấy trước đó. Về cơ bản, nó cho Python biết nơi tìm kiếm các mô-đun. Tuy nhiên, nếu Python tìm thấy một mô-đun trong bộ đệm ẩn của mô-đun, thì nó sẽ không bận tâm tìm kiếm đường dẫn nhập cho mô-đun. Ví dụ: Singletons dưới dạng Mô-đunTrong lập trình hướng đối tượng , một singleton là một lớp có nhiều nhất một thể hiện. Mặc dù có thể triển khai các singleton bằng Python , nhưng thay vào đó, hầu hết các ứng dụng tốt của các singleton có thể được xử lý bởi các mô-đun. Bạn có thể tin tưởng bộ đệm ẩn mô-đun để khởi tạo một lớp chỉ một lần. Để làm ví dụ, hãy quay lại dữ liệu dân số của Liên hợp quốc mà bạn đã thấy trước đó . Mô-đun sau định nghĩa một lớp bao bọc dữ liệu dân số: 9Việc đọc dữ liệu từ đĩa mất một chút thời gian. Vì bạn không mong đợi tệp dữ liệu thay đổi, bạn khởi tạo lớp khi bạn tải mô-đun. Tên của lớp bắt đầu bằng dấu gạch dưới để cho người dùng biết rằng họ không nên sử dụng nó. Bạn có thể sử dụng 00singleton để tạo biểu đồ Matplotlib hiển thị dự báo dân số cho các quốc gia đông dân nhất:>>> 0Điều này tạo ra một biểu đồ như sau: Lưu ý rằng tải dữ liệu vào thời điểm nhập khẩu là một loại antipattern . Tốt nhất, bạn muốn hàng nhập khẩu của mình không có tác dụng phụ càng tốt. Một cách tiếp cận tốt hơn là tải dữ liệu một cách lười biếng khi bạn cần. Bạn có thể làm điều này một cách khá thanh lịch bằng cách sử dụng thuộc tính Mở rộng phần sau để xem ví dụ. Đang tải lại các mô-đunBộ nhớ cache của mô-đun có thể hơi khó chịu khi bạn đang làm việc trong trình thông dịch tương tác. Việc tải lại một mô-đun sau khi bạn thay đổi nó không phải là chuyện nhỏ. Ví dụ: hãy xem mô-đun sau: Là một phần của thử nghiệm và gỡ lỗi mô-đun này, bạn nhập nó vào bảng điều khiển Python: >>> 1Điều này tạo ra một biểu đồ như sau: Lưu ý rằng tải dữ liệu vào thời điểm nhập khẩu là một loại antipattern . Tốt nhất, bạn muốn hàng nhập khẩu của mình không có tác dụng phụ càng tốt. Một cách tiếp cận tốt hơn là tải dữ liệu một cách lười biếng khi bạn cần. Bạn có thể làm điều này một cách khá thanh lịch bằng cách sử dụng thuộc tính Mở rộng phần sau để xem ví dụ. >>> 1Điều này tạo ra một biểu đồ như sau: Lưu ý rằng tải dữ liệu vào thời điểm nhập khẩu là một loại antipattern . Tốt nhất, bạn muốn hàng nhập khẩu của mình không có tác dụng phụ càng tốt. Một cách tiếp cận tốt hơn là tải dữ liệu một cách lười biếng khi bạn cần. Bạn có thể làm điều này một cách khá thanh lịch bằng cách sử dụng thuộc tính Mở rộng phần sau để xem ví dụ. >>> 3Điều này tạo ra một biểu đồ như sau: >>> 4Điều này tạo ra một biểu đồ như sau: Lưu ý rằng tải dữ liệu vào thời điểm nhập khẩu là một loại antipattern . Tốt nhất, bạn muốn hàng nhập khẩu của mình không có tác dụng phụ càng tốt. Một cách tiếp cận tốt hơn là tải dữ liệu một cách lười biếng khi bạn cần. Bạn có thể làm điều này một cách khá thanh lịch bằng cách sử dụng thuộc tính Mở rộng phần sau để xem ví dụ.Đang tải lại các mô-đun Bộ nhớ cache của mô-đun có thể hơi khó chịu khi bạn đang làm việc trong trình thông dịch tương tác. Việc tải lại một mô-đun sau khi bạn thay đổi nó không phải là chuyện nhỏ. Ví dụ: hãy xem mô-đun sau: 5Là một phần của thử nghiệm và gỡ lỗi mô-đun này, bạn nhập nó vào bảng điều khiển Python: >>> 6Điều này tạo ra một biểu đồ như sau: >>> 7Bạn có thể thấy nó 39được nhập từ một tệp, trong khi đó 11là một số loại mô-đun tích hợp sẵn. Có vẻ như các mô-đun tích hợp sẵn không bị che khuất bởi các mô-đun cục bộ.Lưu ý: Các mô-đun tích hợp được biên dịch thành trình thông dịch Python. Thông thường, chúng module nền tảng như 15, 16, và 11. Mô-đun nào được tích hợp tùy thuộc vào trình thông dịch Python của bạn, nhưng bạn có thể tìm thấy tên của chúng trong đó 18.Hãy tìm hiểu sâu hơn nữa về hệ thống nhập của Python. Điều này cũng sẽ cho thấy lý do tại sao các mô-đun tích hợp sẵn không bị che bởi các mô-đun cục bộ. Có một số bước liên quan khi nhập một mô-đun:
Bạn có thể mở rộng hệ thống nhập Python bằng cách triển khai công cụ tìm của riêng bạn và nếu cần, trình tải của riêng bạn. Bạn sẽ thấy một ví dụ hữu ích hơn về công cụ tìm kiếm sau này. Hiện tại, bạn sẽ học cách thực hiện các tùy chỉnh cơ bản (và có thể ngớ ngẩn) của hệ thống nhập. 20 kiểm soát công cụ tìm kiếm nào được gọi trong quá trình nhập:>>> 8Trước tiên, hãy lưu ý rằng điều này trả lời câu hỏi trước đó: các mô-đun tích hợp sẵn không bị che khuất bởi các mô-đun cục bộ vì công cụ tìm tích hợp được gọi trước công cụ tìm đường dẫn nhập, công cụ này tìm thấy các mô-đun cục bộ. Thứ hai, lưu ý rằng bạn có thể tùy chỉnh 20theo ý thích của mình.Để nhanh chóng làm rối phiên Python của bạn, bạn có thể xóa tất cả các công cụ tìm kiếm: >>> 9Vì không có công cụ tìm kiếm, Python không thể tìm hoặc nhập các mô-đun mới. Tuy nhiên, Python vẫn có thể nhập các mô-đun đã có trong bộ nhớ cache của mô-đun vì nó trông ở đó trước khi gọi bất kỳ công cụ tìm kiếm nào. Trong ví dụ trên, 53đã được tải dưới mui xe trước khi bạn xóa danh sách công cụ tìm kiếm. Nếu bạn thực sự muốn làm cho phiên Python của mình hoàn toàn không sử dụng được, thì bạn cũng có thể xóa bộ nhớ cache của mô-đun , 98.Sau đây là một ví dụ hữu ích hơn một chút. Bạn sẽ viết một công cụ tìm để in một thông báo tới bảng điều khiển xác định mô-đun đang được nhập. Ví dụ cho thấy cách thêm công cụ tìm của riêng bạn, mặc dù nó không thực sự cố gắng tìm một mô-đun: 0Tất cả các trình tìm kiếm phải triển khai một 24phương thức lớp, phương thức này sẽ cố gắng tìm một mô-đun nhất định. Có ba cách 24có thể chấm dứt:
Bản 28in một thông báo đến bảng điều khiển và sau đó trả về một cách rõ ràng 26để chỉ ra rằng các công cụ tìm kiếm khác sẽ tìm ra cách thực sự nhập mô-đun.Lưu ý: Vì Python trả về ngầm định 26 từ bất kỳ hàm hoặc phương thức nào mà không có hàm rõ ràng 31, bạn có thể bỏ qua dòng 9 . Tuy nhiên, trong trường hợp này, tốt hơn là nên bao gồm 32để làm rõ rằng 28không tìm thấy mô-đun.Bằng cách chèn 28đầu tiên vào danh sách công cụ tìm kiếm, bạn sẽ nhận được danh sách đang chạy của tất cả các mô-đun đang được nhập:>>> 1Ví dụ: bạn có thể thấy rằng việc nhập 35sẽ kích hoạt việc nhập một số mô-đun khác 35phụ thuộc vào. Lưu ý rằng tùy chọn dài dòng cho trình thông dịch Python 37, cung cấp cùng một thông tin và nhiều hơn nữa.Ví dụ khác, giả sử rằng bạn đang tìm cách loại bỏ thế giới của các biểu thức chính quy . (Bây giờ, tại sao bạn lại muốn một thứ như vậy? Biểu thức chính quy thật tuyệt vời !) Bạn có thể triển khai công cụ tìm kiếm sau để cấm 38mô-đun biểu thức chính quy : 2Nâng cao một 73đảm bảo rằng không có công cụ tìm kiếm nào sau này trong danh sách các công cụ tìm kiếm sẽ được thực thi. Điều này có hiệu quả ngăn bạn sử dụng các biểu thức chính quy trong Python:>>> 3Mặc dù bạn chỉ đang nhập 35, nhưng mô-đun đó đang nhập 38ở hậu trường, do đó, lỗi sẽ xuất hiện.Ví dụ: Tự động cài đặt từ PyPIBởi vì hệ thống nhập Python đã khá mạnh mẽ và hữu ích, có nhiều cách để làm rối tung nó hơn là mở rộng nó một cách hữu ích. Tuy nhiên, ví dụ sau đây có thể hữu ích trong một số trường hợp nhất định. Các Python Package Index (PyPI) là một cửa của bạn cửa hàng phô mai cho việc tìm kiếm các module và các gói của bên thứ ba. Nó cũng là nơi 45tải xuống các gói.Trong các hướng dẫn Real Python khác , bạn có thể đã xem hướng dẫn sử dụng 43để cài đặt các gói và mô-đun của bên thứ ba mà bạn cần để làm theo cùng với các ví dụ. Sẽ không tuyệt vời nếu Python tự động cài đặt các mô-đun bị thiếu cho bạn phải không?Cảnh báo: Trong hầu hết các trường hợp, thực sự sẽ không tuyệt vời nếu Python tự động cài đặt các mô-đun. Ví dụ: trong hầu hết các cài đặt sản xuất, bạn muốn kiểm soát môi trường của mình. Hơn nữa, tài liệu cảnh báo không sử dụng 45theo cách này.Để tránh làm xáo trộn quá trình cài đặt Python của bạn, bạn chỉ nên sử dụng mã này trong những môi trường mà bạn không ngại xóa hoặc cài đặt lại. Công cụ tìm kiếm sau cố gắng cài đặt các mô-đun bằng cách sử dụng 45: 4So với các công cụ tìm bạn đã thấy trước đó, công cụ này phức tạp hơn một chút. Bằng cách đặt công cụ tìm kiếm này cuối cùng trong danh sách công cụ tìm kiếm, bạn biết rằng nếu bạn gọi 46, thì mô-đun sẽ không được tìm thấy trên hệ thống của bạn. Công việc của 24do đó chỉ là làm 48. Nếu quá trình cài đặt hoạt động, thì thông số mô-đun sẽ được tạo và trả về.Cố gắng sử dụng 49thư viện mà không cần tự cài đặt nó:>>> 5Thông thường, 50sẽ tăng a 73, nhưng trong trường hợp 49này là cài đặt và nhập.Mặc dù 46có vẻ hiệu quả, nhưng có một số thách thức với cách tiếp cận này. Một vấn đề chính là tên nhập của một mô-đun không phải lúc nào cũng tương ứng với tên của nó trên PyPI. Ví dụ: trình đọc nguồn cấp dữ liệu Python thực được gọi 54trên PyPI, nhưng tên nhập chỉ đơn giản là 55.Sử dụng 46để nhập và cài đặt 55kết thúc bằng cách cài đặt gói sai:>>> 6Điều này có thể gây ra hậu quả tai hại cho dự án của bạn. Một tình huống trong đó cài đặt tự động có thể khá hữu ích là khi bạn đang chạy Python trên đám mây với quyền kiểm soát hạn chế hơn đối với môi trường của bạn, chẳng hạn như khi bạn đang chạy sổ ghi chép kiểu Jupyter tại Google Colaboratory . Môi trường máy tính xách tay Colab rất tuyệt vời để thực hiện khám phá dữ liệu hợp tác. Một máy tính xách tay điển hình đi kèm với nhiều gói khoa học dữ liệu được cài đặt, bao gồm NumPy , Pandas và Matplotlib và bạn có thể thêm các gói mới với 45. Nhưng bạn cũng có thể kích hoạt cài đặt tự động:Vì 59không khả dụng cục bộ trên máy chủ Colab, mã được sao chép vào ô đầu tiên của sổ ghi chép.Ví dụ: Nhập tệp dữ liệuVí dụ cuối cùng trong phần này được lấy cảm hứng từ bài đăng blog tuyệt vời của Aleksey Bilogur Nhập hầu hết mọi thứ bằng Python: Giới thiệu về Trình tải và Trình tìm mô-đun . Bạn đã thấy cách sử dụng 07để nhập tệp dữ liệu. Tại đây, thay vào đó, bạn sẽ triển khai trình tải tùy chỉnh có thể nhập trực tiếp tệp CSV.Trước đó , bạn đã làm việc với một tệp CSV khổng lồ với dữ liệu dân số. Để làm cho ví dụ về trình tải tùy chỉnh dễ quản lý hơn, hãy xem xét 61tệp nhỏ hơn sau : 7Dòng đầu tiên là tiêu đề đặt tên cho ba trường và hai hàng dữ liệu sau, mỗi hàng chứa thông tin về một nhân viên. Để biết thêm thông tin về cách làm việc với tệp CSV, hãy xem Đọc và Viết tệp CSV bằng Python . Mục tiêu của bạn trong phần này là viết một công cụ tìm và một trình tải cho phép bạn nhập trực tiếp tệp CSV để bạn có thể viết mã như sau: >>> 8Công việc của công cụ tìm kiếm sẽ là tìm kiếm và nhận dạng các tệp CSV. Công việc của trình tải sẽ là nhập dữ liệu CSV. Thông thường, bạn có thể triển khai các trình tìm kiếm và trình tải tương ứng trong một lớp chung. Đó là cách tiếp cận bạn sẽ thực hiện ở đây: 9Có khá nhiều mã trong ví dụ này! May mắn thay, hầu hết công việc được thực hiện trong 24và 63. Hãy xem xét chúng chi tiết hơn.Như bạn đã thấy trước đó, 24chịu trách nhiệm tìm kiếm mô-đun. Trong trường hợp này, bạn đang tìm kiếm tệp CSV, vì vậy bạn tạo tên tệp có 65hậu tố. 49chứa tên đầy đủ của mô-đun được nhập. Ví dụ, nếu bạn sử dụng 67, thì 49sẽ được 69. Trong trường hợp này, tên tệp sẽ là 61.Đối với nhập khẩu cấp cao nhất, 71sẽ được 26. Trong trường hợp đó, bạn tìm kiếm tệp CSV trong đường dẫn nhập đầy đủ, đường dẫn này sẽ bao gồm thư mục làm việc hiện tại. Nếu bạn đang nhập tệp CSV trong một gói, thì tệp đó 71sẽ được đặt thành đường dẫn hoặc các đường dẫn của gói. Nếu bạn tìm thấy tệp CSV phù hợp, thì thông số mô-đun sẽ được trả về. Thông số mô-đun này yêu cầu Python tải mô-đun bằng cách sử dụng 74.Dữ liệu CSV được tải bởi 63. Bạn có thể sử dụng 76từ thư viện chuẩn để phân tích cú pháp tệp thực tế. Giống như hầu hết mọi thứ trong Python, các mô-đun được hỗ trợ bởi từ điển. Bằng cách thêm dữ liệu CSV vào 77, bạn làm cho nó có sẵn dưới dạng thuộc tính của mô-đun.Ví dụ: thêm 78vào từ điển mô-đun trên dòng 44 cho phép bạn liệt kê tên trường trong tệp CSV như sau:>>> 0Nói chung, tên trường CSV có thể chứa khoảng trắng và các ký tự khác không được phép trong tên thuộc tính Python. Trước khi thêm các trường làm thuộc tính trên mô-đun, bạn làm sạch tên trường bằng biểu thức chính quy . Điều này được thực hiện khi 79bắt đầu từ dòng 51 .Bạn có thể xem ví dụ về hiệu ứng này trong 80tên trường ở trên. Nếu bạn nhìn vào tệp CSV ban đầu, thì bạn sẽ thấy rằng tiêu đề 81có dấu cách thay vì dấu gạch dưới.Bằng cách kết nối điều này 74vào hệ thống nhập Python, bạn sẽ có được một chút chức năng miễn phí. Ví dụ: bộ đệm ẩn mô-đun sẽ đảm bảo rằng tệp dữ liệu chỉ được tải một lần.Mẹo và thủ thuật nhập khẩuĐể hoàn thiện hướng dẫn này, bạn sẽ thấy một vài mẹo về cách xử lý các tình huống nhất định thỉnh thoảng xuất hiện. Bạn sẽ thấy cách xử lý các gói bị thiếu, nhập theo chu kỳ và thậm chí các gói được lưu trữ bên trong tệp ZIP. Xử lý các gói trên các phiên bản PythonĐôi khi bạn cần xử lý các gói có tên khác nhau tùy thuộc vào phiên bản Python. Bạn đã thấy một ví dụ về điều này: 07chỉ có sẵn kể từ Python 3.7. Trong các phiên bản Python trước đó, bạn cần cài đặt và sử dụng 10thay thế.Miễn là các phiên bản khác nhau của gói tương thích, bạn có thể xử lý điều này bằng cách đổi tên gói bằng 85: 3Trong phần còn lại của mã, bạn có thể tham khảo 86và không phải lo lắng về việc bạn đang sử dụng 07hoặc 10.Thông thường, dễ nhất là sử dụng một 43câu lệnh để tìm ra phiên bản nào sẽ sử dụng. Một tùy chọn khác là kiểm tra phiên bản của trình thông dịch Python. Tuy nhiên, điều này có thể thêm một số chi phí bảo trì nếu bạn cần cập nhật số phiên bản.Bạn có thể viết lại ví dụ trước như sau: 2Điều này sẽ sử dụng 07trên Python 3.7 và mới hơn trong khi quay trở lại 10các phiên bản cũ hơn của Python. Xem 92dự án để có lời khuyên tốt và có thể kiểm chứng trong tương lai về cách kiểm tra phiên bản Python nào đang chạy.Xử lý các gói bị thiếu: Sử dụng một gói thay thếTrường hợp sử dụng sau có liên quan chặt chẽ với ví dụ trước. Giả sử có một gói tái thực hiện tương thích. Việc thực hiện lại được tối ưu hóa tốt hơn, vì vậy bạn muốn sử dụng nó nếu nó có sẵn. Tuy nhiên, gói gốc dễ có sẵn hơn và cũng mang lại hiệu suất chấp nhận được. Một ví dụ như vậy là 93, đó là phiên bản được tối ưu hóa 94từ thư viện chuẩn. Bạn có thể xử lý các tùy chọn này giống như cách bạn xử lý các tên gói khác nhau trước đó: 3Điều này sẽ sử dụng 93nếu nó có sẵn và rơi trở lại 94nếu không.Một ví dụ tương tự khác là gói UltraJSON , một bộ mã hóa và giải mã JSON cực nhanh có thể được sử dụng để thay thế 86trong thư viện tiêu chuẩn: 4Bằng cách đổi tên 98thành 86, bạn không phải lo lắng về gói nào đã thực sự được nhập.Xử lý các gói bị thiếu: Sử dụng Mock để thay thếVí dụ thứ ba, có liên quan là thêm một gói cung cấp một tính năng tuyệt vời không cần thiết cho ứng dụng của bạn. Một lần nữa, điều này có thể được giải quyết bằng cách thêm 43vào các mục nhập của bạn. Thách thức bổ sung là bạn sẽ thay thế gói tùy chọn như thế nào nếu nó không có sẵn.Ví dụ cụ thể, giả sử rằng bạn đang sử dụng Colorama để thêm văn bản màu trong bảng điều khiển. Colorama chủ yếu bao gồm các hằng số chuỗi đặc biệt thêm màu khi in: >>> 5Thật không may, màu không hiển thị trong ví dụ trên. Trong thiết bị đầu cuối của bạn, nó sẽ trông giống như sau: Trước khi bắt đầu sử dụng màu Colorama, bạn nên gọi điện 01. Đặt 02thành 03có nghĩa là các chỉ thị màu sẽ được tự động đặt lại ở cuối chuỗi. Đó là một cài đặt hữu ích nếu bạn chỉ muốn tô màu một dòng tại một thời điểm.Nếu bạn muốn tất cả đầu ra của mình là (ví dụ) màu xanh lam, thì bạn có thể để 02là 05và thêm 06vào đầu tập lệnh của mình. Các màu sau có sẵn:>>> 6Thật không may, màu không hiển thị trong ví dụ trên. Trong thiết bị đầu cuối của bạn, nó sẽ trông giống như sau: Trước khi bắt đầu sử dụng màu Colorama, bạn nên gọi điện 01. Đặt 02thành 03có nghĩa là các chỉ thị màu sẽ được tự động đặt lại ở cuối chuỗi. Đó là một cài đặt hữu ích nếu bạn chỉ muốn tô màu một dòng tại một thời điểm. 7Nếu bạn muốn tất cả đầu ra của mình là (ví dụ) màu xanh lam, thì bạn có thể để 02là 05và thêm 06vào đầu tập lệnh của mình. Các màu sau có sẵn:Bạn cũng có thể sử dụng 07để kiểm soát kiểu văn bản của mình. Bạn có thể lựa chọn giữa 08, 09và 10.Cuối cùng, 11cung cấp mã để kiểm soát vị trí của con trỏ. Bạn có thể sử dụng nó để hiển thị tiến trình hoặc trạng thái của một tập lệnh đang chạy. Ví dụ sau hiển thị đếm ngược từ 12:>>> 8Thật không may, màu không hiển thị trong ví dụ trên. Trong thiết bị đầu cuối của bạn, nó sẽ trông giống như sau: Trước khi bắt đầu sử dụng màu Colorama, bạn nên gọi điện 01. Đặt 02thành 03có nghĩa là các chỉ thị màu sẽ được tự động đặt lại ở cuối chuỗi. Đó là một cài đặt hữu ích nếu bạn chỉ muốn tô màu một dòng tại một thời điểm. 9Nếu bạn muốn tất cả đầu ra của mình là (ví dụ) màu xanh lam, thì bạn có thể để 02là 05và thêm 06vào đầu tập lệnh của mình. Các màu sau có sẵn:Bạn cũng có thể sử dụng 07để kiểm soát kiểu văn bản của mình. Bạn có thể lựa chọn giữa 08, 09và 10.Cuối cùng, 11cung cấp mã để kiểm soát vị trí của con trỏ. Bạn có thể sử dụng nó để hiển thị tiến trình hoặc trạng thái của một tập lệnh đang chạy. Ví dụ sau hiển thị đếm ngược từ 12: 00Lưu ý cách bộ đếm giữ nguyên vị trí thay vì in trên các dòng riêng biệt như bình thường: Hãy quay trở lại nhiệm vụ trong tầm tay. Đối với nhiều ứng dụng, việc thêm màu vào đầu ra bảng điều khiển của bạn là một điều tuyệt vời nhưng không quá quan trọng. Để tránh thêm phụ thuộc khác vào ứng dụng của mình, bạn chỉ muốn sử dụng Colorama nếu nó có sẵn trên hệ thống và không phá vỡ ứng dụng nếu không có. Để làm điều này, bạn có thể lấy cảm hứng từ thử nghiệm và việc sử dụng mô phỏng . Một mô hình có thể thay thế cho một đối tượng khác trong khi vẫn cho phép bạn kiểm soát hành vi của nó. Đây là một nỗ lực ngây thơ nhằm chế nhạo Colorama:Điều này không hoàn toàn hiệu quả, vì 13được biểu diễn bằng một chuỗi làm rối đầu ra của bạn. Thay vào đó, bạn muốn tạo một đối tượng luôn hiển thị dưới dạng chuỗi trống.Có thể thay đổi giá trị trả lại 67trên 15các đối tượng. Tuy nhiên, trong trường hợp này, sẽ thuận tiện hơn khi viết mô hình của riêng bạn: 16là một chuỗi trống cũng sẽ trả về chuỗi trống khi nó được gọi. Điều này giúp chúng ta tái hiện lại Colorama một cách hiệu quả, chỉ cần không có màu sắc.Bí quyết cuối cùng là 17lợi nhuận riêng của mình, do đó tất cả các phong trào màu sắc, phong cách, và con trỏ đó là các thuộc tính trên 18, 19, 20, và 21đang chế giễu là tốt. 01Các 22mô-đun được thiết kế để trở thành một thả thay thế cho Colorama, vì vậy bạn có thể cập nhật ví dụ đếm ngược sử dụng tìm kiếm và thay thế:Nếu bạn chạy tập lệnh này trên một hệ thống không có sẵn Colorama, thì nó vẫn hoạt động, nhưng có thể trông không đẹp: 02Giờ đây, cấu trúc của tệp JSON trở nên ít phức tạp hơn nhiều để nắm bắt. Bạn có thể sử dụng 29tùy chọn để sắp xếp các khóa theo thứ tự bảng chữ cái.Mặc dù thực hành tốt để phân chia tập lệnh và thư viện, Python có một thành ngữ giúp bạn có thể coi một mô-đun vừa là tập lệnh vừa là thư viện cùng một lúc. Như đã lưu ý trước đó , giá trị của 15biến mô-đun đặc biệt được đặt trong thời gian chạy dựa trên việc mô-đun được nhập hay chạy dưới dạng tập lệnh.Hãy thử nghiệm nó ra! Tạo tệp sau: 03Nếu bạn chạy tệp này, thì bạn sẽ thấy tệp đó 15được đặt thành giá trị đặc biệt 16: 04 Tuy nhiên, nếu bạn nhập mô-đun, thì 15được đặt thành tên của mô-đun:Hành vi này được tận dụng theo mô hình sau: 05Hãy sử dụng điều này trong một ví dụ lớn hơn. Trong một nỗ lực để giữ cho bạn trẻ , tập lệnh sau sẽ thay thế bất kỳ độ tuổi “già” nào ( 34hoặc cao hơn) bằng 02: 06Bạn có thể chạy tập lệnh này dưới dạng tập lệnh và nó sẽ tương tác làm cho độ tuổi bạn nhập trẻ hơn: 07Bạn cũng có thể sử dụng mô-đun như một thư viện có thể nhập. Bài 14kiểm tra trên dòng 12 đảm bảo rằng không có tác dụng phụ khi bạn nhập thư viện. Chỉ các chức năng 37và 38được xác định. Ví dụ, bạn có thể sử dụng thư viện này như sau:>>> 08Nếu không có sự bảo vệ của 14bài kiểm tra, việc nhập sẽ kích hoạt tương tác 40và 41rất khó sử dụng làm thư viện.Chạy tập lệnh Python từ tệp ZIPMột tính năng hơi khó hiểu của Python là nó có thể chạy các tập lệnh được đóng gói thành các tệp ZIP . Ưu điểm chính của việc này là bạn có thể phân phối một gói đầy đủ dưới dạng một tệp duy nhất. Tuy nhiên, lưu ý rằng điều này vẫn yêu cầu Python phải được cài đặt trên hệ thống. Nếu bạn muốn phân phối ứng dụng Python của mình dưới dạng tệp thực thi độc lập, hãy xem Sử dụng PyInstaller để dễ dàng phân phối ứng dụng Python . Nếu bạn cung cấp cho trình thông dịch Python một tệp ZIP , thì nó sẽ tìm kiếm một tệp có tên 38bên trong kho lưu trữ ZIP, giải nén nó và chạy nó. Như một ví dụ cơ bản, hãy tạo 38tệp sau : 09Điều này sẽ in một thông báo khi bạn chạy nó: 10Bây giờ hãy thêm nó vào kho lưu trữ ZIP. Bạn có thể thực hiện việc này trên dòng lệnh: 11Trên Windows, thay vào đó, bạn có thể sử dụng trỏ và nhấp . Chọn tệp trong File Explorer, sau đó nhấp chuột phải và chọn Gửi đến → Thư mục nén (nén) . Vì 16không phải là một cái tên mang tính mô tả nên bạn đã đặt tên cho tệp ZIP 45. Bây giờ bạn có thể gọi nó trực tiếp bằng Python: 12Lưu ý rằng tập lệnh của bạn biết rằng nó nằm bên trong 45. Hơn nữa, gốc của tệp ZIP của bạn được thêm vào đường dẫn nhập của Python để các tập lệnh của bạn có thể nhập các mô-đun khác bên trong cùng một tệp ZIP.Hãy nhớ lại ví dụ trước đó, trong đó bạn đã tạo một bài kiểm tra dựa trên dữ liệu dân số . Có thể phân phối toàn bộ ứng dụng này dưới dạng một tệp ZIP duy nhất. 07sẽ đảm bảo tệp dữ liệu được trích xuất từ kho lưu trữ ZIP khi cần thiết.Ứng dụng bao gồm các tệp sau: 13Bạn có thể thêm chúng vào tệp ZIP theo cách tương tự như bạn đã làm ở trên. Tuy nhiên, Python đi kèm với một công cụ được gọi là 48hợp lý hóa quá trình đóng gói các ứng dụng vào các kho lưu trữ ZIP. Bạn sử dụng nó như sau: 14Về cơ bản, lệnh này thực hiện hai việc: nó tạo một điểm vào và đóng gói ứng dụng của bạn. Hãy nhớ rằng bạn cần một 38tệp làm điểm nhập bên trong kho lưu trữ ZIP của mình. Nếu bạn cung cấp 41tùy chọn với thông tin về cách khởi động ứng dụng của bạn, thì hãy 48tạo tệp này cho bạn. Trong ví dụ này, tệp được tạo 38trông giống như sau: 15Điều này 38được đóng gói, cùng với nội dung của 54thư mục, vào một kho lưu trữ ZIP có tên 55. Các 56tín hiệu hậu tố rằng đây là một tập tin Python quấn vào một kho lưu trữ ZIP.Lưu ý: Theo mặc định, 48không nén bất kỳ tệp nào. Nó chỉ gói chúng thành một tệp duy nhất. Bạn cũng có thể yêu 48cầu nén các tệp bằng cách thêm 59tùy chọn.Tuy nhiên, tính năng này chỉ có sẵn trong Python 3.7 trở lên. Xem 48 tài liệu để biết thêm thông tin.Trên Windows, 56các tệp đã được đăng ký dưới dạng tệp Python. Trên Mac và Linux, bạn có thể 48tạo các tệp thực thi bằng cách sử dụng 63tùy chọn trình thông dịch và chỉ định trình thông dịch nào sẽ sử dụng: 16Các 63tùy chọn bổ sung thêm một công việc ( 65) mà nói với các hệ điều hành như thế nào để chạy các tập tin . Ngoài ra, nó làm cho 56tệp thực thi được để bạn có thể chạy tệp chỉ bằng cách nhập tên của nó: 17Lưu ý 67phía trước tên tệp. Đây là một thủ thuật điển hình trên Mac và Linux để chạy các tệp thực thi trong thư mục hiện tại. Nếu bạn di chuyển các tập tin vào một thư mục trên của bạn 68, hoặc nếu bạn đang sử dụng Windows, sau đó bạn sẽ có thể chỉ sử dụng tên tập tin: 55.Lưu ý: Trên Python 3.6 trở lên, lệnh trước đó sẽ không thành công với thông báo rằng nó không thể tìm thấy tài nguyên dữ liệu dân số trong 26thư mục. Điều này là do một giới hạn trong 71 .Một cách giải quyết là cung cấp đường dẫn tuyệt đối đến 55. Trên Mac và Linux, bạn có thể thực hiện việc này bằng thủ thuật sau: 18Các 73lệnh mở rộng đến đường dẫn của thư mục hiện hành.Hãy kết thúc phần này bằng cách xem một hiệu ứng đẹp của việc sử dụng 07. Hãy nhớ rằng bạn đã sử dụng mã sau để mở tệp dữ liệu: 19Một cách phổ biến hơn để mở tệp dữ liệu là định vị chúng dựa trên 03thuộc tính của mô-đun của bạn : 20Cách tiếp cận này thường hoạt động tốt. Tuy nhiên, nó sẽ sụp đổ khi ứng dụng của bạn được đóng gói thành một tệp ZIP: 21Tệp dữ liệu của bạn nằm trong kho lưu trữ ZIP, vì vậy bạn 19không thể mở tệp đó. 07mặt khác, sẽ trích xuất dữ liệu của bạn vào một tệp tạm thời trước khi mở nó.Xử lý nhập khẩu theo chu kỳNhập theo chu kỳ xảy ra khi bạn có hai hoặc nhiều mô-đun nhập lẫn nhau. Cụ thể hơn, hãy tưởng tượng rằng mô-đun 78sử dụng 79và mô-đun 80nhập tương tự 78.Hệ thống nhập của Python ở một mức độ nào đó được thiết kế để xử lý các chu trình nhập. Ví dụ, đoạn mã sau — mặc dù không hữu ích lắm — chạy tốt: 22Cố gắng nhập cũng như 78nhập thông dịch viên tương tác 80:>>> 23Lưu ý rằng 80được nhập vào giữa quá trình nhập 78, chính xác tại 79câu lệnh trong mã nguồn của 78. Lý do điều này không kết thúc trong đệ quy vô tận là người bạn cũ của chúng ta là bộ nhớ cache của mô-đun.Khi bạn nhập 88, tham chiếu tới 78sẽ được thêm vào bộ đệm ẩn của mô-đun ngay cả trước khi 78được tải. Khi 80cố gắng nhập 78sau, nó chỉ cần sử dụng tham chiếu trong bộ đệm ẩn mô-đun.Bạn cũng có thể có các mô-đun làm điều gì đó hữu ích hơn một chút. Nếu bạn xác định các thuộc tính và chức năng trong các mô-đun của mình, thì tất cả vẫn hoạt động: 24Việc nhập 78hoạt động giống như trước đây:>>> 23Lưu ý rằng 80được nhập vào giữa quá trình nhập 78, chính xác tại 79câu lệnh trong mã nguồn của 78. Lý do điều này không kết thúc trong đệ quy vô tận là người bạn cũ của chúng ta là bộ nhớ cache của mô-đun. 26Khi bạn nhập 88, tham chiếu tới 78sẽ được thêm vào bộ đệm ẩn của mô-đun ngay cả trước khi 78được tải. Khi 80cố gắng nhập 78sau, nó chỉ cần sử dụng tham chiếu trong bộ đệm ẩn mô-đun.>>> 27Lưu ý rằng 80được nhập vào giữa quá trình nhập 78, chính xác tại 79câu lệnh trong mã nguồn của 78. Lý do điều này không kết thúc trong đệ quy vô tận là người bạn cũ của chúng ta là bộ nhớ cache của mô-đun.Khi bạn nhập 88, tham chiếu tới 78sẽ được thêm vào bộ đệm ẩn của mô-đun ngay cả trước khi 78được tải. Khi 80cố gắng nhập 78sau, nó chỉ cần sử dụng tham chiếu trong bộ đệm ẩn mô-đun.Bạn cũng có thể có các mô-đun làm điều gì đó hữu ích hơn một chút. Nếu bạn xác định các thuộc tính và chức năng trong các mô-đun của mình, thì tất cả vẫn hoạt động: >>> 28Lưu ý rằng 80được nhập vào giữa quá trình nhập 78, chính xác tại 79câu lệnh trong mã nguồn của 78. Lý do điều này không kết thúc trong đệ quy vô tận là người bạn cũ của chúng ta là bộ nhớ cache của mô-đun.>>> 29Lưu ý rằng 80được nhập vào giữa quá trình nhập 78, chính xác tại 79câu lệnh trong mã nguồn của 78. Lý do điều này không kết thúc trong đệ quy vô tận là người bạn cũ của chúng ta là bộ nhớ cache của mô-đun.Thông thường, thời điểm dễ dàng nhất để sửa lỗi nhập theo chu kỳ là trước khi bạn triển khai chúng. Nếu bạn thấy các chu kỳ trong bản phác thảo kiến trúc của mình, hãy xem kỹ hơn và cố gắng phá vỡ các chu kỳ . Tuy nhiên, có những thời điểm hợp lý để giới thiệu một chu kỳ nhập khẩu. Như bạn đã thấy ở trên, đây không phải là vấn đề miễn là các mô-đun của bạn chỉ xác định các thuộc tính, hàm, lớp, v.v. Mẹo thứ hai — cũng là một phương pháp thiết kế tốt — là giữ cho các mô-đun của bạn không có tác dụng phụ tại thời điểm nhập . Nếu bạn thực sự cần các mô-đun có chu kỳ nhập và các tác dụng phụ, vẫn còn một cách khác: thực hiện nhập cục bộ bên trong các chức năng của bạn . Lưu ý rằng trong đoạn mã sau, 79được thực hiện bên trong 01. Điều này có hai hệ quả. Đầu tiên, 80chỉ có sẵn bên trong 01chức năng. Quan trọng hơn, quá trình nhập sẽ không xảy ra cho đến khi bạn gọi 01sau khi 78đã được nhập đầy đủ: 30Bây giờ không có vấn đề gì khi nhập và sử dụng 78:>>> 31Lưu ý rằng 80, trên thực tế, không được nhập cho đến khi bạn gọi 01. Để có góc nhìn khác về nhập khẩu theo chu kỳ, hãy xem ghi chú kinh điển của Fredrik Lundh .Nhập hồ sơMột mối quan tâm khi nhập một số mô-đun và gói là nó sẽ thêm vào thời gian khởi động tập lệnh của bạn. Tùy thuộc vào ứng dụng của bạn, điều này có thể quan trọng hoặc không. Kể từ khi phát hành Python 3.7 , bạn đã có một cách nhanh chóng để biết cần bao nhiêu thời gian để nhập các gói và mô-đun. Python 3.7 hỗ trợ 17tùy chọn dòng lệnh, đo lường và in lượng thời gian mà mỗi mô-đun cần để nhập: 32Các 18cột cho thấy thời gian tích lũy của nhập khẩu (trong micro) trên một cơ sở cho mỗi gói. Bạn có thể đọc danh sách như sau: Python dành 19micro để hoàn toàn nhập khẩu 20, trong đó liên quan đến nhập khẩu 11, 39và việc thực hiện C 23là tốt.Các 24chương trình cột thời gian nó đã nhập khẩu chỉ module nhất định, bao gồm bất kỳ nhập khẩu đệ quy. Bạn có thể thấy rằng 11mất 26micro giây để nhập, 39lấy 28, 23lấy 30và quá trình nhập 20chính nó mất 32micro giây. Nói chung, điều này cộng lại thời gian tích lũy là 19micro giây (trong phạm vi sai số làm tròn).Hãy xem 34ví dụ từ phần Colorama : 33Trong ví dụ này, quá trình nhập 22mất gần 0,013 giây. Phần lớn thời gian đó được dành cho việc nhập khẩu Colorama và các phụ thuộc của nó. Các 24cột cho thấy thời gian nhập khẩu trừ nhập khẩu lồng nhau.Đối với một ví dụ cực đoan, hãy xem xét 37singleton trước đó . Vì nó đang tải một tệp dữ liệu lớn nên quá trình nhập cực kỳ chậm. Để kiểm tra điều này, bạn có thể chạy 38dưới dạng tập lệnh với 59tùy chọn: 34Trong trường hợp này, mất gần 2 giây để nhập 37, trong đó khoảng 1,6 giây được dành cho chính mô-đun, chủ yếu để tải tệp dữ liệu. 17là một công cụ tuyệt vời để tối ưu hóa việc nhập của bạn. Nếu bạn cần thực hiện giám sát tổng quát hơn và tối ưu hóa mã của mình, thì hãy xem Chức năng hẹn giờ của Python: Ba cách để theo dõi mã của bạn .Phần kết luậnTrong hướng dẫn này, bạn đã biết về hệ thống nhập Python. Giống như nhiều thứ trong Python, nó khá đơn giản để sử dụng cho các tác vụ cơ bản như nhập các mô-đun và gói. Đồng thời, hệ thống nhập khẩu khá phức tạp, linh hoạt và có thể mở rộng. Bạn đã học được một số thủ thuật liên quan đến nhập mà bạn có thể tận dụng trong mã của riêng mình. Trong hướng dẫn này, bạn đã học cách:
Trong suốt hướng dẫn, bạn đã thấy nhiều liên kết để cung cấp thêm thông tin. Nguồn có thẩm quyền nhất trên hệ thống nhập Python là tài liệu chính thức:
Bạn có thể sử dụng kiến thức của mình về nhập Python bằng cách làm theo các ví dụ trong hướng dẫn này. Nhấp vào liên kết bên dưới để truy cập vào mã nguồn: |