首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

【Python基础】Python数据类型大战:可变与不可变,谁主沉浮?

第1章 Python类型系统概览

1.1 Python中的数据类型简介

在Python里,数据类型扮演着核心角色,它们是构建程序逻辑的基础砖石。Python的数据类型分为两大阵营:基本数据类型和复合数据类型。这两种类型不仅决定了变量所存储数据的特点,而且直接影响到程序运行时的内存分配、性能以及代码的安全性和可读性。

1.1.1 基本数据类型

Python的基本数据类型包括整型(int)、浮点型(float)、字符串(str)、布尔型(bool)以及NoneType。这些类型在创建后其值不可改变:

?整型:如age = 25,表示一个固定的整数值。

?浮点型:如pi = 3.14,用于存储带有小数部分的数值。

?字符串:如name = "Alice",一旦创建,字符串的内容不可直接更改,尽管看起来有“修改”方法,但实际上是在原对象基础上生成新的字符串对象。

?布尔型:True或False,常用于条件判断。

?NoneType:None代表无值或空值,通常用来初始化变量或作为函数没有正常返回值时的默认返回值。

实例演示

#?基本数据类型示例

age?=?25??#?整型

pi_val?=?3.14??#?浮点型

name?=?"Alice"??#?字符串

is_active?=?True??#?布尔型

no_value?=?None??#?NoneType1.1.2 复合数据类型

复合数据类型则能够组合多个值形成更复杂的数据结构。主要包括列表(list)、元组(tuple)、字典(dict)和集合(set):

?列表:有序且可变的元素序列,例如students = ["Alice", "Bob", "Charlie"]。

?元组:有序但不可变的元素序列,例如coordinates = (40.7128, -74.0060),常用于存放固定不变的数据集。

?字典:无序且可变的键值对集合,例如person = {"name": "Alice", "age": 25}。

?集合:无序且唯一元素的集合,例如unique_numbers = {1, 2, 3, 3, 4},自动去除重复元素。

实例演示

#?复合数据类型示例

students?=?["Alice",?"Bob",?"Charlie"]??#?列表

coordinates?=?(40.7128,?-74.0060)??#?元组

person_info?=?{"name":?"Alice",?"age":?25}??#?字典

unique_nums?=?{1,?2,?3,?3,?4}??#?集合,自动去重后为{1,?2,?3,?4}第2章 可变类型详解2.1 可变类型的定义与特性

可变类型是Python中一类允许其内容在创建后发生改变的数据类型。理解并熟练运用这些类型,是实现动态数据管理、高效资源利用的关键。

2.1.1 列表(List)

列表是一种灵活的有序序列,允许添加、删除、替换元素。

2.1.1.1 列表的创建与访问

列表通过方括号[]创建,元素间用逗号分隔。访问元素使用索引,负数索引从后向前计数。

实例演示

#?创建列表

students?=?["Alice",?"Bob",?"Charlie"]

#?访问元素

first_student?=?students[0]??#?"Alice"

last_student?=?students[-1]??#?"Charlie"2.1.1.2 列表的增删改操作

列表提供了丰富的内置方法来改变其内容:

?:append()、extend()、insert()

?:remove()、pop()、del关键字、clear()

?:直接赋值或使用list[index] = new_value

实例演示

#?增加元素

students.append("David")??#?["Alice",?"Bob",?"Charlie",?"David"]

students.extend(["Eve",?"Frank"])??#?["Alice",?"Bob",?"Charlie",?"David",?"Eve",?"Frank"]

students.insert(2,?"Diana")??#?["Alice",?"Bob",?"Diana",?"Charlie",?"David",?"Eve",?"Frank"]

#?删除元素

students.remove("Bob")??#?移除首个匹配项

popped_student?=?students.pop()??#?删除并返回最后一个元素

del?students[3]??#?删除指定位置的元素

students.clear()??#?清空整个列表

#?修改元素

students[1]?=?"Bobby"??#?替换指定位置的元素2.1.2 字典(Dictionary)

字典是一种无序的键值对集合,键必须是唯一的,且不可变。

2.1.2.1 字典的创建与访问

字典使用花括号{}创建,键值对之间用逗号分隔,键与值之间用冒号:分隔。访问元素使用键。

实例演示

#?创建字典

student_grades?=?{"Alice":?95,?"Bob":?8?,?"Charlie":?.jpg}

#?访问元素

