Python自己带有logging模块,其默认支持直接输出到控制台(屏幕),或者经过配置输出到文件中。同时支持TCP、HTTP、GET/POST、SMTP、Socket等协议,将日志信息发送到网络等等。python
日志级别大小关系为:CRITICAL > ERROR > WARNING > INFO > DEBUG > NOTSET,固然也能够本身定义日志级别。服务器
没有配置logging时,日志会直接输出到控制台网络
import logging if __name__ == "__main__": logging.debug("hello debug") logging.info("hello info") logging.warning("hello warning") logging.error("hello error")
输出结果:socket
WARNING:root:hello warning ERROR:root:hello error
是否是少了上面两个? 由于默认状况下,logging的级别为warning.函数
经过logging.basicConfig函数作相关配置:.net
import logging logging.basicConfig(level=logging.DEBUG, format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s',) if __name__ == "__main__": logging.debug("hello debug") logging.info("hello info") logging.warning("hello warning") logging.error("hello error")
输出为:线程
2017-03-28 12:22:55,052 test.py[line:8] DEBUG hello debug 2017-03-28 12:22:55,053 test.py[line:9] INFO hello info 2017-03-28 12:22:55,053 test.py[line:10] WARNING hello warning 2017-03-28 12:22:55,054 test.py[line:11] ERROR hello error
能够指定日期格式。python中时间日期格式化符号:
>%y 两位数的年份表示(00-99)
%Y 四位数的年份表示(000-9999)
%m 月份(01-12)
%d 月内中的一天(0-31)
%H 24小时制小时数(0-23)
%I 12小时制小时数(01-12)
%M 分钟数(00=59)
%S 秒(00-59)
%a 本地简化星期名称
%A 本地完整星期名称
%b 本地简化的月份名称
%B 本地完整的月份名称
%c 本地相应的日期表示和时间表示
%j 年内的一天(001-366)
%p 本地A.M.或P.M.的等价符
%U 一年中的星期数(00-53)星期天为星期的开始
%w 星期(0-6),星期天为星期的开始
%W 一年中的星期数(00-53)星期一为星期的开始
%x 本地相应的日期表示
%X 本地相应的时间表示
%Z 当前时区的名称
%% %号自己debug
代码修改以下:日志
import logging logging.basicConfig(level=logging.DEBUG, format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s', datefmt='%Y-%m-%d %A %H:%M:%S') if __name__ == "__main__": logging.debug("hello debug"); logging.info("hello info"); logging.warning("hello warning"); logging.error("hello error");
经过logging.basicConfig来配置输出文件路径:code
import logging logging.basicConfig(level=logging.DEBUG, format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s', datefmt='%Y-%m-%d %A %H:%M:%S', filename='python.log', filemode='w') if __name__ == "__main__": logging.debug("hello debug"); logging.info("hello info"); logging.warning("hello warning"); logging.error("hello error");
输出结果:
2017-03-28 Tuesday 12:33:29 test.py[line:10] DEBUG hello debug 2017-03-28 Tuesday 12:33:29 test.py[line:11] INFO hello info 2017-03-28 Tuesday 12:33:29 test.py[line:12] WARNING hello warning 2017-03-28 Tuesday 12:33:29 test.py[line:13] ERROR hello error
format: 指定输出的格式和内容,format能够输出不少有用信息,如上例所示:
%(levelno)s: 打印日志级别的数值
%(levelname)s: 打印日志级别名称
%(pathname)s: 打印当前执行程序的路径,其实就是sys.argv[0]
%(filename)s: 打印当前执行程序名
%(funcName)s: 打印日志的当前函数
%(lineno)d: 打印日志的当前行号
%(asctime)s: 打印日志的时间
%(thread)d: 打印线程ID
%(threadName)s: 打印线程名称
%(process)d: 打印进程ID
%(message)s: 打印日志信息
stream: 指定将日志的输出流,能够指定输出到sys.stderr,sys.stdout或者文件,默认输出到sys.stderr,当stream和filename同时指定时,stream被忽略。
经过handler来让日志能够即输出到文件中又输出到控制台上。
import logging logging.basicConfig(level=logging.DEBUG, format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s', datefmt='%Y-%m-%d %A %H:%M:%S', filename='python.log', filemode='w') console = logging.StreamHandler() console.setLevel(logging.INFO) formatter = logging.Formatter('%(levelname)-8s %(message)s') console.setFormatter(formatter) logging.getLogger('').addHandler(console) if __name__ == "__main__": logging.debug("hello debug"); logging.info("hello info"); logging.warning("hello warning"); logging.error("hello error");
控制台输出内容:
INFO hello info WARNING hello warning ERROR hello error
文件中输出内容:
2017-03-28 Tuesday 12:38:19 test.py[line:16] DEBUG hello debug 2017-03-28 Tuesday 12:38:19 test.py[line:17] INFO hello info 2017-03-28 Tuesday 12:38:19 test.py[line:18] WARNING hello warning 2017-03-28 Tuesday 12:38:19 test.py[line:19] ERROR hello error
当日志一直向一个文件中输出,会致使文件太大。因此有时候但愿日志文件滚动生成多个文件。
import logging from logging.handlers import RotatingFileHandler logging.basicConfig(level=logging.DEBUG, format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s', datefmt='%Y-%m-%d %A %H:%M:%S') rfhandler = RotatingFileHandler('python.log', maxBytes=10*1024*1024, backupCount=5) rfhandler.setLevel(logging.INFO) formatter = logging.Formatter('%(asctime)s %(levelname)-8s %(message)s') rfhandler.setFormatter(formatter) logging.getLogger('').addHandler(rfhandler) if __name__ == "__main__": logging.debug("hello debug"); logging.info("hello info"); logging.warning("hello warning"); logging.error("hello error");
当文件输出超过1010241024=10M大小时,就会生成一个新的日志文件。
RotatingFileHandler的参数:
logging的包括以下几种handler方式:
因为StreamHandler和FileHandler是经常使用的日志处理方式,因此直接包含在logging模块中,而其余方式则包含在logging.handlers模块中,
还有一种方式是不在代码中对logging进行配置,而是写到配置文件中。
#logger.conf ############################ [loggers] keys=root,example01,example02 [logger_root] level=DEBUG handlers=hand01,hand02 [logger_example01] handlers=hand01,hand02 qualname=example01 propagate=0 [logger_example02] handlers=hand01,hand03 qualname=example02 propagate=0 ############################ [handlers] keys=hand01,hand02,hand03 [handler_hand01] class=StreamHandler level=INFO formatter=form02 args=(sys.stderr,) [handler_hand02] class=FileHandler level=DEBUG formatter=form01 args=('python.log', 'a') [handler_hand03] class=handlers.RotatingFileHandler level=INFO formatter=form02 args=('python.log', 'a', 10*1024*1024, 5) ############################ [formatters] keys=form01,form02 [formatter_form01] format=%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s datefmt=%a, %d %b %Y %H:%M:%S [formatter_form02] format=%(name)-12s: %(levelname)-8s %(message)s datefmt=
在python使用以下:
import logging import logging.config logging.config.fileConfig("logger.conf") logger = logging.getLogger("example01") if __name__ == "__main__": logger.debug("hello debug"); logger.info("hello info"); logger.warning("hello warning"); logger.error("hello error");
日志的输出格式format:
%(levelno)s: 打印日志级别的数值
%(levelname)s: 打印日志级别名称
%(pathname)s: 打印当前执行程序的路径,其实就是sys.argv[0]
%(filename)s: 打印当前执行程序名
%(funcName)s: 打印日志的当前函数
%(lineno)d: 打印日志的当前行号
%(asctime)s: 打印日志的时间
%(thread)d: 打印线程ID
%(threadName)s: 打印线程名称
%(process)d: 打印进程ID
%(message)s: 打印日志信息
工做中给的经常使用格式是:
format='%(asctime)s - %(filename)s[line:%(lineno)d] - %(levelname)s: %(message)s'
这个格式能够输出日志的打印时间,是哪一个模块输出的,输出的日志级别是什么,以及输入的日志内容。
@待续
参考: http://blog.csdn.net/yatere/article/details/6655445 http://blog.csdn.net/z_johnny/article/details/50812878 http://blog.csdn.net/liuchunming033/article/details/39080457