首页 > 编程语言 >后悔没早知道这些Python特性

后悔没早知道这些Python特性

时间:2023-06-25 18:05:05浏览次数:47  
标签:node 没早 self typing Python key ._ 后悔


写 Python 也好几年时间了。讲道理,在工作中大家肯定遇到过这样的场景:

后悔没早知道这些Python特性_数据

这个故事告诉我们什么?先造轮子再去 GitHub?还是提高下 GitHub 搜索技巧?

都不是!

实际上,在日常的工作中,我们很多需求,无论是常见的、还是不常见的,Python 都为我们提供了一些独特的解决方案,既不需要自己造轮子,也不需要引入新的依赖(引入新的依赖势必会增加项目的复杂度)。

但是 Python 有太多功能和特性被我们忽略了,导致我们在遇到问题的时候,没法第一时间作出良好的决策。

所以,干脆来一起扫清这些被我们忽略的 Python 死角。

装饰器的妙用

我们经常会想完成一些注册&调用的功能,比如我们有四个函数:




现在我们想将这四个函数和 +、-、*、/ 四个操作符绑定,那么我们该怎么做?

可能我们第一反应是这样:

operator_map = {}	
def add(a: int, b: int) -> float:	
    return a + b	
def sub(a: int, b: int) -> float:	
    return a - b	
def mul(a: int, b: int) -> float:	
    return a * b	
def div(a: int, b: int) -> float:	
    return a / b	
operator_map["+"] = add	
operator_map["-"] = sub	
operator_map["*"] = mul	
operator_map["/"] = div

但这样写起来,有一个很大的问题就是太不美观了。因为直接对于 dict 的操作从实际上来讲可维护性是很差的,那么我们这个地方应该怎么做?

在改进这段代码之前,我们首先要明确 Python 中一个很重要的概念,即:函数/方法是:First Class Member 。用不精确的话来讲,就是函数/方法可以作为参数被传递、被使用。

举个例子:

import typing	
def execute(func: typing.Callable, *args, **kwargs) -> typing.Any:	
    return func(*args, **kwargs)	
def print_func(data: int) -> None:	
    print(data)	
execute(print, 2)

大家可以看到我们将 print_func 这个函数作为参数传递给 execute 函数并被调用。

那么我们来改造下之前的代码:



好了,大家看看,目前整体代码的可读性以及可维护性是不是改了很多?

但是我们现在的问题在于,每次都需要在单独调用一次 register_operator 函数,这样也太烦了吧!要不要再改进一下?要得。我们可以用装饰器来改进一下。

首先,看一个最简单的装饰器例子:

import functools	
import typing	
import time	
def execute(func: typing.Callable) -> typing.Callable:	
    @functools.wraps(func)	
    def wraps(*args, **kwargs) -> typing.Any:	
        start_time = time.time()	
        result = func(*args, **kwargs)	
        print("{}".format(time.time() - start_time))	
        return result	
    return wraps	
@execute	
def add(a: int, b: int) -> float:	
    return a + b

我们能看到这段函数的意义是计算函数的执行时间。那么这个原理是什么?

实际上装饰器是一个语法糖,具体可以参见 PEP318 Decorators for Functions and Methods。

简而言之,实际上是 Python 替我们做了一个替换过程。以上面的例子为例,这个替换过程就是 add=execute(add) 。

好了,我们就用这个知识点来改进下之前的代码:

import typing	
operator_map = {}	
def register_operator(operator: str) -> typing.Callable:	
    def wraps(func: typing.Callable) -> typing.Callable:	
        operator_map[operator] = func	
        return func	
    return wraps	
@register_operator("+")	
def add(a: int, b: int) -> float:	
    return a + b	
@register_operator("-")	
def sub(a: int, b: int) -> float:	
    return a - b	
@register_operator("*")	
def mul(a: int, b: int) -> float:	
    return a * b	
@register_operator("/")	
def div(a: int, b: int) -> float:	
    return a / b



这样我们这段代码的注册过程是不是就显得更优雅了?



嗯,是的!实际上 Python 中有很多特性会帮助我们的代码更简洁,更优美。