alice_grade?=?student_grades["Alice"]??#?95

charlie_grade?=?student_grades.get("Charlie")??#?.jpg?或?None(如果不存在)

#?使用get()避免KeyError

default_grade?=?student_grades.get("David",?75)??#?当键不存在时返回默认值2.1.2.2 字典的增删改操作

字典提供了相应的方法来操作键值对:

?:直接赋值或使用update()

?:pop()、popitem()、del关键字、clear()

?:直接赋值

实例演示

#?增加键值对

student_grades["David"]?=?.png??#?直接赋值

student_grades.update({"Eve":?99,?"Frank":?Ⅰ})??#?一次性添加多个键值对

#?删除键值对

removed_grade?=?student_grades.pop("Alice")??#?删除并返回键对应的值

key_value_pair?=?student_grades.popitem()??#?删除并返回一个随机键值对

del?student_grades["Bob"]??#?使用del关键字删除

student_grades.clear()??#?清空字典

#?修改键值对

student_grades["David"]?=?.jpeg??#?直接赋新值覆盖旧值2.1.3 集合(Set)

集合存储无序且不重复的元素。

2.1.3.1 集合的创建与访问

集合使用大括号{}创建,元素间用逗号分隔。由于无序,访问元素通常通过迭代而非索引。

实例演示

#?创建集合

unique_numbers?=?{1,?2,?3,?3,?4}??#?自动去重后为{1,?2,?3,?4}

#?访问元素(迭代访问)

for?num?in?unique_numbers:

print(num)2.1.3.2 集合的增删操作

集合提供了方法进行增删操作:

?:add()、update()

?:remove()、discard()、pop()、clear()

实例演示

#?增加元素

unique_numbers.add(5)??#?单个元素添加

unique_numbers.update([6,?7,?8])??#?一次性添加多个元素

#?删除元素

unique_numbers.remove(?)??#?如果元素不存在会引发KeyError

unique_numbers.discard(?)??#?不存在时不引发异常

popped_number?=?unique_numbers.pop()??#?删除并返回一个随机元素

unique_numbers.clear()??#?清空集合2.2 可变类型的应用场景2.2.1 动态数据结构管理

可变类型适用于需要频繁调整数据结构的场景,如实时统计、动态列表维护等。例如,记录用户购物车商品变化:

cart?=?[]

add_item(cart,?"apple")??#?cart:?["apple"]

add_item(cart,?"banana")??#?cart:?["apple",?"banana"]

remove_item(cart,?"apple")??#?cart:?["banana"]2.2.2 数据共享与同步问题

在多线程或多进程环境中,可变类型可能引发数据竞争和同步问题。使用锁或其他同步机制确保安全访问:

import?threading

data?=?[]

lock?=?threading.Lock()

def?thread_func():

with?lock:

data.append(threading.current_thread().name)

threads?=?[threading.Thread(target=thread_func)?for?_?in?range(5)]

for?t?in?threads:

t.start()

for?t?in?threads:

t.join()

print(data)??#?输出各线程名,无数据竞争2.3 可变类型的内存管理与性能考量2.3.1 引用计数与垃圾回收

Python采用引用计数机制管理内存。当可变类型的引用计数降为零时,其所占内存会被自动释放。

2.3.2 多重引用下的可变类型操作

当一个可变类型被多个引用共享时,对其的修改会影响到所有引用。理解这一点有助于避免意外后果:

list1?=?[1,?2,?3]

list2?=?list1??#?多重引用

list1.append(4)??#?list1:?[1,?2,?3,?4],?list2:?[1,?2,?3,?4]第3章 不可变类型解析3.1 不可变类型的定义与特性

不可变类型在Python中指的是那些一旦创建后就不能改变其内容的对象。这种特性带来了诸多优点,比如安全性更高、易于缓存和优化,同时也利于在并发环境下使用。

3.1.1 字符串(String)3.1.1.1 字符串的创建与访问

字符串是Python中最常见的不可变类型之一。创建字符串时,可以使用单引号'或双引号"包裹字符序列。

text?=?"Hello,?World!"??#?创建字符串

first_char?=?text[0]??#?访问第一个字符

请注意,尽管字符串提供了诸如replace()、upper()等看似“修改”字符串的方法,实际上这些操作并不会改变原始字符串,而是返回一个新的字符串。

3.1.1.2 字符串的“修改”方法与实际效果

original_text?=?"hello"

