你做的 9 件事表明你不是专业的 Python 开发人员

表明,不是,专业,python,开发人员 · 浏览次数 : 130

小编点评

**内容简介** 该文章介绍了 Python 中的 is 和 == 运算符的区别,以及如何使用 Print 语句代替日志记录模块。文章还介绍了如何使用 import * 在命名模块中导入函数和类,以及如何避免使用 is 和 == 运算符。 **内容分析** * **is 和 == 运算符的区别** * is运算符返回 True 如果变量是对象, False 如果变量是值。 * ==运算符返回 True 如果变量值相等, False 如果变量值不相等。 * **使用 Print 语句代替日志记录模块** * Print 语句可以打印变量的值,而日志记录模块可以打印变量的值以及日志消息。 * **使用 import * 在命名模块中导入函数和类** * import * 会导入所有模块中定义的函数和类,并将它们导入到您的代码中。 * **避免使用 is 和 == 运算符** * 使用 is 和 == 运算符可以导致代码错误,因为它们无法确保变量是对象或值。 * **使用 def 函数定义变量** * def 函数可以定义变量,并使用 return 语句返回变量的值。 * **使用 import * 在命名模块中导入函数和类** * import * 会导入所有模块中定义的函数和类,并将它们导入到您的代码中。 **总结** 文章介绍了 Python 中的 is 和 == 运算符的区别,以及如何使用 Print 语句代替日志记录模块。文章还介绍了如何使用 import * 在命名模块中导入函数和类,以及如何避免使用 is 和 == 运算符。

正文

本文转载自国外论坛 medium,原文地址:
https://medium.com/navan-tech/7-java-features-you-might-not-have-heard-of-adee8166d942,由博主简译后给大家带来!

Show me your code and I will tell you who you are.

This article will fix the bad habits you have stuck to over the years or brought from other programming languages.

是的,老外就这么吊,文章开头就是"给我看看你的代码,我来告诉你,你有几斤几两!"
紧接着,老外告诉你这篇文章为什么这么吊,意思是这篇文章可以纠正你多年以来从其他语言坚持而来的坏习惯。。。虽然博主确实编了几年程序。

一、手动格拼接字符串

大多数时候,Python 初学者在组合两个字符串时会使用 + 号。

>>> name = "Ridwan"
>>> age = "22"
>>> print("My Name is " + name + " and I am " + age + " years old")
My Name is Ridwan and I am 22 years old

不要使用 + 号,而应使用 f 字符串,这样可以使您的代码可读、简洁且不易出错。python3.6+开始支持 f 格式字符串

>>> print(f"My Name is {name} and I am {age} years old")
My Name is Ridwan and I am 22 years old

二、使用默认可变参数

在 Python 中,只要您将可变值作为参数传递给函数,默认参数就会在函数被调用时发生变化。这些可变参数通常是列表或字典。

如下:

>>> def append(n, l=[]):
...     l.append(n)
...     return l
...

可以看到 append 函数的第二个参数 l 是一个可变参数,只要您使用值为 n 调用该函数,它就会更改默认值 l。

// 第一次调用
>>> l1 = append(0)
>>> l1
[0]

当您下次在调用 append 函数时,您将看到您使用的先前值附加到空列表参数。

// 第二次调用
>>> l2 = append(1)
>>> l2
[0, 1]

简而言之也就是说由于 l 在 append 函数中被默认初始化为一个 list,第二次调用时,l 并没有重置,导致返回时还带有第一次调用的结果。

这个问题可以通过重写代码来解决,

>>> def append(n, l = None):
...     if l is None:
...         l = []
...     l.append(n)
...     return l
...
>>> l1 = append = [0]
>>> l2 = append = [1]
>>> l1,l2
([0], [1])

现在参数 l 被设置为 None,任何时候函数被调用,即使 l 发生了变化,它也会被重新分配为 None,然后给出一个空列表的值。

三、不使用推导式

Python 推导式 为您提供了一种构建序列的简洁方式,上次我检查过,Python 支持 4 种类型的推导式;

  • 列表推导式
  • 集体推导式
  • 字典推导式
  • 生成器推导式

你可以在这里阅读更多关于他们的信息。

下面的代码将字典中的值除以 2,

>>> numbers = {}
>>> for i in range(10):
...     numbers[i] = i/2
...
>>> numbers
{0: 0.0, 1: 0.5, 2: 1.0, 3: 1.5, 4: 2.0, 5: 2.5, 6: 3.0, 7: 
3.5, 8: 4.0, 9: 4.5}

上面的代码可以写成一行,

>>> {i: i/2 for i in range(10)}
{0: 0.0, 1: 0.5, 2: 1.0, 3: 1.5, 4: 2.0, 5: 2.5, 6: 3.0, 7: 
3.5, 8: 4.0, 9: 4.5}

