返回顶部
首页 > 资讯 > 后端开发 > Python >Python赋值逻辑的实现
  • 309
分享到

Python赋值逻辑的实现

Python赋值逻辑Python赋值 2023-02-22 09:02:31 309人浏览 泡泡鱼

Python 官方文档:入门教程 => 点击学习

摘要

目录摘要:第一章 引例第二章 python 的“反直觉”第三章 回答第一章的问题摘要: 如果你学过 C 语言,那么当你初见 Python 时可能会觉得 Pyt

摘要:

如果你学过 C 语言,那么当你初见 Python 时可能会觉得 Python 的赋值方式略有诡异:好像差不多,但又好像哪里有点不太对劲。

本文比较并解释了这种赋值逻辑上的差异。回答了为什么需要这种赋值逻辑以及如何使用这种赋值逻辑的问题。

当然,即使未学过 C 语言,也可通过本文更好地了解 Python 的赋值逻辑——这种赋值逻辑影响着 Python 的方方面面,从而可以让你更好地理解和编写 Python 程序。

第一章 引例

先来看一组似乎矛盾的代码:

# 代码 1
 
>>> a = 3
>>> b = a
>>> b = 5
>>> a
3

这看上去似乎很好理解。第二步中, a 只是把值复制给 b,然后 b 又被更新为 5a 和 b 是两个独立的变量,那么 a 的值当然不会受到影响。

真的是这样吗?

再来看一段代码:

# 代码 2
 
>>> a = [1, 2, 3]
>>> b = a
>>> b[0] = 1024
>>> a
[1024, 2, 3]

第二步中,a 只是复制把列表复制给 b,然后更新 b[0] 的值,最后输出 a,可是 a 竟然也被改变了。

按照代码 1 的逻辑(即变量之间独立),代码 2 的中的 a 不应该受到影响。

为什么出现了这样的差异?

第二章 Python 的“反直觉”

先不解释上面那个“看似矛盾”的问题。

先来看看另一组简单的 Python 代码在内存中是什么样子的:

# 代码 3
 
b = 3
b = b + 5

它在内存中的操作示意图是这样的:

然而,从代码的的字面意思上看,“把 3 赋给 b,把 b 加 5 之后再赋给 b。”

也就是把代码看成这个样子:

b ← 3b ← b + 5

所以下面这张在内存中的操作图可能更符合我们的直觉:

也即 b + 5 的值又写回到 b 中。典型的 C 程序就是这样的。为变量 b 分配一个 int 型的内存单元,然后将整数 3 存放在该内存单元中。b 就代表了该块内存空间,不再移动,可以更新 b 的值,但 b 在内存中的地址就不再变化了。所以我们说 b = b + 5,就等于 b ← b + 5,把 b 的值加 5 之后还依然放入 b 中。 变量 b 和它所在内存空间紧紧绑定在一起,人形合一。

而再看看上面 Python 中的内存示意图,b + 5 得到了一个新值,然后令 b 指向了这个新值。换句话说,它做的是事情是这样的:

b → 3
b → b + 5

先令 b 指向 3,再令 b 指向 b + 5 这个新值。

C 程序更新的是内存单元中存放的值,而 Python 更新的是变量的指向。
C 程序中变量保存了一个值,而 Python 中的变量指向一个值。

如果说 C 程序是通过操纵内存地址而间接操作数据(每个变量固定对应一个内存地址,所以说操纵变量就是操纵内存地址),数据处于被动地位,那么 Python 则是直接操纵数据,数据处于主动地位,变量只是作为一种引用关系而存在,而不再拥有存储功能。

在 Python 中,每一个数据都会占用一个内存空间,如 b + 5 这个新的数据也占用了一个全新的内存空间。

Python 的这种操作让数据成为主体,数据与数据之间直接进行交互。

而数据在 Python 中被称为对象 (Object)

这句话并不太严谨。不过在这个简单的例子中是成立的。

一个整数 3 是一个 int 型对象,一个 'hello' 是一个字符串对象,一个 [1, 2, 3] 是一个列表对象。

Python 把一切数据都看成「对象」。它为每一个对象分配一个内存空间。 一个对象被创建后,它的 id 就不再发生变化。

id 是 identity 的缩写。意为“身份;标识”。
在 Python 中,可以使用 id(),来获得一个对象的 id,可以看作是该对象在内存中的地址。