接下来这个例子很可能帮我们减轻工作量。



聊聊 OrderedDict



dict 是我们经常使用的一种数据解构。但是在 Python 3.6 之前 dict 都是无序的,即我插入的顺序,和数据在 dict 中存放的顺序并无关联(笔者注:Python 3.6 dict 有序只是新版实现的顺带产物,Python 3.7 正式作为 feature 被固定下来)。



但是很多时候,比如在验签等场景,我们需要保证 dict 数据存放顺序,和我们插入顺序是一致的。那么我们该怎么办?



老板有需求下来了,我们肯定不能告诉老板这个需求没法做。那我们就自己实现一个 ordereddict 吧。于是,想了想,写了如下的代码:



import typing	
class OrderedDict:	
    def __init__(self, *args, **kwargs):	
        self._data = {}	
        self._ordered_key = []	
    def __getitem__(self, key: typing.Any) -> typing.Any:	
        return self._data[key]	
    def __setitem__(self, key: typing.Any, value: typing.Any) -> None:	
        if key not in self._data:	
            return	
        self._data[key] = value	
        self._ordered_key.append(key)	
    def __delitem__(self, key: typing.Any):	
        del self._data[key]	
        self._ordered_key.remove(key)

通过额外维护一个 list 来维护 key 插入的顺序。这段代码,看似完成了我们的需求,但是实则存在很大问题。大家可以猜猜问题在哪?

3,2,1!

揭晓答案,这段代码利用 list 来保证 key 的有序性,在删除的时候, list 的删除操作,是一个时间复杂度 O(n) 的操作。换句话说,我们的删除操作随着内部数据的增多,所需的删除时间也变得越长。这对于某些性能敏感的场景是无法接受的。

那要怎么办呢?事实上,Python 在很早之前就已经内置了有序字典,即很多人可能都用过的 collections.OrderedDict 。

在 OrderedDict 中, Python 维护了一个双向链表解构,来保证插入的有序性,如下图所示:

后悔没早知道这些Python特性_Python_02

在最左侧维护一个卫兵节点,卫兵节点的 next 指针恒指向于数据中最后插入的节点。那么插入新的数据时,我们将新的数据插入到卫兵节点之后,从而达成维护插入顺序的目的。



在删除的时候,通过额外维护的一个字典找到待删除的 key 所对应的节点。这个操作是 O(1) 的复杂度,然后大家都知道,双向链表删除一个节点的时间复杂度也是 O(1) 。通过这样保证我们在即便有大量数据的情况下,也能保证相应的性能。



好了,我们按照这个思路来做一个最简单的实现:



import typing	
class Node:	
    def __init__(self, key: typing.Any, value: typing.Any) -> None:	
        self.key = key	
        self.value = value	
        self.prev = None	
        self.next = None	
class OrderedDict:	
    def __init__(self, *args, **kwargs):	
        self._data = {}	
        self._head = Node(None, None)	
        self._last = self._head	
    def __getitem__(self, key: typing.Any) -> typing.Any:	
        if key in self._data:	
            return self._data[key].value	
        raise ValueError	
    def __setitem__(self, key: typing.Any, value: typing.Any) -> None:	
        if key not in self._data:	
            return	
        value_node = Node(key, value)	
        next_node = self._head.next	
        if not next_node:	
            self._head.next = value_node	
            value_node.prev = self._head	
            self._last = value_node	
        else:	
            value_node.next = next_node	
            next_node.prev = value_node	
            value_node.prev = self._head	
            self._head.next = value_node	
        self._data[key] = value_node	
    def __delitem__(self, key: typing.Any):	
        if key not in self._data:	
            return	
        value_node = self._data[key]	
        if value_node == self._last:	
            self._last = value_node.prev	
            self._last.next = None	
        else:	
            prev_node = value_node.prev	
            next_node = value_node.next	
            prev_node.next = next_node	
            next_node.prev = prev_node	
        del self._data[key]	
        del value_node

(此段代码,如有错乱,烦请将浏览字体调小几号)

