博客
关于我
一文搞懂Python中的所有数组数据类型
阅读量:204 次
发布时间:2019-02-28

本文共 3874 字,大约阅读时间需要 12 分钟。

数组类型是各种编程语言中基本的数组结构了,本文来盘点下Python中各种“数组”类型的实现。

  • list
  • tuple
  • array.array
  • str
  • bytes
  • bytearray
    其实把以上类型都说成是数组是不准确的。这里把数组当作一个广义的概念,即把列表、序列、数组都当作array-like数据类型来理解。

(想自学习编程的小伙伴请搜索,更多行业相关资讯更有行业相关免费视频教程。完全免费哦!)

注意本文所有代码都是在Python3.7中跑的_

0x00 可变的动态列表list

list应该是Python最常用到的数组类型了。它的特点是可变的、能动态扩容,可存储Python中的一切对象,使用时不用指定存储的元素的类型。

使用非常简单

>>> arr = ["one","two","three"]>>> arr[0]'one'# 动态扩容>>> arr.append(4)>>> arr['one', 'two', 'three', 4]# 删除一个元素>>> del arr[2]>>> arr['one', 'two', 4]

0x01 不可变的tuple

tuple的操作与list类似。它的特点是不可变,不能扩容,可存储Python中的一切对象,使用时不用指定存储的元素的类型。

>>> t = 'one','two',3>>> t('one', 'two', 3)>>> t.append(4)AttributeError: 'tuple' object has no attribute 'append'>>> del t[0]TypeError: 'tuple' object doesn't support item deletion

tuple可以使用+运算符,这个运算将创建一个新的tuple对象用于存储数据。

>>> t+(1,)('one', 'two', 3, 1)>>> tcopy = t+(1,)>>> tcopy('one', 'two', 3, 1)>>> id(tcopy)4604415336>>> id(t)4605245696

可以看出tuple执行+运算符之后两个对象的地址是不一样

0x02 array.array

如果在Python中要用到其它语言中类似“数组”的数据结构,就需要用到array模块了。它的特点是可变的、存储相同类型的数值,不能存储对象。

因为array在使用的时候要指定元素数据类型,因此它比listtuple都有比较高效空间性能。

# 使用时指定元素数据类型为`float`>>> arr = array.array('f', (1.0, 1.5, 2.0, 2.5))>>> arrarray('f', [1.0, 1.5, 2.0, 2.5])# 修改一个元素>>> arr[1]=12.45>>> arrarray('f', [1.0, 12.449999809265137, 2.0, 2.5])# 删除一个元素>>> del arr[2]>>> arrarray('f', [1.0, 12.449999809265137, 2.5])# 增加一个元素>>> arr.append(4.89)>>> arrarray('f', [1.0, 12.449999809265137, 2.5, 4.889999866485596])# 如果将一个字符串类型数据存储到一个浮点数的数组将会报错>>> arr[0]='hello'TypeError: must be real number, not str

array中元素的数据类型可以参考下表

Type code C Type Python Type
‘b’ signed char int
‘B’ unsigned char int
‘u’ Py_UNICODE Unicode character
‘h’ signed short int
‘H’ unsigned short int
‘i’ signed int int
‘I’ unsigned int int
‘l’ signed long int
‘L’ unsigned long int
‘q’ signed long long int
‘Q’ unsigned long long int
‘f’ float float
‘d’ double float

0x03 字符串序列str

Python3中使用str对象来表示一个文本字符序列(看,这跟Java中的字符串String是多么相似呢)。它的特点不可变的Unicode字符序列。

str中它的每一个元素都是字符串对象。

>>> s ='123abc'>>> s'123abc'>>> s[0]'1'>>> s[2]'3'# 字符串是不可变的序列,不能删除其中的元素>>> del s[1]TypeError: 'str' object doesn't support item deletion  # 要对字符串进行操作,可以转化成list  >>> sn = list(s)>>> sn['1', '2', '3', 'a', 'b', 'c']>>> sn.append(9)>>> sn['1', '2', '3', 'a', 'b', 'c', 9]# 字符串中的元素也是字符串对象>>> type(s[2])
>>> type(s)

str对象也可以执行+操作,它也会生成一个新对象用于存储。

>>> s2 = s+'33'>>> s2'123abc33'>>> id(s2)4605193648>>> id(s)4552640416

0x04 bytes

bytes对象用于存储字节序列,它的特点是不可变存储,可存储0-256的数值。

>>> b = bytes([0,2,4,8])>>> b[2]4>>> bb'\x00\x02\x04\x08'>>> b[0]=33TypeError: 'bytes' object does not support item assignment>>> del b[0]TypeError: 'bytes' object doesn't support item deletion

0x05 bytearray

bytearray对象与bytes类似,用于存储字节序列。它的特点是可变的,能动态扩容的字节数组。

