python if and_python有趣用法汇总(持续更新)

6486615f-0415-eb11-8da9-e4434bdf6706.png

使用python过程中经常会不经意间遇到非常有趣的用法,于是特意搜集了一些

  • 2019年3月19日更新了and的用法
  • 2019年3月26日更新了for-else的更多讨论
  • 2020年9月8日补充了对and和or的分析

有趣的用法

1.for-else用法

循环正常结束则执行else语句。一般用于循环找符合条件的元素,如果找到则break调出循环,不会触发else;如果没有找到(完整运行循环)则print not found

详见Python中循环语句中的else用法

根据评论区知友的提醒(已置顶),《Effictive Python》一书中对for-else用法提出了质疑,主要观点是可以通过封装成函数来取代这一用法,而封装成函数是更加通用易懂的做法,所以一般不会使用for-else用法。为了不影响文章的紧凑,我把评论区对书上内容的引用放在文末“更新补充”部分,有兴趣的读者可以去看一下。

2.try-else用法

如果没有触发异常就执行else

参考这里

3.解包用法

类似这样a,b,c = ['a', 'b', 'c']

python有趣的解包用法

4.单行if-else

a = 1
b = 3 if a == 1 else 2
print('it is one' if a == 1 else 'no')

5.迭代器传入函数中不用加括号

# 一般是这样
a = (i for i in range(10))
sum(a)
# 我们可以这样
sum((i for i in range(10)))
# 但我们还可以这样
sum(i for i in range(10))
# 类似的有
' '.join(str(i) for i in range(10))

6.对生成器进行筛选

一个素数生成器的例子

7.or的用法

python中x or y表示如果x为真就是x的值,否则为y的值

我们会经常看到类似这样的用法(比如函数的一个value参数没有设置默认值,这样使用就允许它不赋值)

value = value or {}
# 相当于
value = value if value else {}

8.and的用法

python中x and y表示如果x是假,结果就是x的值,否则就是y的值

x and y and z多个and连接时,如果全是真结果就是最后一个的值;如果中间有假的值,结果就是第一个假的值。

举一个例子

def not_empty(a):
    return a and a.strip()

not_empty(' a ')
# 值为 'a'
not_empty(None)
# 不会报错(如果 return a.strip() 就会报错)

# 在处理None的问题上相当于
def not_empty(a):
    if a is None:
        return None
    else:
        return a.strip()

# 或者
def not_empty(a):
    return a.strip() if a is not None else None

细细品味and和or的差别,他们逻辑类似,但是实现的功能是不可以相互替代的

  • or 是结果如果不满意有个善后工作
  • and是要做一件事之前先检验一下,不能做就不让它做

二者都可以用if else替代,而且用一般用if else的可读性还更强,看看下面这个例子

def get_weight_name_v1(logname=None, savename=None):
    log_str = '-log_{}'.format(logname) if logname is not None else ''
    save_str = '-save_{}'.format(savename) if savename is not None else ''
    return 'weight{}{}.ckpt'.format(log_str, save_str)

def get_weight_name_v2(logname=None, savename=None):
    log_str = logname is not None and '-log_{}'.format(logname) or ''
    save_str = savename is not None and '-save_{}'.format(savename) or ''
    return 'weight{}{}.ckpt'.format(log_str, save_str)

get_weight_name_v1('abc', 'bcd') # 'weight-log_abc-save_bcd.ckpt'
get_weight_name_v1('abc', None)  # 'weight-log_abc.ckpt'

get_weight_name_v2(None, 'abc')  # 'weight-save_abc.ckpt'
get_weight_name_v2(None, None)   # 'weight.ckpt'

两个函数的功能是一样的,即传入的参数如果是None,就不加对应部分字符,不是None就加上,看下面四个例子的输出可以很容易明白。而具体看实现,会发现第一种比第二种好懂很多,第二种又andor,比较绕,不了解这一用法的人还得去试。

因此赋值是还是推荐用if/elseand/or更多的应用场景是if判断,要输出这个表达式的True/False,而不是具体什么值。在if语句中,用or表示只要其中一项为True整体就是True,用and表示必须所有项都为True整体才会是True,结合上面对and/or表达式的输出,可以发现输出的值和True/False结果对得上。

另外一点值得注意的是,and/or都有惰性求值的特点,不是将连接的所有项都算出来再统一看True/False,而是从前向后算

  • and前面一项如果是False,则后面的项都不会再计算。相当于从前向后算一直在找False的项。
  • or前面一项如果是True,则后面的项都不会计算。相当于从前向后算一直在找True的项。