modified_text?=?original_text.upper()??#?返回"HELLO",但original_text未变3.1.2 整数(Integer)与浮点数(Float)3.1.2.1 数值类型的创建与运算

在Python中,整数和浮点数也是不可变类型。创建时直接赋值即可,并可通过算术运算符进行常规计算。

number1?=?42??#?整数

number2?=?3.14??#?浮点数

sum_result?=?number1?+?number2??#?计算两个数之和

Python还针对一些较小的整数进行了优化,它们会被缓存以提高性能和减少内存开销。

3.1.2.2 数值类型的缓存机制

a?=?1000

b?=?1000

print(a?is?b)??#?输出True,说明Python缓存了较小整数值

c?=?2?**?53?+?1

d?=?2?**?53?+?1

print(c?is?d)??#?输出False,超过缓存范围的整数不会被缓存3.1.3 元组(Tuple)3.1.3.1 元组的创建与访问

元组是另一种不可变类型,它由一组逗号分隔的值构成,通常用圆括号包围起来。

coordinates?=?(40.7128,?-74.0060)??#?创建元组

latitude?=?coordinates[0]??#?访问元组的第一个元素

与字符串类似,元组不支持直接修改已有元素,尝试这样做将引发错误。

3.1.3.2 元组的解包与替换操作

虽然元组内容不能修改,但可以通过解包、拼接等方式创建新的元组。

old_tuple?=?(1,?2,?3)

new_tuple?=?old_tuple?+?(4,?5)??#?新元组包含原有元素及新增元素

x,?y,?z?=?old_tuple??#?解包元组到单独变量3.2 不可变类型的应用优势3.2.1 作为字典键与集合成员

不可变类型因其不变性可以作为字典的键或集合的成员,因为它们保证了哈希值的稳定性。

my_dict?=?{("apple",?1):?"red",?("banana",?2):?"yellow"}

my_set?=?{(1,?2),?(3,?4)}3.2.2 并发安全与函数式编程

在多线程或函数式编程中,不可变类型能有效防止数据竞争,确保每次操作都是幂等的,提高了代码的可靠性和可预测性。

3.3 不可变类型的内存效率与不变性原理

3.3.1 内存共享与引用透明性

由于不可变对象在创建后无法改变,因此在多个地方引用同一不可变对象时,无需复制其内容,仅需共享相同的内存地址,节省了内存空间。

3.3.2 编译优化与哈希一致性

编译器和解释器能够根据不可变对象的不变性提前进行优化,如预计算哈希值以提高查找速度,这在大量数据处理和算法中尤为重要。

第4章 可变类型与不可变类型比较

4.1 概念差异与操作特性对比

4.1.1 修改行为与内存分配

可变类型和不可变类型最显著的区别在于它们对内容修改的处理方式。可变类型(如列表、字典、集合)允许直接修改其内部状态,而不可变类型(如字符串、整数、浮点数、元组)在创建后其内容始终保持不变。这一差异直接影响内存分配和数据处理的效率。

可变类型修改示例

mutable_list?=?[1,?2,?3]

mutable_list.append(4)??#?修改原列表,不创建新对象

mutable_dict?=?{"a":?1}

mutable_dict["b"]?=?2??#?修改原字典,不创建新对象

不可变类型“修改”示例

immutable_str?=?"hello"

new_str?=?immutable_str.replace("l",?"W")??#?创建新字符串,原字符串不受影响

immutable_tuple?=?(1,?2,?3)

extended_tuple?=?immutable_tuple?+?(4,)??#?创建新元组,原元组保持不变4.1.2 适用场景与编程范式

可变类型适用于需要频繁更新数据结构的场景,如实时统计、动态列表维护等。而不可变类型在需要保证数据一致性、实现函数式编程、提升并发安全性等方面更具优势。

4.2 实际应用中的选择策略

4.2.1 性能考量与资源管理

在性能敏感的应用中,可变类型修改操作通常更快,因为它们直接在现有内存空间上操作,避免了新对象的创建。然而,大量创建短生命周期的可变对象可能导致内存碎片化,增加垃圾回收负担。不可变类型虽在修改时可能产生新对象,但其内存共享特性有利于减少整体内存消耗,特别是在多层嵌套数据结构中。

性能对比示例

import?timeit

setup?=?"""

mutable_list?=?[0]?*?1000

immutable_list?=?tuple(0?for?_?in?range(1000))

"""

