Python核心編程 課後習題 第一部分

Python核心編程 課後習題

 

第一部分 正則表達式

 

1-1 識別後續的字符串:「bat」、「bit」、「but」、「hat」、「hit」或者「hut」

In [3]:
 
 
 
 
 
import re
data = 'bat sad bit ad but d fa hat adfad hit ad da d hut aa'
patt = '[bh][uia]t'
m = re.findall(patt,data)
print(m)
 
 
 
['bat', 'bit', 'but', 'hat', 'hit', 'hut']
 

1-2 匹配由單個空格分隔的任意單詞對,也就是姓和名。

In [75]:
 
 
 
 
 
data = 'adfass323f sadfa aadfa'
patt = '\w+\s\w+'
m = re.match(patt, data)
m.group()
 
 
Out[75]:
'adfass323f sadfa'
 

1-3 匹配由單個逗號和單個空白符分隔的任何單詞和單個字母,如姓氏的首字母。

In [82]:
 
 
 
 
 
data = 'adfaf aa, dadada'
patt = '\w+,\s\w+'
m = re.search(patt, data)
m.group()
 
 
Out[82]:
'aa, dadada'
 

1-4 匹配全部有效 Python 標識符的集合。

In [108]:
 
 
 
 
 
data = 'adaas1212dfdfsadfaaf_'
patt = '[_A-Za-z]+[_\w]+' #有效Python標識符:如下劃線和字母開頭的非空字符,
m = re.match(patt, data)
m.group()
 
 
Out[108]:
'adaas1212dfdfsadfaaf_'
 

1-5 根據讀者當地的格式,匹配街道地址(使你的正則表達式足夠通用,來匹配任意數量的街道單詞,包括類型名稱)。例如,美國街道地址使用以下格式:1180 Bordeaux Drive。使你的正則表達式足夠靈活,以支持多單詞的街道名稱,如 3120 De la Cruz Boulevard。

In [119]:
 
 
 
 
 
data = '3120 De la Cruz Bou levard'
patt = r'\d+.+' #以數字開頭的任何字符串
m = re.match(patt,data)
m.group()
 
 
Out[119]:
'3120 De la Cruz Bou levard'
 

1-6 匹配以「www」起始且以「.com」結尾的簡單Web 域名;例如,www://www. yahoo.com/。選作題:你的正則表達式也能夠支持其餘高級域名,如.edu、.net 等(例如,http://www.foothill.edu)。

In [236]:
 
 
 
 
 
data = 'http://www.y.as.sadf.a121.2asasdoo.cc'
patt = r'http://(www\.[\w|\.]+\.(com|cn|org|net|cc))'
m = re.match(patt,data)
m.group(1)
 
 
Out[236]:
'www.y.as.sadf.a121.2asasdoo.cc'
 

1-7 匹配全部可以表示 Python 整數的字符串集。

In [262]:
 
 
 
 
 
data = '-123141234'
patt = '[-|+]?\d+$'
m = re.match(patt, data)
m.group()
 
 
Out[262]:
'-123141234'
 

1-8 匹配全部可以表示 Python 長整數的字符串集。

In [264]:
 
 
 
 
 
data = '123141234L'
patt = '[-|+]?\d+[lL]'
m = re.match(patt, data)
m.group()
 
 
Out[264]:
'123141234L'
 

1-9 匹配全部可以表示 Python 浮點數的字符串集。

In [272]:
 
 
 
 
 
data = '-12310.09'
patt = '[-|+]?\d+\.\d*'
m = re.match(patt, data)
m.group()
 
 
Out[272]:
'-12310.09'
 

1-10 匹配全部可以表示 Python 複數的字符串集。

In [277]:
 
 
 
 
 
data = '121+989i'
patt = r'\d+(\+\d*i)' #簡單的實部和虛部都是整數
m = re.match(patt, data)
m.group()
 
 
Out[277]:
'121+989i'
 

1-11 匹配全部可以表示有效電子郵件地址的集合(從一個寬鬆的正則表達式開始,而後嘗試使它儘量嚴謹,不過要保持正確的功能)。

In [75]:
 
 
 
 
 
data = 'y.1-a-9.a@t.w-0.9-3-c.com'
patt = r'[a-zA-Z]+([-\.]?\w+)+@((\w+-?\w*)+\.)+\w{2,3}' #6~18個字符,可以使用字母、數字、下劃線,需以字母開頭
m = re.match(patt, data)
m.group()
 
 
Out[75]:
'y.1-a-9.a@t.w-0.9-3-c.com'
 

