diff --git "a/Day91-100/95.\344\275\277\347\224\250Django\345\274\200\345\217\221\345\225\206\344\270\232\351\241\271\347\233\256.md" "b/Day91-100/95.\344\275\277\347\224\250Django\345\274\200\345\217\221\345\225\206\344\270\232\351\241\271\347\233\256.md" index 391d44d85..5229ab1d5 100644 --- "a/Day91-100/95.\344\275\277\347\224\250Django\345\274\200\345\217\221\345\225\206\344\270\232\351\241\271\347\233\256.md" +++ "b/Day91-100/95.\344\275\277\347\224\250Django\345\274\200\345\217\221\345\225\206\344\270\232\351\241\271\347\233\256.md" @@ -1700,9 +1700,9 @@ Celery是一个本身不提供队列服务,官方推荐使用RabbitMQ或Redis # 将来实际部署项目的时候生产者、消费者、消息队列可能都是不同节点 beat_schedule={ 'task1': { - 'task': 'common.tasks.show_msg', + 'task': 'common.tasks.scheduled_task', 'schedule': crontab('*', '*', '*', '*', '*'), - 'args': ('刘强东,奶茶妹妹喊你回家喝奶啦', ) + 'args': ('...', ) }, }, ) @@ -1710,8 +1710,8 @@ Celery是一个本身不提供队列服务,官方推荐使用RabbitMQ或Redis ```Python @app.task - def show_msg(content): - print(content) + def scheduled_task(*args, **kwargs): + pass ``` 7. 启动Celery创建执行定时任务的beat(消息的生产者)。 diff --git a/res/python-qq-group.png b/res/python-qq-group.png index 6cbb6981c..018cdb641 100644 Binary files a/res/python-qq-group.png and b/res/python-qq-group.png differ diff --git a/res/int-is-comparation.png "b/\347\225\252\345\244\226\347\257\207/res/int-is-comparation.png" similarity index 100% rename from res/int-is-comparation.png rename to "\347\225\252\345\244\226\347\257\207/res/int-is-comparation.png" diff --git "a/\347\225\252\345\244\226\347\257\207/\344\270\200\344\270\252\345\260\217\344\276\213\345\255\220\345\212\251\344\275\240\345\275\273\345\272\225\347\220\206\350\247\243\345\215\217\347\250\213.md" "b/\347\225\252\345\244\226\347\257\207/\344\270\200\344\270\252\345\260\217\344\276\213\345\255\220\345\212\251\344\275\240\345\275\273\345\272\225\347\220\206\350\247\243\345\215\217\347\250\213.md" new file mode 100644 index 000000000..646266ac7 --- /dev/null +++ "b/\347\225\252\345\244\226\347\257\207/\344\270\200\344\270\252\345\260\217\344\276\213\345\255\220\345\212\251\344\275\240\345\275\273\345\272\225\347\220\206\350\247\243\345\215\217\347\250\213.md" @@ -0,0 +1,53 @@ +## 一个小例子助你彻底理解协程 + +协程,可能是Python中最让初学者困惑的知识点之一,它也是Python中实现并发编程的一种重要方式。Python中可以使用多线程和多进程来实现并发,这两种方式相对来说是大家比较熟悉的。事实上,还有一种实现并发的方式叫做异步编程,而协程就是实现异步编程的必要方式。 + +所谓协程,可以简单的理解为多个相互协作的子程序。在同一个线程中,当一个子程序阻塞时,我们可以让程序马上从一个子程序切换到另一个子程序,从而避免CPU因程序阻塞而闲置,这样就可以提升CPU的利用率,相当于用一种协作的方式加速了程序的执行。所以,我们可以言简意赅的说:**协程实现了协作式并发**。 + +接下来用一个小例子帮助大家理解什么是协作式并发,先看看下面的代码。 + +```Python +import time + + +def display(num): + time.sleep(1) + print(num) + + +for num in range(10): + display(num) +``` + +上面这段代码相信大家很容看懂,程序会输出0到9的数字,每隔1秒中输出一个数字,因此整个程序的执行需要大约10秒时间。值得注意的是,因为没有使用多线程或多进程,程序中只有一个执行单元,而`time.sleep(1)`的休眠操作会让整个线程停滞1秒钟,对于上面的代码来说,在这段时间里面CPU是完全闲置的没有做什么事情。 + +我们再来看看使用协程会发生什么事情。从Python 3.5开始,使用协程实现协作式编发有了更为便捷的语法,我们可以使用`async`来定义异步函数,可以使用`await`让一个阻塞的子程序将CPU让给与它协作的子程序。在Python 3.7中,`asyanc`和`await`成为了正式的关键字,让开发者有一种喜大普奔的感觉。我们先看看如何定义一个异步函数。 + +```Python +import asyncio + + +async def display(num): + await asyncio.sleep(1) + print(num) +``` + +接下来敲黑板说重点。异步函数不同于普通函数,调用普通函数会得到返回值,而调用异步函数会得到一个协程对象。我们需要将协程对象放到一个事件循环中才能达到与其他协程对象协作的效果,因为事件循环会负责处理子程序切换的操作,简单的说就是让阻塞的子程序让出CPU给可以执行的子程序。 + +我们先通过下面的列表生成式来代码10个协程对象,跟刚才在循环中调用display函数的道理一致。 + +```Python +coroutines = [display(num) for num in range(10)] +``` + +通过下面的代码可以获取事件循环并将协程对象放入事件循环中。 + +```Python +loop = asyncio.get_event_loop() +loop.run_until_complete(asyncio.wait(coroutines)) +loop.close() +``` + +执行上面的代码会发现,10个分别会阻塞1秒钟的协程总共只阻塞了约1秒种的时间,这就说明**协程对象一旦阻塞会将CPU让出而不是让CPU处于闲置状态**,这样就大大的**提升了CPU的利用率**。而且我们还会注意到,0到9的数字并不是按照我们创建协程对象的顺序打印出来的,这正是我们想要的结果啊;另外,多次执行该程序会发现每次输出的结果都不太一样,这正是并发程序本身执行顺序不确定性造成的结果。 + +上面的例子来自于著名的“花书”(《Python高级并发编程》),为了让大家对协程的体会更加深刻,对原书的代码做了小的改动,这个例子虽然简单,但是它已经让你体会到了协作式并发的魅力。在商业项目中,如果需要使用协作式并发,还可以将系统默认的事件循环替换为`uvloop`提供的事件循环,这样会获得更好的性能,因为`uvloop`是基于著名的跨平台异步I/O库libuv实现的。另外,如果要做基于HTTP的网络编程,三方库**aiohttp**是不错的选择,它基于asyncio实现了异步的HTTP服务器和客户端。 \ No newline at end of file diff --git "a/\347\225\252\345\244\226\347\257\207/\347\224\250\345\207\275\346\225\260\350\277\230\346\230\257\347\224\250\345\244\215\346\235\202\347\232\204\350\241\250\350\276\276\345\274\217.md" "b/\347\225\252\345\244\226\347\257\207/\347\224\250\345\207\275\346\225\260\350\277\230\346\230\257\347\224\250\345\244\215\346\235\202\347\232\204\350\241\250\350\276\276\345\274\217.md" index b084543a5..5fa64cc34 100644 --- "a/\347\225\252\345\244\226\347\257\207/\347\224\250\345\207\275\346\225\260\350\277\230\346\230\257\347\224\250\345\244\215\346\235\202\347\232\204\350\241\250\350\276\276\345\274\217.md" +++ "b/\347\225\252\345\244\226\347\257\207/\347\224\250\345\207\275\346\225\260\350\277\230\346\230\257\347\224\250\345\244\215\346\235\202\347\232\204\350\241\250\350\276\276\345\274\217.md" @@ -1,4 +1,4 @@ -## 要不要使用复杂表达式 +## 用函数还是用复杂的表达式 Perl语言的原作者*Larry Wall*曾经说过,伟大的程序员都有三个优点:懒惰、暴躁和自负。乍一看这三个词语没有一个是褒义词,但在程序员的世界里,这三个词有不同的意义。首先,懒惰会促使程序员去写一些省事儿的程序来辅助自己或别人更好的完成工作,这样我们就无需做那些重复和繁琐的劳动;同理能够用3行代码解决的事情,我们也绝不会写出10行代码来。其次,暴躁会让程序员主动的去完成一些你还没有提出的工作,去优化自己的代码让它更有效率,能够3秒钟完成的任务,我们绝不能容忍1分钟的等待。最后,自负会促使程序员写出可靠无误的代码,我们写代码不是为了接受批评和指责,而是为了让其他人来膜拜。