Python 代码实践小结

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介:

最近写了较多的 Python 脚本,将最近自己写的脚本进行一个总结,其中有些是 Python 独有的,有些是所有程序设计中共有的:

考虑使用 Logger(logger 怎么配置,需要输出哪些信息 — 可以反向考虑,自己看到这个 logger 的时候想了解什么信息)

传递的数据结构如何考虑(是否对调用方有先验知识的要求,比如返回一个 Tuple,则需要用户了解 tuple 中元素的顺序,这样情况是否应该进行封装;),数据结构定义清楚了,很多东西也就清楚了。

如何操作数据库(可以学习 sqlalchemy,包括 core 和 orm 两种 api)

异常如何处理(异常应该分开捕获 — 可以清楚的知道什么情况下导致的,异常之后应该打印日志说明出现什么问题,如果情况恶劣需要进行异常再次抛出或者报警)

所有获取资源的地方都应该做 check(a. 没有获取到会怎么办;b.获取到异常的怎么办)

所有操作资源的地方都应该检查是否操作成功

每个函数都应该简短,如果函数过长应该进行拆分(有个建议值,函数包含的行数应该在 20-30 行之间,具体按照这个规范做过一次之后就会发现这样真好)

使用 class 之后,考虑重构 __str__ 函数,用户打印输出(如果不实现 __str__,会调用 __repr__ ),如果对象放到 collection 中之后,需要实现 __repr__ 函数,用于打印整个 collection 的时候,直观显示

如果有些资源会发生变化,可以单独抽取出来,做成函数,这样后续调用就可以不用改变了