1-12 匹配全部可以表示有效的網站地址的集合(URL) (從一個寬鬆的正則表達式開始,而後嘗試使它儘量嚴謹,不過要保持正確的功能)。

In [93]:
 
 
 
 
 
data = 'http://www.ba-i.du.com.cn http://ww.qq.com'
patt = r'http://((\w+[-\.])*\w+)'
m = re.match(patt, data).group()
n = re.findall(patt, data)
print(n)
      
 
 
 
[('www.ba-i.du.com.cn', 'com.'), ('ww.qq.com', 'qq.')]
 

1-13 type()。內置函數 type()返回一個類型對象,以下所示,

該對象將表示爲一個 Pythonic類型的字符串。
>>> type(0)
<type 'int'>
>>> type(.34)
<type 'float'>
>>> type(dir)
<type 'builtin_function_or_method'>
建立一個可以從字符串中提取實際類型名稱的正則表達式。函數將對相似於<type'int' >的字符串返回 int(其餘類型也是如此,如 'float' 、'builtin_function_or_method' 等)。
注意:你所實現的值將存入類和一些內置類型的__name__屬性中。
In [104]:
 
 
 
 
 
data = type(.34)
print(data)
type(0)
 
 
 
<class 'float'>
Out[104]:
int
In [125]:
 
 
 
 
 
import re
data = "type 'int' type 'float'"
patt = r'type \'(\w+)\''
m = re.findall(patt, data)
m
 
 
Out[125]:
['int', 'float']
 

1-14 處理日期。1.2 節提供了來匹配單個或者兩個數字字符串的正則表達式模式,來表示1~9 的月份(0?[1-9])。建立一個正則表達式來表示標準日曆中剩餘三個月的數字。

In [118]:
 
 
 
 
 
date = '10 11 12 09'
patt = '1\d'
m = re.findall(patt,date)
m
 
 
Out[118]:
['10', '11', '12']
 

1-15 處理信用卡號碼。1.2 節還提供了一個可以匹配信用卡(CC)號碼([0-9]{15,16})的正則表達式模式。然而,該模式不容許使用連字符來分割數字塊。建立一個容許使用連字符的正則表達式,可是僅能用於正確的位置。例如,15 位的信用卡號碼使用 4-6-5 的模式,代表 4 個數字-連字符-6 個數字-連字符-5 個數字;16 位的信用卡號碼使用 4-4-4-4 的模式。記住,要對整個字符串進行合適的分組。選作題:有一個判斷信用卡號碼是否有效的標準算法。編寫一些代碼,這些代碼不但可以識別具備正確格式的號碼,並且可以識別有效的信用卡號碼。

In [137]:
 
 
 
 
 
data = '1111-111111-11111'
data2 = '1111-1111-1111-1111'
patt = r'\d{4}-\d{4}(\d{2})?-\d{4}\d?(-\d{4})?'
m = re.match(patt, data)
m.group()
 
 
Out[137]:
'1111-111111-11111'
In [12]:
 
 
 
 
 
import re
data = '1111-111111-11111'
data2 = '1111-1111-1111-1111'
patt = r'\d{4}-(\d{4}(\d{2})?)-(\d{4}\d?)(-\d{4})?'
m = re.match(patt, data2)
if m == None:
    print('Error')
elif len(m.group(1))==6 and len(m.group(3))==5:
    print('Yes, It is 465')
elif len(m.group(1))==4 and len(m.group(3))==4:
    print('Yes, it is 4444')
else:
    print('No')
 
 
 
Yes, it is 4444
In [30]:
 
 
 
 
 
#!/usr/bin/env python
#gendata.py
from random import randrange, choice
from string import ascii_lowercase as lc
#from sys import int
from time import ctime
tlds = ('com', 'edu', 'net', 'org', 'gov')
for i in range(randrange(5,11)):
    dtint = randrange(2**32) #pick date
    dtstr = ctime(dtint) #date string
    llen = randrange(4, 8) #login is shorter
    login = ''.join(choice(lc) for j in range(llen))
    dlen = randrange(llen, 13) #domain is longer
    dom = ''.join(choice(lc) for j in range(dlen))
    print('%s::%s@%s.%s::%d-%d-%d' % (dtstr, login,
              dom, choice(tlds), dtint, llen, dlen))
 
 
 