一个对象被创建后,它不能被直接销毁。因此,在上个例子中,变量 b 首先指向了对象 3,然后继续执行 b + 5b + 5 产生了一个新的对象 8,由于对象 3 不能被销毁,则令 b 指向新的对象 8,而不是用对象 8 去覆盖对象 3。在代码执行完成后,内存中依然有对象 3,也有对象 8,变量 b 指向了对象 8

如果没有变量指向对象 3(即无法引用它了),Python 会使用垃圾回收算法来决定是否回收它(这是自动的,不需要程序编写者操心)。

一个旧的对象不能被覆盖,因旧的对象交互而新产生的数据会放在新的对象中。也就是说每个对象是一个独立的个体,每个对象都有自己的“主权”。因此,两个对象的交互可以产生一个新的对象,而不会对原对象产生影响。在大型程序中,各个对象之间的交互错综复杂,这种独立性则使得这些交互足够安全

C 程序为每个变量都分配一个了固定的内存地址,这保证了 C 变量之间的独立性。

C 语言是变量(也即内存地址)之间的交互,Python 是对象(数据)之间的交互。这是两种不同的交互方式。

那么,Python 这种数据之间直接进行交互的好处体现在哪里?

很遗憾,这并不是本文所要讨论的内容,该部分属于面向对象设计的核心内容。本文只是对 Python 的这种交互方式与 C 语言的交互方式做了一些比较,以区分两者在逻辑与物理上的差异所在。

相信这种逻辑会帮助你更好地编写 Python 程序,并且帮助你在日后更加深入地理解面向对象的程序设计。

本章补充:
Python 的赋值更改的是变量的指向关系,因此,对于 Python,从前向后阅读一个赋值表达式会更加容易理解。

// C 语言
b ← b + 5	// 把 b+5 的值赋给 b
# Python
b → b + 5	# 令 b 指向 b + 5

第三章 回答第一章的问题

先看代码 1:

# 代码 1
 
>>> a = 3
>>> b = a
>>> b = 5
>>> a
3

Python 中所有的数据都是对象,数字类型也不例外。3 是一个 int 类型的对象,5 也是一个 int 型的对象。
第一行,a 指向对象 3
第二行,令 b 也指向 a 所指向的对象 3
第三行,因为对象不可被覆盖(销毁),令 b 指向新对象 5,则只剩下 a 指向对象 3
第四行,输出 a,得到 3

在内存中的操作示意图 (Python):

这与第一章中的解释完全不同,第一章中的解释是用 C 语言解释的:

这是两种完全不一样的机制。

Python 中 b 首先指向了对象 3,然而因为对象之间的独立性,一个对象不能去覆盖另一个对象,则令 b 指向对象 5,而不是将对象 3 在内存中替换为对象 5

再来看代码 2:

# 代码 2
 
>>> a = [1, 2, 3]
>>> b = a
>>> b[0] = 1024
>>> a
[1024, 2, 3]

第一行,令 a 指向一个列表 [1, 2, 3]
第二行,令 b 也指向 a 所指向的列表;
第三行,令 b[0] = 10241024 虽然是一个对象,但它并没有试图覆盖b所指向的对象,而是对该对象的第一个元素进行修改。修改,而不是覆盖,所以它可以原对象进行操作,而不是令 b 指向修改后的对象。
所在第四行输出的 a 所指向的列表也发生了变化。

在内存中的操作示意图 (Python):

这种对象的值可以修改的对象被称为可变对象 (immutable object)。常见的列表、字典为可变对象。

因为它的值可以被修改,因此如果有多个变量指向该列表:

a = [1, 2, 3]
b = a
c = a
d = a
...

那么使用 b, c, d, ... 的任何一个变量都能访问该对象并修改其中的内容。这种特性常常被我们用于函数的参数传递,如果函数的参数是可变对象,那么函数可以对“实参”中的内容进行修改:

>>> a = [1, 2, 3]
>>> def change(t):
		t[0] = 1024
 
>>> change(a)
>>> a
[1024, 2, 3]
>>>  

调用函数 change 时,令 t 也指向了 a 所指向的列表,然后使用 t 更改了列表中的第一个元素,更改,而不是覆盖,因此对 t 所指向的对象的更改也改变了“实参” a 所指向的对象。而 C 语言则因为实参到形参是值传递,则无法改变实参的内容(虽然借助指针可以实现,但这里只说一般情况下)。

