Cộng đồng chia sẻ tri thức Lib24.vn

Nhập xuất trong Python - Hàm xuất

Gửi bởi: Phạm Thọ Thái Dương 20 tháng 11 2020 lúc 15:14:01


Mục lục
* * * * *

Vì sao cần hàm print

Nếu bạn hay dùng interactive prompt thì bạn nhân ra rằng, kết quả luôn xuất hiện sau mỗi dòng code của bạn. Tuy nhiên, nó sẽ không như vậy khi bạn viết những dòng code vào trong một file Python và chạy chương trình đó.

Bạn cần một hàm giúp bạn xuất các nội dung mà bạn muốn cụ thể ở đây là xuất ra Shell (terminal, command prompt, powershell,…). Đó là lí do hàm print ra đời!

Tìm hiểu cách sử dụng hàm print thông qua các parameter

Hàm print có cú pháp như sau

Cú pháp:

print(*objects, sep=' ', end='\n', file=sys.stdout, flush=False)

Chúng ta sẽ tìm hiểu parameter đầu tiên

*objects

* chính là packing argument. Ở đây hiểu nôm na sẽ là nó sẽ gom lại các argument của bạn lại thành một Tuple.

>>> packing = 1, 2, 3, 4 # giống như gọi hàm function(1, 2, 3, 4)
>>> packing
(1, 2, 3, 4)
1234

Khi bạn truyền các argument vào hàm (giá trị 1, giá trị 2, giá trị 3,…) thì nó sẽ gói lại thành một Tuple giống như trên.

>>> print('Kteam')
Kteam
>>> print('Kteam', 'Free Education')
Kteam Free Education
>>> print('Kteam', 'Free Education', 'one more argument')
Kteam Free Education one more argument
1234567

Nhờ như vậy, bạn có thể truyền argument vào hàm print với số lượng bất kì. Điều này giúp bạn không phải ép kiểu dữ liệu, để rồi nối chúng lại với nhau thành một giá trị rồi mới truyền cho hàm print.

>>> print('Kteam' + 69)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: must be str, not int
>>> print('Kteam' + str(69))
Kteam69
>>> print('Kteam', 69)
Kteam 69
>>> print(123, [1, 2, 3], 'Kteam')
123 [1, 2, 3] Kteam
1234567891011

Chắc bạn cũng nhận ra một chút khác biệt ở hai trường hợp bên dưới. 

>>> print('Kteam' + 'Python')
KteamPython
>>> print('Kteam', 'Python')
Kteam Python
12345

Để hiểu điều đó, chúng ta tới với parameter tiếp theo

sep (separate – chia ra, phân ra)

Giá trị mặc định của parameter này là một khoảng trắng. Khi các argument bạn ném vào cho hàm print để hàm print in ra nội dung, như đã biết là nó sẽ được gói vào một Tuple. Các giá trị trong Tuple sẽ được nối với nhau bằng parameter sep.

Lưu ý: Khi truyền giá trị vào cho parameter theo cách keyword argument thì sẽ không bị packing. Nghĩa là sẽ không bị gói vào trong giá trị của parameter object.

>>> print('Kteam', 'Python', 'Course') # sep mặc định là 1 khoảng trắng
Kteam Python Course
>>> print('Kteam', 'Python', 'Course', sep='---')
Kteam---Python---Course
>>> print('Kteam', 'Python', 'Course', sep='|||')
Kteam|||Python|||Course
>>> print('Kteam', 'Python', 'Course', sep='\n')
Kteam
Python
Course
>>> print('Kteam', 'Python', 'Course', sep='')
KteamPythonCourse
12345678910111213

Tiếp theo là một parameter khá rắc rối

end (kết thúc bằng)

Đầu tiên, hãy chạy một file Python với nội dung sau đây.

print('line 1')
print('line 2')
print('line 3')
1234

Kết quả bạn nhận được chắc chắn sẽ là

line 1
line 2
line 3
1234

Nếu bạn từng học qua ngôn ngữ lập trình C hoặc C++ hay là Java cũng có thể là C#. Bạn sẽ nhận thấy, mỗi lần print, chúng sẽ tự xuống dòng.

Đó là nhờ parameter end. Nó sẽ tự thêm một kí tự newline (\n) vào cuối để có thể đưa con trỏ xuống dòng mới thay vì bạn phải tự thêm \n như một số ngôn ngữ lập trình khác (một số ngôn ngữ lập trình có hỗ trợ thêm phương thức giúp xuất nội dung và tự động xuống dòng)

Và đương nhiên, chúng ta cũng có thể thay đổi giá trị của parameter này.

>>> print('a line without newline', end='')
a line without newline>>> print('a line without newline', end='|||')
a line without newline|||>>> print()

>>>
123456

Bạn cũng thấy nếu không có end bằng một kí tự newline thì interactive prompt lộn xộn thế nào.

Nhưng đó không phải vấn đề. Hãy cẩn thận khi sử dụng print mà không có newline.

Hãy tạo một file Python có nội dung như sau:

from time import sleep # nhập hàm sleep từ thư viện time

print('start....')
sleep(3) # dừng chương trình 3 giây
print('end...')
123456

Khi chạy chương trình, bạn sẽ thấy xuất hiện dòng ⋆t....⋆t.... sau đó 3 giây sau sẽ xuất hiện tới dòng end...end....

Kết quả này hoàn toàn bình thường và đúng như những gì dự đoán. Nhưng hãy thử thay đổi một tí:

from time import sleep # nhập hàm sleep từ thư viện time

print('start....', end='') # in ra nội dung và kết thúc bới một chuỗi  rỗng
sleep(3) # dừng chương trình 3 giây
print('end...')
123456

Lần này đã có khác biệt. Bạn sẽ không thấy gì xuất hiện ban đầu, mãi đến 3 giây sau bạn mới thấy dòng ⋆t....end...⋆t....end.... Kết quả thì đúng, nhưng cách kết quả được xuất ra thì không giống như bạn nghĩ.

Vì sao lại vậy? Đó là do mỗi lần hàm print nhận được các giá trị bạn muốn in. Các giá trị đó được gói trong một Tuple. Tiếp đến, hàm print nạp từng giá trị trong Tuple vào bộ nhớ đệm. 

Hoặc khi kết thúc chương trình, những gì còn trong bộ đệm cũng sẽ được xuất ra.

Một số ví dụ

Ví dụ 1: Hãy thử một vài ví dụ khác để hiểu thêm

from time import sleep # nhập hàm sleep từ thư viện time

print('line 1\n', 'line2', end='')
sleep(3) # dừng chương trình 3 giây
print('end...')
123456

Kết quả xuất hiện sẽ là l∈e1l∈e1 > đợi 3 giây > xuất hiện các nội dung còn lại. Vì chuỗi 'line 1\n' có kí tự newline nên chuỗi đó được xuất ra. Còn chuỗi 'line 2' thì không nên vẫn nằm trong bộ nhớ đệm.Ví dụ 2:

from time import sleep # nhập hàm sleep từ thư viện time

print('line 1', 'lin\ne2', end='')
sleep(3) # dừng chương trình 3 giây
print('end...')
123456

Kết quả sẽ là xuất in hai chuỗi l∈e1l∈e1 và l∈e2l∈e2 > đợi 3 giây > xuất nội dung còn lại.

Quy trình sẽ là nạp chuỗi line 1 vào bộ nhớ đêm, nạp tiếp chuỗi line 2 vào bộ nhớ đệm, thấy chuỗi line 2 có kí  tự newline, xuất những gì có trong bộ nhớ đệm ra. Sau đó đợi 3 giây và rồi xuất nội dung còn lại.

file

Mặc định hàm print sẽ ghi nội dung vào file sys.stdout. Cũng nhờ vậy, bạn mới thấy được nội dung trên shell. Đương nhiên, dựa vào đây, ta cũng có thể sử dụng hàm print như là phương thức write trong việc ghi file.

>>> with open('printtext.txt', 'w') as f:
...     print('printed by print function', file=f)
...
>>> with open('printtext.txt') as f:
...     f.read()
...
'printed by print function\n'
12345678

flush

Parameter cuối cùng - flush. Giá trị mặc định giá trị là False. Liên quan khá nhiều đến parameter end lúc nãy thế nên ta hãy quay lại ví dụ lúc nãy.

from time import sleep # nhập hàm sleep từ thư viện time

print('start...', end='')
sleep(3) # dừng chương trình 3 giây
print('end...')
123456

Sau 3 giây chương trình mới có kết quả. Bạn cũng đã biết vì sao rồi, đúng chứ?

Nào, hãy để cho parameter flush giá trị True

from time import sleep # nhập hàm sleep từ thư viện time

print('start...', end='', flush=True)
sleep(3) # dừng chương trình 3 giây
print('end...')
123456

Kết quả bây giờ vẫn vậy, nhưng quá trình xuất kết quả có chút khác biệt. Bạn ngay lập tức nhìn thấy nội dung dòng print đầu tiên. Đó là nhờ parameter flush. Nếu là True, nó sẽ yêu cầu bộ đệm xuất những gì có trong bộ đệm ra.

Print trong Python 3.X và Python 2.X có gì khác nhau?

Print trong Python 3.X là một hàm, như đã giới thiệu. Còn với Python 2.X nó là một câu lệnh.

# print trong Python 2.X
print 'Kteam'
print 'Kteam', 'Free Education'
# tương tự với trong Python 3.X sẽ là
print('Kteam')
print('Kteam', 'Free Education')
1234567

Một số bạn nhầm lần rằng Print Python 2.X cũng có thể sử dụng như Python 3.X

# print trong Python 2.X
print('Kteam')
# và nhận được kết quả giống như Python 3.X
print('Kteam')
12345

Nhưng bản chất là khác nhau

# print trong Python 2.X
print('Kteam')
# tương đương với Python 3.X là
print(('Kteam'))
12345

Đây là interactive prompt của Python 2.X. Ta sẽ thử một ví dụ để làm rõ điều này

>>> print('Kteam')
Kteam
>>> print('Kteam', 'Free Education')
('Kteam', 'Free Education')
12345

Bạn cũng thấy, cặp dấu () không phải là  một cặp dấu ngoặc như cách gọi hàm. Đó giống như việc bạn đặt một giá trị trong cặp dấu ngoặc đơn mà  thôi. Và vì nó có một giá trị nên không có sự khác biệt

Còn khi bạn đặt hai giá trị trở lên, Python hiểu đó là một Tuple.

Một đoạn code nhỏ dành cho bạn tự nhiên cứu:

from time import sleep

your_name = "Henry"
your_great = "Hello! My name is "

for c in your_great + your_name:
    print(c, end='', flush=True)
    sleep(0.1)
print()

Được cập nhật: 15 tháng 4 lúc 21:24:05 | Lượt xem: 1037

Các bài học liên quan