所以不要让自己过得不好,开始使用推导式。

推导式确实算是 python 开发一大利器,用好推导式,早下班!

四、检查 Equality 而不是 Identity

如下,

a = [1, 2, 3]
b = [1, 2, 3]

如果我让你检查这两个变量是否相同,你首先想到的是,

>>> a == b
True

问题是你需要知道 Identity 和 Equality 之间的区别。

如果检查a和b的内存地址,

>>> id(a), id(b)
(1838093945856, 1838093487488)

您可以看到尽管它们具有相同的对象,但它们都有不同的内存地址。

这就是为什么当你运行代码时,

>>> a == b
True
You get True, but when you run

>>> a is b
False

当你调用 a is b 返回 False时,a 和 b 就不是相等的了。

在运行如下代码,

>>> c = [1,2,3]
>>> d = c
>>> id(c), id(d)
(1838089019712, 1838089019712)

可以看到,c和d是相等且相同的,c中的对象也被赋值给了d。

>>> c == d
True
>>> c is d
True

这意味着 c 和 d 具有相同的值和内存地址。

因此你可以说 c 是相同的并且等于 d。

写这篇文章就是为了让你知道 is 和 == 的区别,前者是用来检查 identity ,后者是用来检查 equality 的。

所有相同的变量都相等,但并非所有相等的变量都相同。

这一段看着有点绕,说人话就是 == 比较的是连个变量的值是否相等,is 比较的是两个比变量的内存地址相等!,我相信大伙都明白哈,不然白看了这么多年八股文。

五、不使用元组解包

任何时候你在 Python 中创建一个元组 a_tuple = 1,2,3 ,它会默认进行元组打包,

>>> a_tuple = 1,2,3
>>> a_tuple
(1, 2, 3)

然后可以通过索引访问元组内元素

>>> x = a_tuple[0]
>>> y = a_tuple[1]
>>> z = a_tuple[2]
>>> print(x, y, z)
1, 2, 3

其实无需使用多行代码访问元组中的元素,您可以通过元组解包自动在一行代码中完成。

>>> x,y,z = a_tuple
>>> print(x, y, z)
1, 2, 3

元组解包也是 python 中常用的开发技巧,提升开发效率。

六、创建您自己的索引计数器变量

这个在其他编程语言中很常见,你被要求创建一个索引计数器变量,然后你输入类似的东西;

>>> a_list = [1,2,3,4,5,6,7,8,9,10]
>>> index = 0
>>> for elem in a_list:
...     print(index, elem)
...     index += 1
...
0 1
1 2
2 3
3 4
4 5
5 6
6 7
7 8
8 9
9 10

相反,使用 enumerate 函数使您的代码看起来像 Pythonic(这里指写出python风格的代码);

>>> for index, elem in enumerate(a_list):
...     print(index, elem)
...
0 1
1 2
2 3
3 4
4 5
5 6
6 7
7 8
8 9
9 10

这里批评下 Java,这么多年了,foreach 循环也拿不到索引。

七、使用 Print 语句代替日志记录模块

这在小型项目中可能无关紧要,但肯定会对大型项目有所帮助。

不要用打印语句乱扔代码,而是使用日志记录。

>>> print('This is a warning message')
This is a warning message
>>> print('This is an error message')
This is an error message
>>> print('This is a critical message')
This is a critical message

日志记录有助于向您的用户显示有用的消息,以增加对代码库中发生的事情的更多上下文和理解。

>>> import logging
>>> logging.warning('This is a warning message')
WARNING:root:This is a warning message
>>> logging.error('This is an error message')
ERROR:root:This is an error message
>>> logging.critical('This is a critical message')
CRITICAL:root:This is a critical message

这年头还有人在线上不用文件记录日志吗?有的话告诉我一声,
我肯定拿刀找他!

八、使用 import * 在命名模块中导入函数和类

这种坏习惯有时在新手中很常见。

使用 import * 导入会破坏您的命名空间,方法是将该命名模块中的所有函数和类导入您的代码,这可能会与您定义的函数或导入的其他库的函数发生冲突。

反正博主从来不用 import *,至于你用不用我不知道,但是我建议你不要用。

九、不关注 pep8

pep8就是Python官方指定的编码规范

我们大多数人都犯了这个罪,

我承认我有罪

在我被取消之前😂,我知道我在这篇文章中的一些代码可能违反了 PEP-8 的规则,但事实是痛苦的,需要被告知,遵循 PEP-8 风格和指南让其他人更容易阅读并理解你的代码。

老外的这篇文章在 medium 的邮件推荐中,可能老外知道自己这篇文章的示例代码也违反了 pep8 规范,怕被取消推荐了,我猜的,不保真。