def contain_a(l):
    if l is not None and 'a' in l:
        print(l)
        
contain_a(None)

正常如果运行'a' in None是会报错的,但上面的代码不会报错,因为and后面的部分没有执行。

9.if value:

# 要用
if value:
# 不要用
if value == True:

这里总结一下这种情况下什么时候是True,什么时候是False

False: 0 0.0 '' [] {} () set() None False

True:

  • ' ' 'anything' [''] [0] (None, )
  • 没有内容的可迭代对象

另外要注意一点,我们用if判断一个对象是不是None的时候,要if a is None而不要直接if a,因为如果是后者,有非常多不是None的情况也会判定为False,比如空字符串、空列表等,为了精确指定None还是要用前者,这也是一种规范。

10.下划线的特殊使用

python中下划线是一种特殊的变量和符号,有一些特殊的用途

详见python中下划线的使用

11.文档字符串

python有一种独一无二的注释方式,在包、模块、函数、类中第一句,使用'''doc'''这样三引号注释,就可以在对象中用__doc__的方式提取;更方便的是,查看一个函数帮助文档时(比如jupyter中shift+tab),会把文档字符串显示出来。

比较规范的写法是这样的(这里参考grequests模块的写法)

def myfun(a, b):
    '''add two numbers
    :param a: one number
    :param b: another number
    :returns: a number
    '''
    print(a + b)

print(myfun.__doc__)

# 结果为
add two numbers
    :param a: one number
    :param b: another number
    :returns: a number

其实参数还有其他的写法,如numpy库的写法,可以看这里

除此之外,函数注释还有另一种方式,函数名可以直接调用某个参数的注释,详见Python 的函数注释。还可以参考PEP 257

有用的函数

1.sum的本质

本质:sum(iterable, start=0)将可迭代对象使用+连接

所以sum([[1,2],[3,4]], [])返回结果为[1, 2, 3, 4]

2.range(start, stop[, step])

可以直接用for i in range(10, 0, -1)降序循环

3.enumerate循环索引

for index, item in enumerate(['a', 'b', 'c']):
    print(index, item)
输出:
0 a
1 b
2 c

4.管道操作

func1(func2(func3(a)))写成类似a %>% func3 %>% func2 %>% func1,清晰展示函数执行的顺序,增强可读性

python本身不带有这样的用法,只是一些库提供了这样的用法,比如pandas和syntax_sugar

参考stackoverflow上的这个回答

其他

另外,就是一些基础的

  • 列表推导式
  • 装饰器
  • 生成器
  • map reduce filter
  • 链式比较
  • 类的魔术方法

上面很多在廖雪峰python教程中都能找到

另外,可以参考知乎上的两个问题

  • Python 有哪些好玩的语法糖?
  • Python 有哪些优雅的代码实现让自己的代码更pythonic?

阅读优秀的代码也是提高编程水平的好方法,参考下面这个问题

初学 Python,有哪些 Pythonic 的源码推荐阅读?

学习代码规范可以参考下面资料

  • PEP8
  • Python 代码、单元测试和项目规范
  • google开源项目风格指南

更新补充

  1. for-else的更多讨论

下面引用《Effictive Python》一书中内容

a = 4
b = 9
for i in range(2, min(a, b) + 1):
print('Testing', i)
if a % i == 0 and b % i == 0:
print('Not coprime')
break
else:
print('Coprime')

随后作者写到:

In practice, you wouldn’t write the code this way. Instead, you’d write a helper function to do the calculation. Such a helper function is written in two common styles.
The first approach is to return early when you find the condition you’re looking for. You return the default outcome if you fall through the loop.
def coprime(a, b):
for i in range(2, min(a, b) + 1):
if a % i == 0 and b % i == 0:
return False
return True
The second way is to have a result variable that indicates whether you’ve found what you’re looking for in the loop. You break out of the loop as soon as you find something.
def coprime2(a, b):
is_coprime = True
for i in range(2, min(a, b) + 1):
if a % i == 0 and b % i == 0:
is_coprime = False
break
return is_coprime

结尾:

Both of these approaches are so much clearer to readers of unfamiliar code. The expressivity you gain from the else block doesn’t outweigh the burden you put on people (including yourself) who want to understand your code in the future. Simple constructs like loops should be self-evident in Python. You should avoid using else blocks after loops entirely.

总结起来就是for-else的优势是可以被写函数的方式替代的

专栏信息

专栏主页:python编程

专栏目录:目录

版本说明:软件及包版本说明

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值