但在函数以外的区域,我们要尽量避免这样使用,这很容易导致出错(当然,有时候会很有用,这取决于你的程序)。比如,在多人协作编程时,如果甲不小心修改了某可变对象,那么乙、丙、丁等用到该对象的人都会受到影响。

而对于不可变对象 (immutable object),即其值无法更改的对象,传入函数时则不会影响“实参”的值:

>>> a = 5
>>> def add(n):
		n = n + 2
 
>>> add(a)
>>> a
5

调用函数 add 时,令 n 也指向了 a 所指向的对象 5, 再执行 n = n + 2n 所指向的对象 5 与对象 2 相加得到了一个新的对象 7由于一个对象不能覆盖另一个对象,则 n 指向新的对象 7,而没有改变原对象。因此 a 的值未发生变化。虽然与 C 程序的结果一致,但与 C 程序的机制完全不同,C 程序之所以没改变 a,是因为调用函数时只发生了值传递,即只把 a 的值复制给了 n

不要混淆这两种赋值逻辑,它们有着完全不同的物理实现方式。

不同的思维逻辑会导致不同的编写逻辑。尽管这两种逻辑在很多情况下的结果是一致的,但并不能就简单地认为它们是一致的。否则在一些小的细节方面出了错误,就会难以理解。只能死记硬背,把一些东西当作 Python 的特例来记,虽然「唯手熟尔」也可以让你走得很远,但思维正确时,不仅可以走得更远,也会走得更加轻松。

比如,当你的思维清晰时,以下问题的答案自然也就水落石出了:

  • 为什么列表的方法的返回值大多是 None
  • 为什么字符串的方法的返回值大多是一个新的对象?
  • 为什么 Python 中没有自增/自减运算符?
  • 为什么有的可变对象传入函数之后,却不能被函数修改“实参”的值?
    (比如将上面的 change 函数的主体改成 t = t[1:]。调用函数之后,a 所指向的对象并没有发生改变。)
  • ……

这些内容与本文主题不大相关,所以不再列出答案。

有趣的补充:

1. 数字是一个天然的不可变对象(immutable object)。
对于 n = n + 2,有人可能会说,为什么不能把它看成像列表那样的修改,修改后 n 依然指向的是原对象,这样的话执行 add(a) 之后,a 就会变成 7 了,可为什么不是这样?
因为每一个数字都是一个单个的对象,而对象不能覆盖对象。所以该句实际上是: a 指向的对象加上对象 2,产生了一个新的对象,然后令 a 指向了新对象 a + 2
因此,数字类型并不存在修改这一说,它是一个天然的不可变对象。

2. 为什么 Python 中没有自增(++)、自减(--)运算符?
自增或自减运算符,在 C 语言中很常用,简洁实用。但在 Python 中却一定不会有。上节说到,数字是天然的不可变对象,所谓自增就是自身增加,所以它无法自增。它只能从一个对象指向下一个对象。可以这样写 a += 1
3. 既然 Python 更改的只是引用关系,那么如何复制一个列表?

# 答案:
## 1. 使用 list 的 copy 方法
b = a.copy()
## 2. 使用 slice 操作
b = a[:]	# slice 操作返回一个新的对象
# 答案:
## 1. 使用 list 的 copy 方法
b = a.copy()
## 2. 使用 slice 操作
b = a[:]	# slice 操作返回一个新的对象

到此这篇关于Python赋值逻辑的实现的文章就介绍到这了,更多相关Python赋值逻辑内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Python赋值逻辑的实现