mutable_test?=?"""

for?i?in?range(len(mutable_list)):

mutable_list[i]?+=?1

"""

immutable_test?=?"""

new_list?=?tuple(value?+?1?for?value?in?immutable_list)

"""

print(timeit.timeit(mutable_test,?setup=setup,?number=1000))??#?可变类型修改时间

print(timeit.timeit(immutable_test,?setup=setup,?number=1000))??#?不可变类型“修改”时间4.2.2 数据安全与并发控制

在多线程或异步编程环境中,可变类型可能导致竞态条件和数据不一致。使用不可变类型能有效避免这些问题,因为它们天然具备线程安全属性,无需额外同步机制。

并发安全示例

import?threading

mutable_data?=?[0]

def?increment(data):

data[0]?+=?1??#?未加锁,存在竞态条件

threads?=?[threading.Thread(target=increment,?args=(mutable_data,))?for?_?in?range(10)]

for?t?in?threads:

t.start()

for?t?in?threads:

t.join()

print(mutable_data[0])??#?结果不确定,可能不是预期的10

综上所述,可变类型与不可变类型的选择应根据具体应用场景、性能需求、数据安全性和代码风格等因素综合考虑。理解它们各自的优势与局限性,有助于编写出高效、稳定且易于维护的Python代码。

第5章 高级主题:混合使用可变类型与不可变类型

5.1 封装与函数返回值

5.1.1 使用不可变类型封装可变数据

在实际编程中,我们常常结合使用可变类型与不可变类型,以达到平衡灵活性与安全性的目的。例如,在处理可变数据时,可以通过不可变类型封装结果,确保数据在传递过程中的完整性。

#?封装可变数据到不可变容器

inventory?=?{"apples":?10,?"oranges":?20}

frozen_inventory?=?frozenset(inventory.items())??#?使用frozenset封装库存信息

#?在函数中接收不可变数据,保护原数据不被篡改

def?calculate_stock_delta(frozen_stock,?new_stock):

delta?=?{}

for?old_item,?old_count?in?frozen_stock:

if?old_item?in?new_stock:

delta[old_item]?=?new_stock[old_item]?-?old_count

return?delta

new_inventory?=?{"apples":?8,?"oranges":?30,?"bananas":?15}

stock_changes?=?calculate_stock_delta(frozen_inventory,?new_inventory)5.1.2 函数返回值的可变性考虑

函数返回值的可变性对代码质量和潜在风险有着重要影响。一般情况下,尽量避免函数返回可变对象的引用,除非明确告知调用者并确保他们了解后果。为了降低意外修改的风险,可以返回不可变对象副本或者转换为不可变形式。

#?示例:返回不可变列表的副本

def?get_sorted_names(names_list):

sorted_names?=?sorted(names_list)

return?tuple(sorted_names)??#?返回元组以保证不可变

names?=?["Alice",?"Bob",?"Charlie"]

sorted_names?=?get_sorted_names(names)

#?调用者无法直接修改排序后的名字5.2 设计模式与最佳实践5.2.1 不可变数据结构在设计模式中的应用

不可变数据结构在许多设计模式中扮演关键角色,如享元模式(Flyweight Pattern),其中共享的不可变对象可以大大减少内存占用。在函数式编程中,不可变数据也是实现纯函数的重要工具。

5.2.2 利用可变类型优化特定算法

对于某些算法,尤其是涉及大量数据操作和中间结果的情况,适当使用可变类型能够提高执行效率。例如,在动态规划算法中,可以利用列表或字典作为缓存容器,逐步更新中间结果。

#?示例:斐波那契数列,使用可变列表缓存已计算结果

def?fibonacci(n,?memo={}):

if?n?<=?0:

return?0

elif?n?==?1:

return?1

elif?n?not?in?memo:

memo[n]?=?fibonacci(n?-?1)?+?fibonacci(n?-?2)

return?memo[n]

print(fibonacci(10))??#?利用可变字典memo记录递归计算的中间结果

总之,在Python编程实践中,巧妙地混合使用可变类型与不可变类型可以帮助我们构建出更健壮、高效且易于维护的软件系统。理解何时应该使用哪种类型,并学会在不同场景下切换,是Python开发者必备的一项技能。通过深入研究,我们可以发现更多关于如何有效结合两种类型的最佳实践。

第6章 Python标准库中可变类型与不可变类型的扩展模块

6.1 collections模块

