翻譯:《實用的Python編程》01_04_Strings

目錄 | 上一節 (1.3 數字) | 下一節 (1.5 列表)html

1.4 字符串

本節介紹處理文本的方法。python

表示字面量文本

在程序中字符串字面量使用引號來書寫。git

# 單引號(Single quote)
a = 'Yeah but no but yeah but...'

# 雙引號(Double quote)
b = "computer says no"

# 三引號(Triple quotes)
c = '''
Look into my eyes, look into my eyes, the eyes, the eyes, the eyes,
not around the eyes,
don't look around the eyes,
look into my eyes, you're under.
'''

一般,字符串只能佔一行。三引號捕獲在引號結束以前的全部文本,包含全部的格式。github

使用單引號(')和雙引號(「)沒有區別。可是,以什麼樣的引號開始字符串,必須以什麼樣的引號結束字符串。正則表達式

字符串轉義碼

轉義碼被用於表示控制字符和不能輕易在鍵盤上輸入的字符。如下是一些常見的轉義碼:數據庫

'\n'      換行符(Line feed)
'\r'      回車符(Carriage return)
'\t'      製表符(Tab)
'\''      字面量單引號(Literal single quote)
'\"'      字面量雙引號(Literal double quote)
'\\'      字面量反斜槓(Literal backslash)

字符串表示

字符串中的每一個字符在內部被存儲爲所謂的 Unicode 「代碼點(code-point)」,代碼點是一個整數。可使用下列轉移序列指定確切的代碼點 。數組

a = '\xf1'          # a = 'ñ'
b = '\u2200'        # b = '∀'
c = '\U0001D122'    # c = '𝄢'
d = '\N{FOR ALL}'   # d = '∀'

全部可用的字符碼請參考 Unicode 字符數據庫函數

字符串索引

能夠像訪問數組那樣訪問字符串的單個字符。你可使用從 0 開始的整數索引,負索引指定相對於字符串尾部的位置。測試

a = 'Hello world'
b = a[0]          # 'H'
c = a[4]          # 'o'
d = a[-1]         # 'd' (end of string)

你也能夠指定一個索引範圍來切割或者選擇子串:ui

d = a[:5]     # 'Hello'
e = a[6:]     # 'world'
f = a[3:8]    # 'lo wo'
g = a[-5:]    # 'world'

不包括結尾索引處的字符。缺失的索引假定爲字符串的開始或者結尾。

字符串操做

字符串的操做包括:拼接,長度計算,成員判斷和複製。

# Concatenation (+)
a = 'Hello' + 'World'   # 'HelloWorld'
b = 'Say ' + a          # 'Say HelloWorld'

# Length (len)
s = 'Hello'
len(s)                  # 5

# Membership test (`in`, `not in`)
t = 'e' in s            # True
f = 'x' in s            # False
g = 'hi' not in s       # True

# Replication (s * n)
rep = s * 5             # 'HelloHelloHelloHelloHello'

字符串的方法

字符串具備對數據執行各類操做的方法。

示例:刪除開頭或者結尾處的任何空白。

s = '  Hello '
t = s.strip()     # 'Hello'

示例:大小寫轉換。

s = 'Hello'
l = s.lower()     # 'hello'
u = s.upper()     # 'HELLO'

示例:文本替換。

s = 'Hello world'
t = s.replace('Hello' , 'Hallo')   # 'Hallo world'

更多字符串方法:

字符串具備各類各樣的方法用於測試和處理文本數據。

下面是字符串方法的一小部分示例:

s.endswith(suffix)     # Check if string ends with suffix
s.find(t)              # First occurrence of t in s
s.index(t)             # First occurrence of t in s
s.isalpha()            # Check if characters are alphabetic
s.isdigit()            # Check if characters are numeric
s.islower()            # Check if characters are lower-case
s.isupper()            # Check if characters are upper-case
s.join(slist)          # Join a list of strings using s as delimiter
s.lower()              # Convert to lower case
s.replace(old,new)     # Replace text
s.rfind(t)             # Search for t from end of string
s.rindex(t)            # Search for t from end of string
s.split([delim])       # Split string into list of substrings
s.startswith(prefix)   # Check if string starts with prefix
s.strip()              # Strip leading/trailing space
s.upper()              # Convert to upper case

字符串的可變性

字符串是「不可變的」或者說是隻讀的。一旦建立,字符串的值就沒法修改。

>>> s = 'Hello World'
>>> s[1] = 'a'
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'str' object does not support item assignment
>>>

全部處理字符串數據的操做和方法始終會建立一個新的字符串。

字符串轉換

使用 str() 函數能夠將任何值轉換爲字符串。 str() 函數獲得的結果是一個字符串,該字符串包含的文本與 print() 語句產生的文本相同。

>>> x = 42
>>> str(x)
'42'
>>>

字節字符串

一般,在底層 I/O 中會遇到 8 位字節的字符串(譯註:字節字符串),它是這樣寫的:

data = b'Hello World\r\n'

經過把小寫的 b 放到第一個引號以前來指定一個字節字符串而不是文本字符串(譯註:在字符串前面加 b 表示這是使用 ASCII 編碼的字節字符串)。大部分經常使用的文本字符串操做可應用於字節字符串。

len(data)                         # 13
data[0:5]                         # b'Hello'
data.replace(b'Hello', b'Cruel')  # b'Cruel World\r\n'

字節字符串索引有點不一樣,由於它返回的是整數形式的字節值:

data[0]   # 72 (ASCII code for 'H')

字節字符串與文本字符串之間的轉換:

text = data.decode('utf-8') # bytes -> text
data = text.encode('utf-8') # text -> bytes

'utf-8' 這個參數指定了字符的編碼方式。其它常見的編碼方式有 'ascii''latin1'

原始字符串

原始字符串是未解釋的帶有反斜槓的字符串字面量。經過在原始引號以前添加 「r」 前綴來指定。

>>> rs = r'c:\newdata\test' # Raw (uninterpreted backslash)
>>> rs
'c:\\newdata\\test'

輸出的字符串是包含在引號裏面的字面量文本,與輸入的文本徹底相同。這在反斜槓有特殊意義的狀況下頗有用。例如:文件名、正則表達式等。

f-Strings

具備格式化表達式替換的字符串。

>>> name = 'IBM'
>>> shares = 100
>>> price = 91.1
>>> a = f'{name:>10s} {shares:10d} {price:10.2f}'
>>> a
'       IBM        100      91.10'
>>> b = f'Cost = ${shares*price:0.2f}'
>>> b
'Cost = $9110.00'
>>>

注意: 這要求 Python 3.6 或者更新的版本. 格式化代碼的含義稍後介紹。

練習

在這些習題中,你將嘗試對 Python 字符串類型進行操做。你應該在 Python 交互提示符下操做,在該提示符下能夠輕鬆地查看到結果。重要提示:

在應該與解釋器進行交互的習題中,
>>> 當 Python 但願你輸入一個新的語句, 你將得到一個解釋器提示符。習題中某些語句會跨越多行——要使這些語句執行,你可能須要多按幾回回車鍵。提醒你,在作這些示例時,請勿輸入 >>> 提示符。

經過定義一個包含一系列股票代號的字符串開始吧。字符串以下所示:

>>> symbols = 'AAPL,IBM,MSFT,YHOO,SCO'
>>>

練習 1.13:提取單個字符和子串

字符串是字符數組。嘗試提取一些字符:

>>> symbols[0]
?
>>> symbols[1]
?
>>> symbols[2]
?
>>> symbols[-1]        # Last character
?
>>> symbols[-2]        # Negative indices are from end of string
?
>>>

在 Python 語言中,字符串是隻讀的。

嘗試經過將 symbols 字符串的第一個字符變爲小寫字母 ‘a’ 來驗證這一點。

>>> symbols[0] = 'a'
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: 'str' object does not support item assignment
>>>

練習 1.14:字符串拼接

儘管字符串數據是隻讀的,可是你始終能夠將變量從新分配給新建立的字符串。

嘗試下面的語句,該語句將一個新的股票代碼 「GOOG」 拼接到 symbols 字符串的末尾。

>>> symbols = symbols + 'GOOG'
>>> symbols
'AAPL,IBM,MSFT,YHOO,SCOGOOG'
>>>

糟糕!這不是咱們想要的。修改它使得變量 symbols 保存的值爲 'AAPL,IBM,MSFT,YHOO,SCO,GOOG'

>>> symbols = ?
>>> symbols
'AAPL,IBM,MSFT,YHOO,SCO,GOOG'
>>>

'HPQ' 添加到 symbols 字符串的前面:

>>> symbols = ?
>>> symbols
'HPQ,AAPL,IBM,MSFT,YHOO,SCO,GOOG'
>>>

在這些示例中,表面上看起來原始字符串像正在被修改,明顯違反了字符串是隻讀的。實際上不是這樣的。每次,這些操做都會建立一個全新的字符串。當變量名 symbols 被從新分配,它會指向一個新建立的字符串。而後,舊的字符串被銷燬,由於它再也不被使用了。

練習 1.15:成員測試(子串測試)

嘗試使用 in 操做符檢查子串。請在交互提示符下嘗試這些操做。

>>> 'IBM' in symbols
?
>>> 'AA' in symbols
True
>>> 'CAT' in symbols
?
>>>

爲何檢查 AA 的時候返回 True ?

練習 1.16:字符串方法

在 Python 交互提示符下,嘗試一些新的字符串方法。

>>> symbols.lower()
?
>>> symbols
?
>>>

請記住,字符串始終是隻讀的。若是你想要保存操做的結果,你須要把它放置到一個變量中。

>>> lowersyms = symbols.lower()
>>>

請嘗試更多的操做:

>>> symbols.find('MSFT')
?
>>> symbols[13:17]
?
>>> symbols = symbols.replace('SCO','DOA')
>>> symbols
?
>>> name = '   IBM   \n'
>>> name = name.strip()    # Remove surrounding whitespace
>>> name
?
>>>

練習 1.17:f-strings

有時你想建立一個字符串並把其它變量的值嵌入到其中。

要作到這點,可使用 f-strings。示例:

>>> name = 'IBM'
>>> shares = 100
>>> price = 91.1
>>> f'{shares} shares of {name} at ${price:0.2f}'
'100 shares of IBM at $91.10'
>>>

練習 1.10 中修改 mortgage.py 程序來使用 f-strings 建立它的輸出。

嘗試實現它,使得輸出可以很好地對齊。

練習 1.18:正則表達式

基本字符串操做的一個侷限性在於它們不支持任何類型的高級模式匹配。爲此,你須要使用 Python 的 re 模塊和正則表達式。正則表達式處理是一個大的主題,這裏只是一個簡短的示例:

>>> text = 'Today is 3/27/2018. Tomorrow is 3/28/2018.'
>>> # Find all occurrences of a date
>>> import re
>>> re.findall(r'\d+/\d+/\d+', text)
['3/27/2018', '3/28/2018']
>>> # Replace all occurrences of a date with replacement text
>>> re.sub(r'(\d+)/(\d+)/(\d+)', r'\3-\1-\2', text)
'Today is 2018-3-27. Tomorrow is 2018-3-28.'
>>>

有關 re 模塊的更多信息,請查看官方文檔:https://docs.python.org/library/re.html

說明

當你開始嘗試使用解釋器時,你老是但願瞭解更多有關不一樣對象所支持的操做。例如,如何找出哪些操做是對字符串是有效的?

根據你的 Python 環境,你可能能夠經過 tab 鍵補全來查看可用方法的列表。例如,嘗試輸入下面的代碼:

>>> s = 'hello world'
>>> s.<tab key>
>>>

若是單擊 tab 鍵沒有任何做用,你可使用 Python 的內建函數 dir()。示例:

>>> s = 'hello'
>>> dir(s)
['__add__', '__class__', '__contains__', ..., 'find', 'format',
'index', 'isalnum', 'isalpha', 'isdigit', 'islower', 'isspace',
'istitle', 'isupper', 'join', 'ljust', 'lower', 'lstrip', 'partition',
'replace', 'rfind', 'rindex', 'rjust', 'rpartition', 'rsplit',
'rstrip', 'split', 'splitlines', 'startswith', 'strip', 'swapcase',
'title', 'translate', 'upper', 'zfill']
>>>

dir() 函數生成一個在 (.) 後出現的全部操做的列表。

使用 help() 函數能夠獲取有關特定操做的更多信息。

>>> help(s.upper)
Help on built-in function upper:

upper(...)
    S.upper() -> string

    Return a copy of the string S converted to uppercase.
>>>

目錄 | 上一節 (1.3 數字) | 下一節 (1.5 列表)

注:完整翻譯見 https://github.com/codists/practical-python-zh

相關文章
相關標籤/搜索