>>> ba = bytearray((1,3,5,7,9))>>> babytearray(b'\x01\x03\x05\x07\t')>>> ba[1]3# 删除一个元素>>> del ba[1]>>> babytearray(b'\x01\x05\x07\t')>>> ba[0]=2>>> ba[0]2# 添加一个元素>>> ba.append(6)# 只能添加字节>>> ba.append(s)TypeError: 'str' object cannot be interpreted as an integer>>> babytearray(b'\x02\x05\x07\t\x06')# 字节的范围是0-256>>> ba[2]=288ValueError: byte must be in range(0, 256)

bytearray可以转化成bytes对象,但效率不是很高。

# bytearray转成bytes将生成一个新对象>>> bn = bytes(ba)>>> id(bn)4604114344>>> id(ba)4552473544

0x06 各个类型相互转化

tuple->list

>>> tuple(l)('a', 'b', 'c')

list->tuple

>>> t('a', 'b', 'c')>>> list(t)['a', 'b', 'c']

str->list

>>> l = list('abc')>>> l['a', 'b', 'c']

list->str

>>> l['a', 'b', 'c']>>> ''.join(l)'abc'

str->bytes

>>> s = '123'>>> bytes(s)TypeError: string argument without an encoding>>> bytes(s,encoding='utf-8')b'123'# 或者使用str的encode()方法>>> s.encode()b'123'

bytes->str

>>> b = b'124'>>> bb'124'>>> type(b)
>>> str(b,encoding='utf-8')'124'# 或使用bytes的decode()>>> b.decode()'124'

0x07 总结

这些数据类型都是Python自带的,在实际开发中应该根据具体需求选择合适的数据类型。例如当要存储的元素类型是多种多样的,那么就应该使用list或者tuple。而array.array相对来说拥有较好的空间性能,但它只能存储单一类型。

我相信在很多业务场景中listtuple是可以满足需求的,只是其它数据结构也要有所了解,在我们做一些基础组件时,会考虑数据结构的性能,或者阅读他人的代码时,能做到心中有数。

转载地址:http://pddp.baihongyu.com/

你可能感兴趣的文章
NHibernate学习[1]
查看>>
NHibernate异常:No persister for的解决办法
查看>>
Nhibernate的第一个实例
查看>>
NHibernate示例
查看>>
nid修改oracle11gR2数据库名
查看>>
NIFI1.21.0/NIFI1.22.0/NIFI1.24.0/NIFI1.26.0_2024-06-11最新版本安装_采用HTTP方式_搭建集群_实际操作---大数据之Nifi工作笔记0050
查看>>
NIFI1.21.0_java.net.SocketException:_Too many open files 打开的文件太多_实际操作---大数据之Nifi工作笔记0051
查看>>
NIFI1.21.0_Mysql到Mysql增量CDC同步中_日期类型_以及null数据同步处理补充---大数据之Nifi工作笔记0057
查看>>
NIFI1.21.0_Mysql到Mysql增量CDC同步中_补充_插入时如果目标表中已存在该数据则自动改为更新数据_Postgresql_Hbase也适用---大数据之Nifi工作笔记0058
查看>>
NIFI1.21.0_Mysql到Mysql增量CDC同步中_补充_更新时如果目标表中不存在记录就改为插入数据_Postgresql_Hbase也适用---大数据之Nifi工作笔记0059
查看>>
NIFI1.21.0_NIFI和hadoop蹦了_200G集群磁盘又满了_Jps看不到进程了_Unable to write in /tmp. Aborting----大数据之Nifi工作笔记0052
查看>>
NIFI1.21.0_Postgresql和Mysql同时指定库_指定多表_全量同步到Mysql数据库以及Hbase数据库中---大数据之Nifi工作笔记0060
查看>>
NIFI1.21.0最新版本安装_连接phoenix_单机版_Https登录_什么都没改换了最新版本的NIFI可以连接了_气人_实现插入数据到Hbase_实际操作---大数据之Nifi工作笔记0050
查看>>
NIFI1.21.0最新版本安装_配置使用HTTP登录_默认是用HTTPS登录的_Https登录需要输入用户名密码_HTTP不需要---大数据之Nifi工作笔记0051
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_增删改数据分发及删除数据实时同步_通过分页解决变更记录过大问题_02----大数据之Nifi工作笔记0054
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_增加修改实时同步_使用JsonPath及自定义Python脚本_03---大数据之Nifi工作笔记0055
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表多表增量同步_插入修改删除增量数据实时同步_通过分页解决变更记录过大问题_01----大数据之Nifi工作笔记0053
查看>>
NIFI1.21.0通过Postgresql11的CDC逻辑复制槽实现_指定表或全表增量同步_实现指定整库同步_或指定数据表同步配置_04---大数据之Nifi工作笔记0056
查看>>
NIFI1.23.2_最新版_性能优化通用_技巧积累_使用NIFI表达式过滤表_随时更新---大数据之Nifi工作笔记0063
查看>>
NIFI从MySql中增量同步数据_通过Mysql的binlog功能_实时同步mysql数据_根据binlog实现update数据实时同步_实际操作05---大数据之Nifi工作笔记0044
查看>>