不推荐

>>> def function():
...     x = [1,2,3]
...     y= [2,3,5]
...     z = [1, 2,3]
...
>>> def value(x = 7):
...     ...
...

推荐

>>> def function():
...     x = [1, 2, 3]
...     y = [2, 3, 5]
...     z = [1, 2, 3]
...
>>> def number(x=7):
...     ...
...

要阅读更多关于 PEP-8 风格和指南的信息,请查看这篇文章 https://realpython.com/python-pep8/。

感谢阅读,希望这篇文章值得你花时间。

公众号【waynblog】每周更新博主最新技术文章,欢迎大家关注

与你做的 9 件事表明你不是专业的 Python 开发人员相似的内容:

你做的 9 件事表明你不是专业的 Python 开发人员

本文转载自国外论坛 medium,原文地址: https://medium.com/navan-tech/7-java-features-you-might-not-have-heard-of-adee8166d942,由博主简译后给大家带来! Show me your code and I wil

springboot~封装依赖引用包jar还是pom,哪种更规范

将多个第三方包封装成一个项目后,如果你的目的是让其他开发人员可以直接引用这些依赖,一般来说有两种常见的方式: 打成JAR包:将封装好的项目编译打包成JAR文件,其他开发人员可以将这个JAR文件添加到他们的项目中,并在项目的构建工具(比如Maven)中配置该JAR作为依赖。这样做的好处是简单直接,其他

DevOps|1024程序员节怎么做?介绍下我的思路

1024,祝每个程序员小哥哥小姐姐节日快乐。 因为在研发效能部门,我支持过几次 1024 程序员节的活动,所以经常有朋友问我1024 程序员节怎么做,本篇就是简单介绍下我的思路,希望对你有用。 1024程序员节的由来 俄罗斯把每年第256(=2^8)天,即平年9月13日或闰年9月12日定为国际程序员

面向对象编程

在编写软件时,你所做的大部分工作就是创建和连接多个值和方法,让他们一起工作,以便提供应用程序的功能。面向对象编程可以帮助你更容易地,并且是声明式地实现这些功能。 在这篇文章中,你将了解到在JavaScript中开始使用类和面向对象编程方法所需要的一切。 前置知识 在阅读本篇文章之前,你需要掌握Jav

[转帖]公司架构师常常提起的DNS负载均衡是个什么鬼?

https://cloud.tencent.com/developer/article/2190719?areaSource=105001.2&traceId=7RuArY2Tm1MQWwQaMnx-Q 传统思路的局限性 CDN是怎么做的 你知道DNS是怎么工作的吗? 神奇的解释权机制(SOA) D

个人语录

任泽平曾问过俞敏洪,新东方为什么非得转型?俞敏洪这样回答: 「别人做过的事情,只要你去做,并且把它做得比别人更好,说不定你也能成功。前提条件就是你必须要比别人做得好,比别人更加有耐心。所以我并不认为一个企业,一定要去做别人没做过的事情。」

脆弱性-鲁棒性-反脆弱性

哈喽大家好,我是咸鱼 在开始本篇文章之前,我想先问小伙伴们一个问题: 每个人都渴望稳定且有序的生活,但如果一个人的生活过于稳定有秩序且可预测,会有什么不好的影响吗? 如果你每天做同样的事情,都按照同样的方式来度过,一旦出现不可预测的变故,你有应对的策略吗? 在《贝叶斯算法人生》中,我说过这个世界是充

想做长期的 AB 实验?快来看看这些坑你踩了没

作者:江颢 1.什么是长期的 AB 实验 大部分情况下,我们做的 AB 实验都是短期的,一到两周或者一个月之内的,通过分析这段时期内测得的实验效应得出实验结论,并最终进行推广。 长期实验即运行时间达数月甚至数年的实验,实验的长期效应指的是需要数月数年的 AB 实验才能积累的实验效应。 那什么场景下还

【Flink入门修炼】2-3 Flink Checkpoint 原理机制

如果让你来做一个有状态流式应用的故障恢复,你会如何来做呢? 单机和多机会遇到什么不同的问题? Flink Checkpoint 是做什么用的?原理是什么?

不抖机灵!让工程师来告诉你做芯片是如何烧钱的!

这是IC男奋斗史的第33篇原创 本文3742字,预计阅读8分钟。 大家应该都知道做芯片是一件非常烧钱的事情。经常看到新闻通稿,某某公司融资了xx亿,外行乍看之下觉得好多钱啊,但实际上可能只够该公司烧一年。那么做芯片到底有多烧钱?钱都花在哪哪些地方了?这篇文章杰哥将从芯片设计公司的角度切入,详细讲解芯