本文链接: https://lsjlt.com/news/197062.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Python赋值逻辑的实现
    目录摘要:第一章 引例第二章 Python 的“反直觉”第三章 回答第一章的问题摘要: 如果你学过 C 语言,那么当你初见 Python 时可能会觉得 Pyt...
    99+
    2023-02-22
    Python赋值逻辑 Python赋值
  • Python赋值逻辑如何实现
    本篇内容介绍了“Python赋值逻辑如何实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!第一章 引例先来看一组似乎矛盾的代码:# ...
    99+
    2023-07-05
  • python逻辑值检测的实现方法
    小编给大家分享一下python逻辑值检测的实现方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!Python主要用来做什么Python主要应用于:1、Web开发;...
    99+
    2023-06-14
  • python逻辑值检测怎么实现
    本篇内容介绍了“python逻辑值检测怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!概念在python中,任何物体都可以检测逻辑值。...
    99+
    2023-06-30
  • Opportunity的chance of success的赋值逻辑是什么
    这篇文章主要介绍“Opportunity的chance of success的赋值逻辑是什么”,在日常操作中,相信很多人在Opportunity的chance of success的赋值逻辑是什么问题上存在疑惑,小编查阅了各式资料,整理出简...
    99+
    2023-06-04
  • JavaScript中有哪些逻辑赋值运算符
    这期内容当中小编将会给大家带来有关JavaScript中有哪些逻辑赋值运算符,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。JavaScript是什么JS是JavaScript的简称,它是一种直译式的脚本语...
    99+
    2023-06-14
  • js如何使用逻辑赋值运算符
    这篇文章主要介绍js如何使用逻辑赋值运算符,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!逻辑赋值运算符逻辑赋值运算符是由逻辑运算符&&、||、和赋值运算符=组合而成。const a ...
    99+
    2023-06-27
  • JavaScript 中如何使用逻辑赋值运算符
    本篇文章为大家展示了JavaScript 中如何使用逻辑赋值运算符,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。无条件 vs 有条件数学运算符,例如 +是无条件的。...
    99+
    2024-04-02
  • Python实现逻辑回归(Logistic Regression)
    💥 项目专栏:【Python实现经典机器学习算法】附代码+原理介绍 文章目录 前言一、基于原生Python实现逻辑回归算法二、逻辑回归模型的算法原理三、算法实现3.1 ...
    99+
    2023-08-31
    python 逻辑回归 机器学习 人工智能 sklearn 原力计划
  • python实现逻辑回归的方法示例
    本文实现的原理很简单,优化方法是用的梯度下降。后面有测试结果。 先来看看实现的示例代码: # coding=utf-8 from math import exp import matplotlib.p...
    99+
    2022-06-04
    示例 逻辑 方法
  • pytorch实现逻辑回归
    本文实例为大家分享了pytorch实现逻辑回归的具体代码,供大家参考,具体内容如下 一、pytorch实现逻辑回归 逻辑回归是非常经典的分类算法,是用于分类任务,如垃圾分类任务,情感...
    99+
    2024-04-02
  • python的numpy模块实现逻辑回归模型
    使用python的numpy模块实现逻辑回归模型的代码,供大家参考,具体内容如下 使用了numpy模块,pandas模块,matplotlib模块 1.初始化参数 def initi...
    99+
    2024-04-02
  • python怎么实现多重赋值
    这篇文章主要为大家展示了“python怎么实现多重赋值”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“python怎么实现多重赋值”这篇文章吧。多重赋值你是否曾想...
    99+
    2024-04-02
  • python如何实现多重赋值
    小编给大家分享一下python如何实现多重赋值,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!多重赋值在给几个变量赋值时,可以使用多重赋值。通过同样的习惯用法,可以交换同一列表中的两个变量或两个元素。这一特征与之后要介绍的元...
    99+
    2023-06-27
  • 浅谈Python实现opencv之图片色素的数值运算和逻辑运算
    数值运算 代码: # -*- coding=GBK -*- import cv2 as cv # 数值运算:加减乘除 def shu_image(src11, src22...
    99+
    2024-04-02
  • PostgreSQL的逻辑复制怎么实现
    在 PostgreSQL 中,逻辑复制是通过订阅和发布的方式实现的。下面是实现逻辑复制的步骤: 创建发布者(publisher):...
    99+
    2024-04-02
  • python实现梯度下降求解逻辑回归
    本文实例为大家分享了python实现梯度下降求解逻辑回归的具体代码,供大家参考,具体内容如下 对比线性回归理解逻辑回归,主要包含回归函数,似然函数,梯度下降求解及代码实现 线性回归 ...
    99+
    2024-04-02
  • Logistic回归(逻辑回归)及python代码实现
    文章目录 Logistic(Logistic Regression,LR)回归原理讲解参数计算 python代码实现生成数据集不使用其他库实现定义激活函数(标准Logistic函数即Sig...
    99+
    2023-10-24
    回归 逻辑回归 人工智能 python 机器学习 分类
  • python用列表实现多重赋值
    变量的值和列表中的个数需要相等。 否则会出错误。 ...
    99+
    2023-01-31
    赋值 列表 python
  • python赋值和交换怎么实现
    本文小编为大家详细介绍“python赋值和交换怎么实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“python赋值和交换怎么实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。赋值原理在赋值的右边形成一个新的元...
    99+
    2023-06-30
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作