|
第二章 序列
1. 序列——python最基本的数据结构
>>> edward = ['Edward Gumby', 42]
>>> john = ['John Smith', 50]
>>> database = [edward, john]
>>> database
[['Edward Gumby', 42], ['John Smith', 50]]
2. 序列操作
(1)索引
>>> greeting = 'Hello'
>>> greeting[0] #从左边开始计数,从0开始依次递增
'H'
>>> greeting[4]
'o'
>>> greeting[-1] #从右边开始计数,从-1开始依次递减
'o'
>>> greeting[-5]
'H'
>>> 'Hello'[1] #对字符串直接使用索引
'e'
>>> fourth = raw_input('Year: ')[3]
Year: 2011
>>> fourth
'1'
应用索引的一个完整示例
#根据给定的年月日以数字形式打印日期
months = ['January', 'February', 'March', 'April', 'May', 'June', 'July',
'August', 'Semptember', 'October', 'November', 'December']
#以1~31的数字作为结尾的列表
endings = ['st', 'nd', 'rd'] + 17 * ['th']\ #"\"表示该行未完
+ ['st', 'nd', 'rd'] + 7 * ['th']\
+ ['st']
year = raw_input("Year: ")
month = raw_input("Month(1-12): ")
day = raw_input("Day(1-31): ")
month_number = int(month)
day_number = int(day)
#将月份和天书减1,以获得正确的索引
month_name = months[month_number - 1]
day_name = day + endings[day_number - 1]
print month_name + ' ' + day_name + ',' + year
#运行结果
Year: 2011
Month(1-12): 9
Day(1-31): 5
Semptember 5th,2011
(2)分片
>>> numbers = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] #索引左边从0开始递增,右边从-1开始递减
>>> numbers[7:10] #(1)开始点的元素包含在结果内,而不包含结束点元素
[8, 9, 10]
>>> numbers[-3:-1] #同样也可使用负数来表示,但因不含结束点元素,所以无法得到与(1)同样效果
[8, 9]
>>> numbers[-3:0] #开始点索引比结束点索引晚出现在序列中,结果为空序列,倒数第三个比第一个数出现晚
[]
>>> numbers[-3:] #与(1)效果相同,若要包括序列结尾元素,只需置空结束点索引即可。
[8, 9, 10]
>>> numbers[:3] #同样,可将开始点索引置空
[1, 2, 3]
>>> numbers[:] #均置空,实现序列复制
[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
#步长
>>> numbers[0:10:1] #默认情况下步长为1
[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
>>> numbers[0:10:2] #步长为2
[1, 3, 5, 7, 9]
>>> numbers[::2] #(2)开始点和结束点为空时,默认从开头取至结尾[1, 3, 5, 7, 9]
>>> numbers[10:0:-1] #步长可以为负数,但要保证开始点索引比结束点大
[10, 9, 8, 7, 6, 5, 4, 3, 2]
>>> numbers[10:0:-2]
[10, 8, 6, 4, 2]
>>> numbers[::-2] #同(2),但因为从右往左,所以开始点为空默认从序列头部10,结束点为空默认为尾部0
[10, 8, 6, 4, 2]
>>> numbers[10::-2] #同理
[10, 8, 6, 4, 2]
>>> numbers[:0:-2] #同理
[10, 8, 6, 4, 2]
(3)加法——只有同类型的序列才可以相加(连接)
>>> [1, 2, 3] + [4, 5, 6]
[1, 2, 3, 4, 5, 6]
>>> "Hello. " + "world!"
'Hello. world!'
>>> [1, 2, 3] + "world!"
Traceback (most recent call last):
File "<pyshell#25>", line 1, in <module>
[1, 2, 3] + "world!"
TypeError: can only concatenate list (not "str") to list
(4)序列乘以一个数n,生成新序列,重复原序列n次。
>>> "python" * 5
'pythonpythonpythonpythonpython'
>>> [42] * 5
[42, 42, 42, 42, 42]
>>> [None] * 5 #占用5个元素的空间,却不包含任何内容,None注意首字母大写
[None, None, None, None, None]
乘法的一个实例
#以正确的宽度在居中的”盒子“内打印一个句子
sentence = raw_input("Sentence: ")
screen_width = 80
text_width = len(sentence) #len():获取序列长度
box_width = text_width + 6
left_margin = (screen_width - box_width) // 2 #"//"整除,使得box在screen中居中
print
print ' ' * left_margin + '+' + '-' * (box_width - 2) + '+'
print ' ' * left_margin + '|' + ' ' * (box_width - 2) + '|'
print ' ' * left_margin + '|' + ' ' + sentence + ' |'
print ' ' * left_margin + '|' + ' ' * (box_width - 2) + '|'
print ' ' * left_margin + '+' + '-' * (box_width - 2) + '+'
print
#显示结果
Sentence: Hello world
+---------------+
| |
| Hello world |
| |
+---------------+
(5)成员资格——检查一个值是否存在于序列中
>>> permission= 'rw'
>>> 'w' in test #可用于检查文件是否可读写
True
>>> 's' in test
False
>>> users = ['mlh', 'foo', 'bar']
>>> raw_input("Enter your name: ") in users #检查用户是否在列表中
Enter your name: mlh
True
>>> 'test' in 'yourtest' #直接检查
True
完整示例
#检查用户名和PIN码
database = [
['albert', '1234'],
['dilbert', '2345'],
['simith', '3456'],
['johns', '4567']
]
username = raw_input("user name: ")
pin = raw_input("PIN code: ")
if [username, pin] in database: print "Access granted"
else: print "Access rejected"
#运行结果
>>>
user name: alber
PIN code: 1234
Access rejected
>>> ================================ RESTART ================================
>>>
user name: albert
PIN code: 1234
Access granted
(5)len(), max(), min()
>>> numbers = [2, 3, 5]
>>> len(numbers)
3
>>> max(numbers)
5
>>> min(numbers)
2
>>> max(2,4,9)
9
>>> min(1, 4, 5)
1
3. 列表——可进行修改
因为字符串不能被修改,所以将字符串转换为列表后进行修改
(1)列表操作
>>> list("Hello") #list()将字符串转为列表
['H', 'e', 'l', 'l', 'o']
>>> x = [1, 1, 1]
>>> x[1] = 2 #根据索引赋值,直接修改内容
>>> x
[1, 2, 1]
>>> del x[2] #del删除元素
>>> x
[1, 2]
>>> name = list('Perl')
>>> name
['P', 'e', 'r', 'l']
>>> name[1:] = list('ython') #使用分片进行赋值,长度可以与原长度不同
>>> name
['P', 'y', 't', 'h', 'o', 'n']
>>> name[0:0] = list('Hello ') #插入
>>> name
['H', 'e', 'l', 'l', 'o', ' ', 'P', 'y', 't', 'h', 'o', 'n']
>>> name[0:6] = [] #相当于del
>>> name
['P', 'y', 't', 'h', 'o', 'n']
#还可以使用步长,负数索引等。
(3)列表方法——与函数的不同,方法是与对象紧密联系的函数
#append()在列表末尾追加元素,它直接修改列表,而不是生成新的列表
>>> lst = [1, 2, 3]
>>> lst.append(4)
>>> lst
[1, 2, 3, 4]
#count()统计元素在列表中出现次数
>>> ['to', 'be', 'or', 'to'].count('to')
2
>>> x = [[1,2], 1, 1, [2, 1, [1, 2]]]
>>> x.count(1)
2
>>> x.count([1,2])
1
#extend()将一个列表B追加到另一个列表A之后,修改原列表A
>>> a = [1, 2, 3]
>>> b = [4, 5, 6]
>>> a.extend(b) #a列表被修改
>>> a
[1, 2, 3, 4, 5, 6]
>>> a = [1, 2, 3]
>>> b = [4, 5, 6]
>>> a + b #a列表未被修改,生成了一个包含a和b的副本的新列表
[1, 2, 3, 4, 5, 6] #若用a = a + b,效率比extend低
>>> a
[1, 2, 3]
>>> a[len(a):] = b #用分片可以实现对a列表的修改,可读性差
>>> a
[1, 2, 3, 4, 5, 6]
#index()找出匹配项的第一个索引位置
>>> a = ['1', '2', '3', '1']
>>> a.index('1')
0
#insert()插入
>>> numbers = [1, 2, 3, 5, 6, 7]
>>> numbers.insert(3, 'four') #在索引3之前插入,插入到第三个元素之后
>>> numbers
[1, 2, 3, 'four', 5, 6, 7]
>>> numbers = [1, 2, 3, 5, 6, 7]
>>> numbers[3:3] = ['four'] #分片实现插入,同样在在索引3之前插入,但可读性差
>>> numbers
[1, 2, 3, 'four', 5, 6, 7]
>>> #pop()移除列表元素(默认最后一个),并返回该元素的值,是唯一一个及修改列表又返回元素值的列表方法
>>> #可以用pop()构造栈,用append模拟push,先入后出
>>> x = [1, 2, 3]
>>> y = x.pop()
>>> y
3 #输出的是要移除的元素值
>>> x.append(y)
>>> x
[1, 2, 3]
#上述可写成
>>> x.append(x.pop()) #pop后又加入,相当于不变
>>> x
[1, 2, 3]
>>>
#remove()删除第一个匹配项
>>> x = ['to', 'be', 'or', 'not', 'to', 'be']
>>> x.remove('be')
>>> x
['to', 'or', 'not', 'to', 'be']
#reverse()将列表中元素反向存放
>>> x = [1, 2, 3]
>>> x.reverse()
>>> x
[3, 2, 1]
#sort()对原列表进行排序,原列表发生改变
>>> x = [4, 6, 2, 1, 7, 9]
>>> x.sort()
>>> x
[1, 2, 4, 6, 7, 9]
>>>
#如何实现排序副本,并保持原列表不变呢
#错误方法
>>> x = [4, 6, 2, 1, 7, 9]
>>> y = x.sort()
>>> print y
None #因为sort()无返回值
#正确方法
>>> x = [4, 6, 2, 1, 7, 9]
>>> y = x[:] #高效的复制方法,单纯的将x赋值给y是没用的,这用是x、y都指向同一个列表
>>> y.sort()
>>> x #x未发生改变
[4, 6, 2, 1, 7, 9]
>>> y #只对x的副本y进行了排序
[1, 2, 4, 6, 7, 9]
>>>
#sorted()另一种获取副本的方法
>>> x = [4, 6, 2, 1, 7, 9]
>>> y = sorted(x)
>>> x
[4, 6, 2, 1, 7, 9] #x未发生改变
>>> y
[1, 2, 4, 6, 7, 9] #只对x的副本y进行了排序
>>> #简单写法
>>> sorted(x) #x未变,只对x的副本排序
[1, 2, 4, 6, 7, 9]
>>>
>>> cmp(2,1) #cmp(x,y),当x>y时,返回正数
1
>>> cmp(1,2) #当x<y时,返回负数
-1
>>> cmp(1,1) #当x=y时,返回0
0
>>>
>>> #cmp作为sort的参数????????
>>> numbers = [2, 9, 3 ,6]
>>> numbers.sort(cmp) #不明白???与一般的.sort()有什么区别??
>>> numbers
[2, 3, 6, 9]
>>>
>>> #sort()的可选参数,key和reverse
>>> x = ['ab', 'abc', 'a', 'abcd']
>>> x.sort(key=len) #以元素的长度来排序
>>> x
['a', 'ab', 'abc', 'abcd']
>>>
>>> x = [4, 6, 2, 1, 7, 9]
>>> x.sort(reverse=True) #反向排序
>>> x
[9, 7, 6, 4, 2, 1]
4. 元组——不可变的序列
元组用()表示,而列表用[]表示
>>> 1, 2, 3 #用逗号隔开即可创建新元组
(1, 2, 3)
>>> (1, 2, 3) #同上
(1, 2, 3)
>>> () #空元组
()
>>> 42 #数字
42
>>> 42, #添加逗号,就变成了只有一个元素的元组
(42,)
>>> (42,) #同上
(42,)
>>> (42) #没有逗号只有括号,同样是数字,而不是元组
42
>>> 3 * (40 + 2) #数字计算
126
>>> 3 * (40 + 2,) #元组相乘
(42, 42, 42)
#tuple()以一序列作为参数,将其转换为元组
>>> tuple([1, 2, 3]) #列表转换为元组
(1, 2, 3)
>>> tuple('abc') #字符串转换为元组
('a', 'b', 'c')
>>> tuple((1,2,3)) #元组转换为元组
(1, 2, 3)
#一些基本操作
>>> x = 1, 2, 3
>>> x[1] #根据索引取值
2
>>> x[0:2] #分片,元组分片后仍为元组,而列表分片后仍未列表
(1, 2)
5. 元组不可修改,与列表相比存在的意义?
(1)元组可以在映射中当键使用,而列表不可以
(2)很多内建函数和方法的返回值都是元组
6. 总结
列表和元组都属于序列,列表可以修改,元组不可变。内容比较简单,同样繁琐易忘
涉及函数
cmp(x,y) 比较x,y
len(seq) 返回序列长度
list(seq) 序列转为列表
max(args)
min(args)
reversed(seq) 对列表反响迭代
sorted(seq) 返回原列表的排序列表副本 x.sort() 属于方法,而不是函数
tuple(seq)序列转为元组 |
|