نصائح وحيل من my Telegram-channelpythonetc ، ديسمبر 2019



إنها مجموعة جديدة من النصائح والحيل حول Python والبرمجة من قناة Telegram-channelpythonetc.

المنشورات السابقة .


من الواضح أن المهام asyncio المختلفة لها أكوام مختلفة. يمكنك الاطلاع عليها جميعًا في أي وقت باستخدام asyncio.all_tasks() للحصول على جميع المهام التي تعمل حاليًا و task.get_stack() للحصول على مكدس لكل مهمة.

 import linecache import asyncio import random async def producer(queue): while True: await queue.put(random.random()) await asyncio.sleep(0.01) async def avg_printer(queue): total = 0 cnt = 0 while True: while queue.qsize(): x = await queue.get() total += x cnt += 1 queue.task_done() print(total / cnt) await asyncio.sleep(1) async def monitor(): while True: await asyncio.sleep(1.9) for task in asyncio.all_tasks(): if task is not asyncio.current_task(): f = task.get_stack()[-1] last_line = linecache.getline( f.f_code.co_filename, f.f_lineno, f.f_globals, ) print(task) print('\t', last_line.strip()) print() async def main(): loop = asyncio.get_event_loop() queue = asyncio.Queue() loop.create_task(producer(queue)) loop.create_task(producer(queue)) loop.create_task(producer(queue)) loop.create_task(avg_printer(queue)) loop.create_task(monitor()) loop = asyncio.get_event_loop() loop.create_task(main()) loop.run_forever() 

لتجنب إرسال linecache باستخدام كائن مكدس مباشرة واستخدام الوحدة النمطية linecache يمكنك استدعاء task.print_stack() بدلاً من ذلك.


يمكنك ترجمة أو حذف أحرف سلسلة (مثل الأداة المساعدة tr ) باستخدام طريقة translate str :

 >>> 'Hello, world!'.translate({ ... ord(','): ';', ... ord('o'): '0', ... }) 'Hell0; w0rld!' 

الوسيطة الوحيدة translate هي تعيين رموز رموز القاموس إلى أحرف (أو رموز). عادة ما يكون أكثر ملاءمة لإنشاء مثل هذا القاموس مع طريقة ثابتة str.maketrans :

 >>> 'Hello, world!'.translate(str.maketrans({ ... ',': ';', ... 'o': '0', ... })) 'Hell0; w0rld!' 

أو حتى:

 >>> 'Hello, world!'.translate(str.maketrans( ... ',o', ';0' ... )) 'Hell0; w0rld!' 

الوسيطة الثالثة هي لحذف الأحرف:

 >>> tr = str.maketrans(',o', ';0', '!') >>> tr {44: 59, 111: 48, 33: None} >>> 'Hello, world!'.translate(tr) 'Hell0; w0rld' 


لا mypy تعريفات الأنواع العودية:

 from typing import Optional, Dict from pathlib import Path TreeDict = Dict[str, 'TreeDict'] def tree(path: Path) -> TreeDict: return { f.name: tree(f) if f.is_dir() else None for f in path.iterdir() } 

الخطأ هو Cannot resolve name "TreeDict" (possible cyclic definition) .

ترقبوا هنا: https://github.com/python/mypy/issues/731


وظيفة عادية تحتاج فقط إلى استدعاء نفسها لتصبح متكررة. الأمر ليس بهذه السهولة بالنسبة للمولدات: عادة ما يتعين عليك استخدام yield from المولدات العودية:

 from operator import itemgetter tree = { 'imgs': { '1.png': None, '2.png': None, 'photos': { 'me.jpg': None }, }, 'MANIFEST': None, } def flatten_tree(tree): for name, children in sorted( tree.items(), key=itemgetter(0) ): yield name if children: yield from flatten_tree(children) print(list(flatten_tree(tree))) 


يمكنك استخدام ليس فقط مع المتغيرات ولكن مع أي تعبير. يتم تقييمه على كل تكرار:

 >>> log2 = {} >>> key = 1 >>> for log2[key] in range(100): ... key *= 2 ... >>> log2[16] 4 >>> log2[1024] 10 

Source: https://habr.com/ru/post/ar484006/


All Articles