这只是一个 OrderedDict 的简化版,如果想完成一个完整的 OrderedDict 还有很多很多的 corner case 要去处理。不过现在,我们可以使用内置的数据结构去完成我们需求。怎么样,是不是有了一种幸福的感觉?

随意聊聊

通过今天的两个例子,我们发现 Python 提供了相当多的功能去帮助我们完成日常的工作与学习任务。同时通过去深入地了解 Python 内部的一些功能实现,以便我们能更好地去学习一些知识。比如,上文提到的 OrderedDict 的实现,会让我们学到双头链表的一种非常典型的应用,与此同时,双头链表也会用于诸如 LRU 这样非常常用的数据解构的实现。所以,多去深入了解 Python 的方方面面,有助于我们整体能力的提升。

那么既然电脑上已经有了 Python,为何不将这些 tricks 用起来,让 Python 更 6 呢?安利这位来自德国的大咖,资深养蛇玩家 Dan Bader。

他的博客 dbader.org  每月有超过 20 万的浏览量,Real Python 视频累计浏览量超过百万。订阅他的邮件,每天都会准时收到他更新 Python 技巧,一日不差,是一个极其自律的人。




后悔没早知道这些Python特性_数据_03



畅销书 Python Tricks 作者,影响全球 1 000 000 以上程序员的 PythonistaCafe 社区创始人,Real Python 培训机构总编,拥有近 20 年软件开发经验。巴德尔毕业于欧洲历史悠久的慕尼黑工业大学,该校以优异的科教质量闻名,截至 2018 年已经培养出 17 位诺贝尔奖得主。


他把自己对 Python 的理解成书,试图通过一些常用的小例子帮助更多开发者拥抱 Python。


后悔没早知道这些Python特性_Python_04

精进Python不二之选


《深入理解Python特性》


Dan Bader 著


孙波翔 译


上市两个月获 Amazon 百余条五星评价,详解用好 Python 需要了解的最重要特性,与《流畅的Python》互为补充,Python 进阶必备。帮助 Python 开发人员挖掘这门语言及相关程序库的优秀特性,避免重复劳动,同时写出简洁、流畅、易读、易维护的代码。



他是这么说的:


When I started thinking about the book I'd been a lead developer at this software company in Vancouver, Canada for a while and we were doing lots of Python web development.

最开始有写书计划的时候,我在温哥华的一个软件公司担任开发团队的头儿,我们那会儿正在做大量有关 Python web 的开发工作。

I'd been a passionate user of Python for several years at that point and I always tried going out of my way to teach others at the company how to write better, cleaner, and more Pythonic code.

那时,我是一个狂热的 Python 用户,而且我不遗余力地想教会公司其他人如何编写更简洁易用,更 Pythonic 的代码。

I felt like there were already lots of good resources out there that taught people how to get started with Python, but there was a lack of material that showed the little "tricks" and idioms that are a sign of someone graduating from beginner level to intermediate and beyond.

我意识到已经有很多资源可以教会人们如何入门 Python,但是还是缺少一些能够帮助了解其特性和习惯用法的内容,这样的内容可以帮助一些初学者进阶到更高的水平。

I also wanted to make learning this stuff fun and engaging. I got so much joy from using Python in my day to day work, I wanted to get a chance to share that love and passion for Python as a programming language.

我还想把学习这些东西变得更有趣。在工作中使用 Python 使我得到了很多乐趣,我也想有机会分享对 Python 这门编程语言的热爱。

Writing Python Tricks was my outlet for that. My ideal reader is someone who really wants to make Python their own and learn how to write clean and Pythonic code. I want them to discover best practices and little-known tricks to round out their knowledge, and to fill up their "Python toolbox" one step at a time.

写这本书就是我对那份热爱的实现。我想写给那些想用好 Python 或想写出 Pythonic 代码的人。我希望他们发现这些 Python 的最佳实践和鲜为人知的技巧以完整他们的知识体系,并且一步步地填满他们的“Python工具箱”。


每一个大佬都有一个济世的梦想,他们善于分享,拥抱开源,并致力于建立良好的生态,不愧是养蛇专业户。


好了!不能再说了,我要去码代码了。回见!

