[TOC]
### list
#### list是一种有序的集合
Python内置的一种数据类型是列表:list。list是一种有序的集合,可以随时添加和删除其中的元素。
比如,列出班里所有同学的名字,就可以用一个`list`表示:
```
>>> classmates = ['Michael', 'Bob', 'Tracy']
>>> classmates
['Michael', 'Bob', 'Tracy']
```
- 变量`classmates`就是一个`list`。用`len()`函数可以获得`list`元素的个数:
```
>>> len(classmates)
3
```
- 用索引来访问`list`中每一个位置的元素,记得索引是从`0`开始的:
```
>>> classmates[0]
'Michael'
>>> classmates[1]
'Bob'
>>> classmates[2]
'Tracy'
>>> classmates[3]
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
IndexError: list index out of range
```
当索引超出了范围时,Python会报一个`IndexError`错误,所以,要确保索引不要越界,记得最后一个元素的索引是`len(classmates) - 1`。
>[info] 如果要取最后一个元素,除了计算索引位置外,还可以用`-1`做索引,直接获取最后一个元素:
```
>>> classmates[-1]
'Tracy'
```
>[info] 以此类推,可以获取倒数第2个、倒数第3个:
```
>>> classmates[-2]
'Bob'
>>> classmates[-3]
'Michael'
>>> classmates[-4]
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
IndexError: list index out of range
```
>[warning] 当然,倒数第`4`个就越界了。
#### `list`是一个可变的有序表
`list`是一个可变的有序表,所以,可以往`list`中追加元素到末尾:
```
>>> classmates.append('Adam')
>>> classmates
['Michael', 'Bob', 'Tracy', 'Adam']
```
也可以把元素插入到指定的位置,比如索引号为`1`的位置:
```
>>> classmates.insert(1, 'Jack')
>>> classmates
['Michael', 'Jack', 'Bob', 'Tracy', 'Adam']
```
- 要删除`list`末尾的元素,用`pop()`方法:
```
>>> classmates.pop()
'Adam'
>>> classmates
['Michael', 'Jack', 'Bob', 'Tracy']
```
- 要删除指定位置的元素,用`pop(i)`方法,其中`i`是索引位置:
```
>>> classmates.pop(1)
'Jack'
>>> classmates
['Michael', 'Bob', 'Tracy']
```
- 要把某个元素替换成别的元素,可以直接赋值给对应的索引位置:
```
>>> classmates[1] = 'Sarah'
>>> classmates
['Michael', 'Sarah', 'Tracy']
```
#### `list`里面的元素的数据类型也可以不同
`list`里面的元素的数据类型也可以不同,比如:
```
>>> L = ['Apple', 123, True]
```
- `list`元素也可以是另一个`list`,比如:
```
>>> s = ['python', 'java', ['asp', 'php'], 'scheme']
>>> len(s)
4
```
要注意`s`只有`4`个元素,其中`s[2]`又是一个`list`,如果拆开写就更容易理解了:
```
>>> p = ['asp', 'php']
>>> s = ['python', 'java', p, 'scheme']
```
要拿到`'php'`可以写`p[1]`或者`s[2][1]`,因此`s`可以看成是一个二维数组,类似的还有三维、四维……数组,不过很少用到。
- 如果一个`list`中一个元素也没有,就是一个空的`list`,它的长度为0:
```
>>> L = []
>>> len(L)
0
```
### tuple
#### 不能修改的有序表
另一种有序列表叫元组:`tuple`。`tuple`和`list`非常类似,但是`tuple`一旦初始化就不能修改,比如同样是列出同学的名字:
```
>>> classmates = ('Michael', 'Bob', 'Tracy')
```
现在,`classmates`这个`tuple`不能变了,它也没有`append()`,`insert()`这样的方法。其他获取元素的方法和`list`是一样的,你可以正常地使用`classmates[0]`,`classmates[-1]`,但不能赋值成另外的元素。
#### 不可变的`tuple`有什么意义?
>[info] 不可变的`tuple`有什么意义?因为`tuple`不可变,所以代码更安全。如果可能,能用`tuple`代替`list`就尽量用`tuple`。
#### tuple的陷阱
>[info] tuple的陷阱:当你定义一个tuple时,在定义的时候,tuple的元素就必须被确定下来,比如:
```
>>> t = (1, 2)
>>> t
(1, 2)
```
- 如果要定义一个空的`tuple`,可以写成`()`:
```
>>> t = ()
>>> t
()
```
- **但是,要定义一个只有`1`个元素的`tuple`,如果你这么定义:**
```
>>> t = (1)
>>> t
1
```
**定义的不是`tuple`,是`1`这个数!**这是因为括号`()`既可以表示`tuple`,又可以表示数学公式中的小括号,这就产生了歧义,因此,Python规定,这种情况下,按小括号进行计算,计算结果自然是`1`。
>[info] 所以,只有1个元素的tuple定义时必须加一个逗号,,来消除歧义:
```
>>> t = (1,)
>>> t
(1,)
```
**Python在显示只有`1`个元素的`tuple`时,也会加一个逗号,,以免你误解成数学计算意义上的括号。**
#### 一个“可变的”`tuple`
最后来看一个“可变的”`tuple`:
```
>>> t = ('a', 'b', ['A', 'B'])
>>> t[2][0] = 'X'
>>> t[2][1] = 'Y'
>>> t
('a', 'b', ['X', 'Y'])
```
这个tuple定义的时候有`3`个元素,分别是`'a'`,`'b'`和一个`list`。不是说`tuple`一旦定义后就不可变了吗?怎么后来又变了?
**别急,我们先看看定义的时候`tuple`包含的`3`个元素:**
![image](http://www.liaoxuefeng.com/files/attachments/001387269705541ad608276b6f7426ca59b8c2b19947243000/0)
**当我们把`list`的元素`'A'`和`'B'`修改为`'X'`和`'Y'`后,`tuple`变为:**
![image](http://www.liaoxuefeng.com/files/attachments/001387269768140c7d5ca167342402989dfc75343fe900b000/0)
**表面上看,tuple的元素确实变了,但其实变的不是tuple的元素,而是list的元素。**`tuple`一开始指向的`list`并没有改成别的`list`,所以,**`tuple`所谓的“不变”是说,`tuple`的每个元素,指向永远不变。**即指向`'a'`,就不能改成指向`'b'`,指向一个`list`,就不能改成指向其他对象,但指向的这个`list`本身是可变的!
#### tuple “指向不变”性
>[info] 理解了“指向不变”后,要创建一个内容也不变的`tuple`怎么做?那就必须保证`tuple`的每一个元素本身也不能变。
### 练习
#### 请用索引取出下面`list`的指定元素:
```
# -*- coding: utf-8 -*-
L = [
['Apple', 'Google', 'Microsoft'],
['Java', 'Python', 'Ruby', 'PHP'],
['Adam', 'Bart', 'Lisa']
]
```
### 小结
#### `list`和`tuple`
`list`和`tuple`是Python内置的有序集合,一个可变,一个不可变。根据需要来选择使用它们。
### 练习参考源码
```
# -*- coding: utf-8 -*-
L = [
['Apple', 'Google', 'Microsoft'],
['Java', 'Python', 'Ruby', 'PHP'],
['Adam', 'Bart', 'Lisa']
]
# 打印Apple:
print(L[0][0])
# 打印Python:
print(L[2][1])
# 打印Lisa:
print(L[-1][-1])
```
- Python教程
- Python简介
- 安装Python
- Python解释器
- 第一个 Python 程序
- 使用文本编辑器
- Python代码运行助手
- 输入和输出
- 源码
- learning.py
- Python基础
- 数据类型和变量
- 字符串和编码
- 使用list和tuple
- 条件判断
- 循环
- 使用dict和set
- 函数
- 调用函数
- 定义函数
- 函数的参数
- 递归函数
- 高级特性
- 切片
- 迭代
- 列表生成式
- 生成器
- 迭代器
- 函数式编程
- 高阶函数
- map/reduce
- filter
- sorted
- 返回函数
- 匿名函数
- 装饰器
- 偏函数
- Python函数式编程——偏函数(来自博客)
- 模块
- 使用模块
- 安装第三方模块
- 面向对象编程
- 类和实例
- 访问限制
- 继承和多态
- 获取对象信息
- 实例属性和类属性
- 面向对象高级编程
- 使用__slots__
- 使用@property
- 多重继承
- 定制类
- 使用枚举类
- 使用元类
- 错误、调试和测试
- 错误处理
- 调试
- 单元测试
- 文档测试
- IO编程
- 文件读写
- StringIO和BytesIO
- 操作文件和目录
- 序列化
- 进程和线程
- 多进程
- 多线程
- ThreadLocal
- 进程 vs. 线程
- 分布式进程
- 正则表达式
- 常用内建模块
- datetime
- collections
- base64
- struct
- hashlib
- itertools
- contextlib
- XML
- HTMLParser
- urllib
- 常用第三方模块
- PIL
- virtualenv
- 图形界面
- 网络编程
- TCP/IP简介
- TCP编程
- UDP编程
- 电子邮件
- SMTP发送邮件
- POP3收取邮件
- 访问数据库
- 使用SQLite
- 使用MySQL
- 使用SQLAlchemy
- Web开发
- HTTP协议简介
- HTML简介
- WSGI接口
- 使用Web框架
- 使用模板
- 异步IO
- 协程
- asyncio
- async/await
- aiohttp
- 实战
- Day 1 - 搭建开发环境
- Day 2 - 编写Web App骨架
- Day 3 - 编写ORM
- Day 4 - 编写Model
- Day 5 - 编写Web框架
- Day 6 - 编写配置文件
- Day 7 - 编写MVC
- Day 8 - 构建前端
- Day 9 - 编写API
- Day 10 - 用户注册和登录
- Day 11 - 编写日志创建页
- Day 12 - 编写日志列表页
- Day 13 - 提升开发效率
- Day 14 - 完成Web App
- Day 15 - 部署Web App
- Day 16 - 编写移动App
- FAQ
- 期末总结