log模块 logging 模块提供模块级别的函数记录日志 日志相关概念 - 日志 - 日志的级别(level) - 不同的用户关注不同的程序信息 - DEBUG - INFO - NOTICE - ERROR - CRITTCAL - ALERT - EMERGENCY - LOG的作用 - 调试 - 了解软件的运行情况 - 分析定位问题 - 日志信息 - time - 地点 - level - 内容 logging模块 - 模块中的自定义级别 - DEBUG - INFO - WARNING - ERROR - CRITICAL - 初始化/写日志实例需要指定级别,只有当级别等于或高于制定级别才被记录 - 使用方式 - 直接使用logging(封装了其他组件) - logging四大组件直接定制 logging模块级别的日志 - 使用一下几个函数 - logging.debug(msg, *args, **kwargs) 创建一条严重级别为DEBUG的日志记录 - logging.info(msg, *args, **kwargs) 创建一条严重级别为INFO的日志记录 - logging.warning(msg, *args, **kwargs) 创建一条严重级别为WARNING的日志记录 - logging.error(msg, *args, **kwargs) 创建一条严重级别为ERROR的日志记录 - logging.critical(msg, *args, **kwargs) 创建一条严重级别为CRITICAL的日志记录 - logging.log(level, *args, **kwargs) 创建一条严重级别为level的日志记录 - logging.basicConfig(**kwargs) 对root logger进行一次性配置 logging.basicConfig(**kwargs) - 只在第一次调用的时候起作用 - 参数format有默认参数名 - 不配置logger则使用默认值 - 默认输出 sys.stderr - 默认级别 WARNING - 默认格式 level:log_name:content
1 import logging2 LOG_FORMAT = "%(asctime)s=====%(levelname)s++++++%(message)s" # 自定义格式3 logging.basicConfig(filename="rz.log", level=logging.DEBUG, format=LOG_FORMAT) # 没有自定义,则默认不显示warning级别一下的日志4 logging.debug("This is a debug log")5 logging.info("This is a info log")6 logging.warning("This is a warning log")7 logging.error("This is a error log")8 logging.critical("This is a critical log")
logging模块的处理流程 - 四大组件 - 日志器(Logger):产生日志的一个接口 - 处理器(Handler):把产生的日志发送到相应的目的地 - 过滤器(Filter):更精细的控制日志的输出 - 格式器(Formatter):对输出信息进行格式化 多线程 多进程 程序:一堆代码以文本形式存入一个文档 进程:代码运行的状态 包含地址空间,内存,数据栈 每个进程有自己完全独立的运行环境,多进程共享数据是个问题 线程:一个进程的独立运行片段每一个进程可以有多个线程 轻量化的进程 一个进程的多个线程间共享数据和上下文运行环境 共享互斥问题 全局解释器锁(GIL) Python代码的执行是由python虚拟机进行控制 在主循环中只能够有一个控制线程在执行 Python中的有关包 thread: 有问题,不好用,py3改成了_thread threading:通行的包
1 ''' 2 利用time函数生成两个函数 3 顺序调用、 4 计算总的运行时间 5 ''' 6 7 import time 8 import _thread 9 def loop1():10 # ctime 得到当前时间11 print('Start loop 1 at : ', time.ctime())12 # 睡眠时间 单位毫秒13 time.sleep(4)14 print('End loop 1 at :', time.ctime())15 16 def loop2():17 print("Start loop 2 at :", time.ctime())18 time.sleep(2)19 print('End loop 2 at :', time.ctime())20 21 def main():22 print("Starting at :", time.ctime())23 # 启动多线程的意思是用多线程去执行某个函数24 # 启动多线程函数为start_new_thead25 # 参数两个,一个是需要运行的函数名,第二个是函数的参数作为元组使用,为空则使用空元组26 # 注意:如果函数只有一个参数,需要参数后有一个逗号27 _thread.start_new_thread(loop1, ())28 _thread.start_new_thread(loop2, ())29 print("All done at", time.ctime())30 if __name__ == '__main__':31 main()
threading的使用 直接利用threading.Thread生成Thread实例 1.t = threading.Thread(target=...., args=(...,)) 2.t.start()启动多线程 3.t.join() 等待多线程执行完成
1 import time 2 import threading 3 def loop1(in1): 4 # ctime 得到当前时间 5 print('Start loop 1 at : ', time.ctime()) 6 print("我是参数", in1) 7 # 睡眠时间 单位毫秒 8 time.sleep(4) 9 print('End loop 1 at :', time.ctime())10 11 def loop2(in1, in2):12 print("Start loop 2 at :", time.ctime())13 print("我是参数", in1, "和参数", in2)14 time.sleep(2)15 16 print('End loop 2 at :', time.ctime())17 '''18 主线程执行完自己的任务以后,就退出了,19 此时子线程会继续执行自己的任务,直到自己的任务结束20 '''21 def main():22 print("Starting at :", time.ctime())23 # 生成treading.Thread实例24 t1 = threading.Thread(target=loop1, args=("老王",))25 t1.start()26 t2 = threading.Thread(target=loop2, args=("老张", "老李"))27 t2.start()28 29 print("All done at", time.ctime())30 if __name__ == '__main__':31 main()32 33 34 '''join所完成的工作就是线程同步,即主线程任务结束之后,35 进入阻塞状态,一直等待其他的子线程执行结束之后,主线程在终止36 '''37 import time38 import threading39 def loop1(in1):40 # ctime 得到当前时间41 print('Start loop 1 at : ', time.ctime())42 print("我是参数", in1)43 # 睡眠时间 单位毫秒44 time.sleep(4)45 print('End loop 1 at :', time.ctime())46 47 def loop2(in1, in2):48 print("Start loop 2 at :", time.ctime())49 print("我是参数", in1, "和参数", in2)50 time.sleep(2)51 52 print('End loop 2 at :', time.ctime())53 54 def main():55 print("Starting at :", time.ctime())56 # 生成treading.Thread实例57 t1 = threading.Thread(target=loop1, args=("老王",))58 t1.start()59 t2 = threading.Thread(target=loop2, args=("老张", "老李"))60 t2.start()61 t1.join() # 加入join62 t2.join() # 加入join63 print("All done at", time.ctime())64 if __name__ == '__main__':65 main()
守护线程 daemon 如果在程序中,将子线程设置为守护线程,则子线程会在主线程结束的时候退出 一般认为守护线程不重要或者不能离开子线程单独执行的时候使用
1 ''' 2 子线程t1设置为守护线程,由于主线程比子线程结束早,导致子线程没有输出最后一句就被终止了 3 ''' 4 import time 5 import threading 6 7 def fun(): 8 print("Start fun") 9 time.sleep(2)10 print("End fun")11 12 print("Main thread")13 14 t1 = threading.Thread(target=fun, args=())15 t1.setDaemon(True) # 将子线程设置为守护线程,16 t1.start()17 18 time.sleep(1)19 print("Main thread end")
线程的常用属性 threading.currentThread: 返回当前线程变量 threading.enumerate: 返回一个包含正在运行的线程list threading.activeCount: 返回正在运行的线程数量 thr.setName: 给线程设置名字 thr.getName: 得到线程的名字
1 import time 2 import threading 3 def loop1(): 4 print('Start loop 1 at : ', time.ctime()) 5 time.sleep(4) 6 print('End loop 1 at :', time.ctime()) 7 8 def loop2(): 9 print("Start loop 2 at :", time.ctime())10 time.sleep(2)11 print('End loop 2 at :', time.ctime())12 13 def loop3():14 print("Start loop 3 at :",time.ctime())15 time.sleep(5)16 print("End loop 3 at :", time.ctime())17 18 19 def main():20 print("Starting at :", time.ctime())21 # 生成treading.Thread实例22 t1 = threading.Thread(target=loop1, args=( ))23 t1.setName("THR_1") # 给线程设置名字24 t1.start()25 t2 = threading.Thread(target=loop2, args=( ))26 t2.setName("THR_2") # 给线程设置名字27 t2.start()28 t3 = threading.Thread(target=loop3, args=( ))29 t3.setName("THR_3") # 给线程设置名字30 t3.start()31 32 # 预期三秒后,thread2已经自动结束33 time.sleep(3)34 # enumerate 得到正在运行子线程,即子线程1和子线程335 for thr in threading.enumerate():36 print("正在运行的线程名字是: {0}".format(thr.getName()))37 38 print("正在运行的子线程数量为: {0}".format(threading.activeCount()))39 print("All done at", time.ctime())40 if __name__ == '__main__':41 main()42 # 一定要有while语句43 # 因为启动多线程后本程序就作为主程序存在44 # 如果主线程执行完毕,则子线程可能也需要终止45 while True:46 time.sleep(10)