标签:node,没早,self,typing,Python,key,._,后悔
From: https://blog.51cto.com/u_15767091/6547477

相关文章

  • Python 中的 JSON 操作:简单、高效的数据交换格式
    在现代的数据交换和存储中,JSON(JavaScriptObjectNotation)作为一种轻量级的数据交换格式,备受青睐。它不仅易于阅读和理解,还可以灵活地表达和存储高维数据。本文将介绍如何在Python中操作JSON文件,实现数据的序列化和反序列化。1.JSON数据格式JSON格式采用键值对的方式......
  • python基础
    输入#程序会停止,直到接受到你输入的值为止name=input("请输入您的名字")数据类型(字面量)数字:int整数float浮点数complex复数(4+3j)bool布尔字符串:str字符串列表:list列表元组:tuple元组集合:set集合字典:dict字典字符串拼接print("我是"+name)#普通拼......
  • Python动态修改实例对象的方法
    代码如下:importtypes#定义一个类classMyClass:deforiginal_method(self):#原始的执行函数print("原始的执行函数")#创建类的实例my_object=MyClass()#定义新的执行函数defnew_function(self):#在这里定义新的执行函数prin......
  • python基础day31 面向对象
    面向过程在支持面向对象的语言中,都有两大范式:1.面向过程;2.面向对象面向过程:核心就是过程二字,即是先干什么,再干什么,最后干什么,就是机械式的思维方式举例: 把大象放进冰箱需要几步? 1.把冰箱门打开2.把大象放进去3.关上冰箱门代码案例:实现面向过程的例子......
  • 【python基础】文件-文件路径
    1.文件路径我们发现不管是写入还是写出操作,我们提供的都是文件名,其实这里准确说应该是文件路径。当我们简单把文件名传递给open函数时,Python将在当前执行程序的文件所在的目录中查找文件名所代表的文件。根据组织文件的方式,可能需要打开不在当前执行程序文件所属目录中的文件。......
  • Python爬虫高并发爬取数据
    高效爬虫可以在较短的时间内获取更多的数据,提高数据的采集速度。这对于需要大量数据支撑的数据分析、机器学习、人工智能等任务非常重要。高效爬虫可以获取更多的原始数据,并允许更精准的数据清洗和处理。这样可以提高数据的质量和关联性,使得后续的分析和挖掘工作更加准确和有价值。......
  • 《最新出炉》系列初窥篇-Python+Playwright自动化测试-4-playwright等待浅析
    1.简介在介绍selenium的时候,宏哥也介绍过等待,是因为在某些元素出现后,才可以进行操作。有时候我们自己忘记添加等待时间后,查了半天代码确定就是没有问题,奇怪的就是获取不到元素。然后搞了好久,或者经过别人的提示才恍然大悟没有添加等待时间。而playwright为了避免我们犯这么low的......
  • 面试Python开发的这道题超简单,我却搞砸了!
    题图 | Shutterstock/studiostoks这道算法题明明超简单……上午10点,在T公司的会议室里,小R正在参加一场他准备了好几天的技术面试。整体来说,他在这场面试中的表现还不错。无论坐在小R对面的面试官提出什么问题,他都能侃侃而谈、对答如流。从单体应用聊到微服务,从虚拟机聊到云计算......
  • 数学竟然可以这样学,用Python魔法突破数学结界!
    今年的高考刚刚过去,在数学考完的当天,“高考数学”又一次荣登微博热搜榜榜首。对于这场数学考试,可谓几家欢喜几家愁,图灵君浏览着微博上读者的留言深有感触。(选自微博账号@四川校园君)对于部分参加高考的同学来说,这场数学考试可能意味着他们与数学的缘分到此为止;然而,对于更多的小伙伴......
  • python操作rabbitmq
     rabbitmq安装部署   RabbitMq生产者消费者模型生产者(producter) 队列消息的产生者,复制生产消息,并将消息传入队列生产者代码:importpikaimportjsoncredentials=pika.PlainCredentials('admin','admin')#mq用户名和密码,用于认证#虚拟队列需要指定参数vir......