Mon May 11 05:53:50 1981::pjtf@lqhklujz.edu::358379630-4-8
Mon Mar 14 12:54:36 2089::bkucdxr@ozypualw.edu::3761614476-7-8
Sat Apr 18 07:56:58 2043::aothxkd@vpyrdhb.edu::2312927818-7-7
Sun Mar  2 03:00:18 2025::lehjbwe@xfstllxouzx.net::1740855618-7-11
Fri Jul  6 11:51:49 2063::jneswg@bgfyknkxshg.com::2950919509-6-11
Fri Dec  7 09:25:45 2040::coejb@tmknvw.gov::2238456345-5-6
Sat Apr 28 13:21:13 2091::ajummm@tpxrzyav.net::3828576073-6-8
Thu Oct  6 04:48:41 2011::aqjgy@xlbduwhrtn.com::1317847721-5-10
Tue Oct 16 21:14:24 2057::mfqbo@hlvprgl.gov::2770463664-5-7
 

1-16 爲 gendata.py 更新代碼,使數據直接輸出到 redata.txt 而不是屏幕。

In [28]:
 
 
 
 
 
#!/usr/bin/env python
#gendata.py
from random import randrange, choice
from string import ascii_lowercase as lc
#from sys import int
from time import ctime
tlds = ('com', 'edu', 'net', 'org', 'gov')
f = open('redata.txt','w')
for i in range(randrange(5,11)):
    dtint = randrange(2**32) #pick date
    dtstr = ctime(dtint) #date string
    llen = randrange(4, 8) #login is shorter
    login = ''.join(choice(lc) for j in range(llen))
    dlen = randrange(llen, 13) #domain is longer
    dom = ''.join(choice(lc) for j in range(dlen)) 
    data = '%s::%s@%s.%s::%d-%d-%d\n' % (dtstr, login,
              dom, choice(tlds), dtint, llen, dlen)
    f.write(data)
    print('%s::%s@%s.%s::%d-%d-%d' % (dtstr, login,
              dom, choice(tlds), dtint, llen, dlen))
f.close()
 
 
 
Tue Jul 11 20:42:22 2045::udnuh@dophcy.org::2383389742-5-6
Fri Aug 27 11:25:17 2088::yrrjkn@jeofozsgij.com::3744415517-6-10
Mon Dec 27 00:07:53 2027::irroqg@esxxsydenrrz.net::1829837273-6-12
Thu Mar 23 19:43:25 2073::pwxmhj@fewylzchwuog.com::3257495005-6-12
Tue Mar  5 21:18:24 2058::xzyjyj@pybizkhnxsv.org::2782559904-6-11
Wed Feb 23 04:03:57 2061::voplu@plehkhmqixoc.com::2876328237-5-12
Thu Mar 28 13:53:51 2097::otpb@mdpzs.org::4015288431-4-5
Thu Jun  5 01:15:08 2064::szor@czvvpeefz.net::2979825308-4-9
Thu Jul 27 19:44:50 2084::ajsol@gczigg.net::3615536690-5-6
Tue Sep 18 05:03:34 2007::bvboc@qnyznjngbxtl.org::1190063014-5-12
 

1-17 判斷在redata.txt中一週的每一天出現的次數(換句話說,讀者也能夠計算所選擇的年份中每月中出現的次數)。

In [48]:
 
 
 
 
 
#Fweekday = ['Mon','Tue','Wed','Thu','Fri','Sat','Sun']
import re
patt = r'^\w{3}\s'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['Tue ']
['Fri ']
['Thu ']
['Sun ']
['Fri ']
['Tue ']
In [50]:
 
 
 
 
 
import re
patt = r'\s\w{3}\s'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
[' Jul ']
[' Sep ']
[' Apr ']
[' Jul ']
[' Sep ']
[' May ']
 

1-18 經過確認整數字段中的第一個整數匹配在每一個輸出行起始部分的時間戳,確保在redata.txt 中沒有數據損壞。

In [3]:
 
 
 
 
 
import re
patt = r'^\w{3}\s\w{3}\s\s?\d{1,2}\s\d{2}:\d{2}:\d{2}'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['Tue Jul 22 21:43:16']
['Fri Sep  6 19:02:00']
['Thu Apr 15 16:56:13']
['Sun Jul 28 20:56:50']
['Fri Sep  8 07:04:11']
['Tue May 10 05:08:50']
 

建立如下正則表達式。1-19 提取每行中完整的時間戳。

In [4]:
 
 
 
 
 