附上一份 Python2.7 代码(将一些私有的东西进行了修改)


 
 
  1. # -*- coding:utf-8 -*- 
  2.   
  3. from sqlalchemy import create_engine 
  4. import logging 
  5. from logging.config import fileConfig 
  6. import requests 
  7. import Clinet # 私有的模块 
  8.   
  9. fileConfig("logging_config.ini"
  10. logger = logging.getLogger("killduplicatedjob"
  11.   
  12. #配置可以单独放到一个模块中 
  13. DB_USER = "xxxxxxx" 
  14. DB_PASSWORD = "xxxxxxxx" 
  15. DB_PORT = 111111 
  16. DB_HOST_PORT = "xxxxxxxxxx" 
  17. DB_DATA_BASE = "xxxxxxxxxxx" 
  18.   
  19. REST_API_URL = "http://sample.com" 
  20.   
  21. engine = create_engine("mysql://%s:%s@%s:%s/%s" % (DB_USER, DB_PASSWORD, DB_HOST_PORT, DB_PORT, DB_DATA_BASE)) 
  22.   
  23. # 这个 class 是为了在函数间传递时,不需要使用方了解属性的具体顺序而写的,也可以放到一个单独的模块中 
  24. class DuplicatedJobs(object): 
  25.  def __init__(self, app_id, app_name, user): 
  26.  self.app_id = app_id 
  27.  self.app_name = app_name 
  28.  self.user = user 
  29.   
  30. def __repr__(self): 
  31.  return '[appid:%s, app_name:%s, user:%s]' % (self.app_id, self.app_name, self.user
  32.   
  33. def find_duplicated_jobs(): 
  34.  logger.info("starting find duplicated jobs"
  35.  (running_apps, app_name_to_user) = get_all_running_jobs() 
  36.  all_apps_on_yarn = get_apps_from_yarn_with_queue(get_resource_queue()) 
  37.   
  38. duplicated_jobs = [] 
  39.  for app in all_apps_on_yarn: 
  40.  (app_id, app_name) = app 
  41.   
  42. if app_id not in running_apps: 
  43.  if not app_name.startswith("test"): 
  44.  logger.info("find a duplicated job, prefixed_name[%s] with appid[%s]" % (app_name, app_id)) 
  45.  user = app_name_to_user[app_name] 
  46.  duplicated_jobs.append(DuplicatedJobs(app_id, app_name, user)) 
  47.  else
  48.  logger.info("Job[%s] is a test job, would not kill it" % app_name) 
  49.   
  50. logger.info("Find duplicated jobs [%s]" % duplicated_jobs) 
  51.   
  52. return duplicated_jobs 
  53.   
  54. def get_apps_from_yarn_with_queue(queue): 
  55.  param = {"queue": queue} 
  56.  r = requests.get(REST_API_URL, params=param) 
  57.  apps_on_yarn = [] 
  58.  try: 
  59.  jobs = r.json().get("apps"
  60.  app_list = jobs.get("app", []) 
  61.  for app in app_list: 
  62.  app_id = app.get("id"
  63.  name = app.get("name"
  64.  apps_on_yarn.append((app_id, name)) 
  65.   
  66. except Exception as e: #Exception 最好进行单独的分开,针对每一种 Exception 进行不同的处理 
  67.  logger.error("Get apps from Yarn Error, message[%s]" % e.message) 
  68.   
  69. logger.info("Fetch all apps from Yarn [%s]" % apps_on_yarn) 
  70.   
  71. return apps_on_yarn 
  72.   
  73. def get_all_running_jobs(): 
  74.  job_infos = get_result_from_mysql("select * from xxxx where xx=yy"
  75.   
  76. app_ids = [] 
  77.  app_name_to_user = {} 
  78.  for (topology_id, topology_name) in job_infos: 
  79.  status_set = get_result_from_mysql("select * from xxxx where xx=yy"
  80.  application_id = status_set[0][0] 
  81.  if "" != application_id: 
  82.  configed_resource_queue = get_result_from_mysql( 
  83.  "select * from xxxx where xx=yy"
  84.  app_ids.append(application_id) 
  85.  app_name_to_user[topology_name] = configed_resource_queue[0][0].split(".")[1] 
  86.   
  87. logger.info("All running jobs appids[%s] topology_name2user[%s]" % (app_ids, app_name_to_user)) 
  88.  return app_ids, app_name_to_user 
  89.   
  90. def kill_duplicated_jobs(duplicated_jobs): 
  91.  for job in duplicated_jobs: 
  92.  app_id = job.app_id 
  93.  app_name = job.app_name 
  94.  user = job.user 
  95.  logger.info("try to kill job[%s] with appid[%s] for user[%s]" % (app_name, app_id, user)) 
  96.  try: 
  97.  Client.kill_job(app_id, user
  98.  logger.info("Job[%s] with appid[%s] for user[%s] has been killed" % (app_name, app_id, user)) 
  99.  except Exception as e: 
  100.  logger.error("Can't kill job[%s] with appid[%s] for user[%s]" % (app_name, app_id, user)) 
  101.   
  102. def get_result_from_mysql(sql): 
  103.  a = engine.execute(sql) 
  104.  return a.fetchall() 
  105.   
  106. # 因为下面的资源可能发生变化,而且可能包含一些具体的逻辑,因此单独抽取出来,独立成一个函数 
  107. def get_resource_queue(): 
  108.  return "xxxxxxxxxxxxx" 
  109.   
  110. if __name__ == "__main__"
  111.  kill_duplicated_jobs(find_duplicated_jobs()) 
  112.   

其中 logger 配置文件如下(对于 Python 的 logger,官方文档写的非常好,建议读一次,并且实践一次)


 
 
  1. [loggers] 
  2. keys=root, simpleLogger 
  3.   
  4. [handlers] 
  5. keys=consoleHandler, logger_handler 
  6.   
  7. [formatters] 
  8. keys=formatter 
  9.   
  10. [logger_root] 
  11. level=WARN 
  12. handlers=consoleHandler 
  13.   
  14. [logger_simpleLogger] 
  15. level=INFO 
  16. handlers=logger_handler 
  17. propagate=0 
  18. qualname=killduplicatedjob 
  19.   
  20. [handler_consoleHandler] 
  21. class=StreamHandler 
  22. level=WARN 
  23. formatter=formatter 
  24. args=(sys.stdout,) 
  25.   
  26. [handler_logger_handler] 
  27. class=logging.handlers.RotatingFileHandler 
  28. level=INFO 
  29. formatter=formatter 
  30. args=("kill_duplicated_streaming.log""a", 52428800, 3,) 
  31.   
  32. [formatter_formatter] 
  33. format=%(asctime)s %(name)-12s %(levelname)-5s %(message)s 


本文作者:佚名

来源:51CTO

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
3天前
|
存储 人工智能 运维
【01】做一个精美的打飞机小游戏,浅尝阿里云通义灵码python小游戏开发AI编程-之飞机大战小游戏上手实践-优雅草央千澈-用ai开发小游戏尝试-分享源代码和游戏包
【01】做一个精美的打飞机小游戏,浅尝阿里云通义灵码python小游戏开发AI编程-之飞机大战小游戏上手实践-优雅草央千澈-用ai开发小游戏尝试-分享源代码和游戏包
【01】做一个精美的打飞机小游戏,浅尝阿里云通义灵码python小游戏开发AI编程-之飞机大战小游戏上手实践-优雅草央千澈-用ai开发小游戏尝试-分享源代码和游戏包
|
1月前
|
开发框架 数据建模 中间件
Python中的装饰器:简化代码,增强功能
在Python的世界里,装饰器是那些静悄悄的幕后英雄。它们不张扬,却能默默地为函数或类增添强大的功能。本文将带你了解装饰器的魅力所在,从基础概念到实际应用,我们一步步揭开装饰器的神秘面纱。准备好了吗?让我们开始这段简洁而富有启发性的旅程吧!
43 6
|
2月前
|
机器学习/深度学习 算法 数据挖掘
线性回归模型的原理、实现及应用,特别是在 Python 中的实践
本文深入探讨了线性回归模型的原理、实现及应用,特别是在 Python 中的实践。线性回归假设因变量与自变量间存在线性关系,通过建立线性方程预测未知数据。文章介绍了模型的基本原理、实现步骤、Python 常用库(如 Scikit-learn 和 Statsmodels)、参数解释、优缺点及扩展应用,强调了其在数据分析中的重要性和局限性。
85 3
|
13天前
|
Python
课程设计项目之基于Python实现围棋游戏代码
游戏进去默认为九路玩法,当然也可以选择十三路或是十九路玩法 使用pycharam打开项目,pip安装模块并引用,然后运行即可, 代码每行都有详细的注释,可以做课程设计或者毕业设计项目参考
54 33
|
14天前
|
JavaScript API C#
【Azure Developer】Python代码调用Graph API将外部用户添加到组,结果无效,也无错误信息
根据Graph API文档,在单个请求中将多个成员添加到组时,Python代码示例中的`members@odata.bind`被错误写为`members@odata_bind`,导致用户未成功添加。
37 10
|
1月前
|
数据可视化 算法 数据挖掘
Python量化投资实践:基于蒙特卡洛模拟的投资组合风险建模与分析
蒙特卡洛模拟是一种利用重复随机抽样解决确定性问题的计算方法,广泛应用于金融领域的不确定性建模和风险评估。本文介绍如何使用Python和EODHD API获取历史交易数据,通过模拟生成未来价格路径,分析投资风险与收益,包括VaR和CVaR计算,以辅助投资者制定合理决策。
77 15
|
1月前
|
数据可视化 Python
以下是一些常用的图表类型及其Python代码示例,使用Matplotlib和Seaborn库。
通过这些思维导图和分析说明表,您可以更直观地理解和选择适合的数据可视化图表类型,帮助更有效地展示和分析数据。
73 8
|
1月前
|
API Python
【Azure Developer】分享一段Python代码调用Graph API创建用户的示例
分享一段Python代码调用Graph API创建用户的示例
56 11
|
1月前
|
测试技术 Python
探索Python中的装饰器:简化代码,增强功能
在Python的世界中,装饰器是那些能够为我们的代码增添魔力的小精灵。它们不仅让代码看起来更加优雅,还能在不改变原有函数定义的情况下,增加额外的功能。本文将通过生动的例子和易于理解的语言,带你领略装饰器的奥秘,从基础概念到实际应用,一起开启Python装饰器的奇妙旅程。
44 11
|
1月前
|
测试技术 开发者 Python
探索Python中的装饰器:从入门到实践
装饰器,在Python中是一块强大的语法糖,它允许我们在不修改原函数代码的情况下增加额外的功能。本文将通过简单易懂的语言和实例,带你一步步了解装饰器的基本概念、使用方法以及如何自定义装饰器。我们还将探讨装饰器在实战中的应用,让你能够在实际编程中灵活运用这一技术。
40 7