Python标准库中的collections模块提供了多种高级容器类型,这些类型在功能上是对内置可变类型(如列表、字典、集合)的扩展和增强,旨在满足特定场景下的高效数据管理需求。

?namedtuple:创建具有命名字段的不可变元组子类,提供更友好的访问方式。

from?collections?import?namedtuple

Person?=?namedtuple('Person',?['name',?'age'])

bob?=?Person(name="Bob",?age=30)

print(bob.name)??#?输出?"Bob"

?deque:双端队列,支持高效地在两端添加和移除元素,适用于实现滑动窗口、缓存队列等功能。

from?collections?import?deque

window?=?deque(maxlen=3)

window.append(1)

window.append(2)

window.append(3)

print(window)??#?输出?deque([1,?2,?3])

window.append(4)??#?由于长度限制,最左边的元素(1)被移除

print(window)??#?输出?deque([2,?3,?4])

?Counter:计数器,用于跟踪元素出现次数,尤其适合处理文本统计、数据分析等问题。

from?collections?import?Counter

word_counts?=?Counter(["apple",?"banana",?"apple",?"orange"])

print(word_counts)??#?输出?Counter({'apple':?2,?'banana':?1,?'orange':?1})

?defaultdict:带默认值的字典,当访问不存在的键时,会自动使用预设的工厂函数创建默认值。

from?collections?import?defaultdict

word_lengths?=?defaultdict(int)

word_lengths["apple"]?=?5

word_lengths["banana"]?=?6

print(word_lengths["pear"])??#?输出?0,因为pear不在字典中,自动使用默认值int()初始化为06.2 itertools模块

itertools模块提供了大量的高效、内存友好的迭代器生成器函数,这些函数常用于处理序列数据,尤其适用于不可变类型(如字符串、元组)的迭代操作。

?count:生成无限递增的整数序列。

import?itertools

counter?=?itertools.count(start=1,?step=2)

for?i?in?range(5):

print(next(counter))??#?输出?1,?3,?5,?7,?9

?cycle:无限循环地迭代给定序列。

import?itertools

colors?=?["red",?"green",?"blue"]

color_cycle?=?itertools.cycle(colors)

for?_?in?range(.png):

print(next(color_cycle))??#?输出?"red",?"green",?"blue",?"red",?...

?groupby:对连续重复的键值进行分组。

import?itertools

words?=?["apple",?"apple",?"banana",?"banana",?"apple"]

grouped_words?=?itertools.groupby(words)

for?key,?group?in?grouped_words:

print(key,?list(group))??#?输出?"apple",?["apple",?"apple"],?"banana",?["banana",?"banana"],?"apple",?["apple"]

?accumulate:计算序列中累积的和或应用其他二元操作。

import?itertools

numbers?=?[1,?2,?3,?4,?5]

cumulative_sum?=?itertools.accumulate(numbers)

print(list(cumulative_sum))??#?输出?[1,?3,?6,?10,?15]

这些扩展模块提供了强大的工具,使得Python开发者能够更灵活、高效地处理可变类型与不可变类型数据,进一步提升代码的性能和可读性。通过深入学习和实践,开发者能够更好地利用Python标准库的强大功能,解决各种复杂的数据处理问题。

第7章 结论

Python的类型系统深刻影响着程序设计的各个方面。可变类型如列表、字典和集合,凭借其灵活性成为构建动态数据结构、处理共享数据的理想工具,但同时也需关注内存管理、引用计数与垃圾回收带来的挑战,以及在多线程环境下的同步问题。不可变类型如字符串、数值类型和元组,则因内存效率高、并发安全和引用透明性而在函数式编程、哈希键和数据一致性方面表现出色。在实际编程中,二者互补互融,通过collections和itertools等模块,开发者能够结合使用两者优化数据结构和算法设计。

深入理解可变类型与不可变类型的特性与差异,是Python开发者进阶的关键步骤。它不仅有助于提高代码质量,确保资源的有效利用,还能指导我们在性能考量、数据安全及并发控制等方面做出明智决策。同时,借助不可变类型强化设计模式,以及在适当场景下利用可变类型优化算法,都能极大提升应用程序的效能和可靠性。

  • 发表于:
  • 原文链接https://page.om.qq.com/page/OS4Pgs7GZ24xf8WYFq8aVhBQ0
  • 腾讯「腾讯云开发者社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券
http://www.vxiaotou.com