import re
patt = r'^\w{3}\s\w{3}\s\s?\d{1,2}\s\d{2}:\d{2}:\d{2}'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['Tue Jul 22 21:43:16']
['Fri Sep  6 19:02:00']
['Thu Apr 15 16:56:13']
['Sun Jul 28 20:56:50']
['Fri Sep  8 07:04:11']
['Tue May 10 05:08:50']
 

1-20 提取每行中完整的電子郵件地址。

In [7]:
 
 
 
 
 
import re
patt = r'\w+@\w+\.\w{3}'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['qevi@hubpbwd.org']
['anbbwuj@guxnppqjbc.org']
['cckac@idfxboaglql.org']
['fgtfs@uwwmdqdemn.gov']
['dfzxv@yjukqnjo.gov']
['rbbwftb@hpjstxgnk.org']
 

1-21 僅僅提取時間戳中的月份。

In [9]:
 
 
 
 
 
import re
patt = r'\s\w{3}\s'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
[' Jul ']
[' Sep ']
[' Apr ']
[' Jul ']
[' Sep ']
[' May ']
 

1-22 僅僅提取時間戳中的年份。

In [12]:
 
 
 
 
 
import re
patt = r'(\d{4}):'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['2104']
['1985']
['2032']
['2052']
['2102']
['2072']
 

1-23 僅僅提取時間戳中的時間(HH:MM:SS)。

In [24]:
 
 
 
 
 
import re
patt = r'\d{2}:\d{2}:\d{2}'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['22:23:19']
['17:59:34']
['20:53:48']
['00:55:14']
['12:33:38']
 

1-24 僅僅從電子郵件地址中提取登陸名和域名(包括主域名和高級域名一塊兒提取)。

In [15]:
 
 
 
 
 
import re
patt = r'@(\w+\.\w{3})'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
['hubpbwd.org']
['guxnppqjbc.org']
['idfxboaglql.org']
['uwwmdqdemn.gov']
['yjukqnjo.gov']
['hpjstxgnk.org']
 

1-25 僅僅從電子郵件地址中提取登陸名和域名(包括主域名和高級域名)。

In [17]:
 
 
 
 
 
import re
patt = r'@(\w+)\.(\w{3})'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
[('hubpbwd', 'org')]
[('guxnppqjbc', 'org')]
[('idfxboaglql', 'org')]
[('uwwmdqdemn', 'gov')]
[('yjukqnjo', 'gov')]
[('hpjstxgnk', 'org')]
 

1-26 使用你的電子郵件地址替換每一行數據中的電子郵件地址。

In [31]:
 
 
 
 
 
import re
patt = r'\w+@\w+\.\w{3}'
f = open('redata.txt','r')
for i in f:
    #print(i)
    l = re.sub(patt,'xx@yy.com',i)
    print(l)
f.close()
 
 
 
Tue Jul 11 20:42:22 2045::xx@yy.com::2383389742-5-6

Fri Aug 27 11:25:17 2088::xx@yy.com::3744415517-6-10

Mon Dec 27 00:07:53 2027::xx@yy.com::1829837273-6-12

Thu Mar 23 19:43:25 2073::xx@yy.com::3257495005-6-12

Tue Mar  5 21:18:24 2058::xx@yy.com::2782559904-6-11

Wed Feb 23 04:03:57 2061::xx@yy.com::2876328237-5-12

Thu Mar 28 13:53:51 2097::xx@yy.com::4015288431-4-5

Thu Jun  5 01:15:08 2064::xx@yy.com::2979825308-4-9

Thu Jul 27 19:44:50 2084::xx@yy.com::3615536690-5-6

Tue Sep 18 05:03:34 2007::xx@yy.com::1190063014-5-12

 

1-27 從時間戳中提取月、日和年,而後以「月,日,年」的格式,每一行僅僅迭代一次。

In [35]:
 
 
 
 
 
import re
patt = r'(\w{3})\s\s?(\d{1,2})\s\d{2}:\d{2}:\d{2}\s(\d{4})'
f = open('redata.txt','r')
for i in f:
    print(re.findall(patt, i.rstrip()))
f.close()
 
 
 
[('Jul', '11', '2045')]
[('Aug', '27', '2088')]
[('Dec', '27', '2027')]
[('Mar', '23', '2073')]
[('Mar', '5', '2058')]
[('Feb', '23', '2061')]
[('Mar', '28', '2097')]
[('Jun', '5', '2064')]
[('Jul', '27', '2084')]
[('Sep', '18', '2007')]
相關文章
相關標籤/搜索