PHP滚动日志的代码实现


Posted in PHP onJune 10, 2015

PHP滚动日志类库

PHP记录日志,我之前接触过的有按照年月分文件夹,然后按照日分文件的日志记录方式,这种方式有利有弊,有他的使用场景,我今天要说的是另一种日志记录方式——文件滚动方式记录日志,当然了,这种滚动机制也可以加在前面那种日志记录方式中。

如何让日志滚动起来

滚动日志,顾名思义,记录一个模块的日志用一系列的日志文件,同一模块文件个数有限制,最多maxNum个,大小也有限制,最大maxSize字节,文件名有一定的命名方式,比如:testlog.log、testlog_1.log,testlog_2.log、、、、、、其中testlog.log是正在使用的日志文件,当testlog.log文件大小到达限制maxSize的时候就会向后滚动日志文件,就像下面这样:

testlog_2.log  -> testlog_3.log

testlog_1.log  -> testlog_2.log

testlog.log  -> testlog_1.log

testlog.log #0kb

当日志文件个数到达限制maxNum的时候就会启动淘汰机制,删除最老的日志,比如说maxNum设置为10,这个时候算上testlog.log一共最多有10个文件,当滚动的时候如果存在testlog_9.log就会从testlog_8.log开始滚动,覆盖掉testlog_9.log,这样就可以保证日志正常记录,而且不会出现很大很大的日志文件,保证日志系统的正常运行。

代码实现

<?php
final class LOGS {
 private $level;
 private $maxFileNum;
 private $maxFileSize;
 private $logPath;
 private $file;

 //日志的级别DEBUG,MSG,ERR
 const LOGS_DEBUG = 0;
 const LOGS_MSG = 1;
 const LOGS_ERR = 2;

 private static $instance = null;

 private function __construct(){}

 public static function getInstance()
 {
 if(self::$instance == null)
 {
  self::$instance = new self();
 }
 return self::$instance;
 }

 /**
 * @Desc 初始化
 * @Param $level int 记录级别
 * @Param $maxNum int 最大日志文件数目
 * @Param $maxSize int 最大日志文件大小
 * @Param $logPath string 日志文件保存路径
 * @Param $file string 日志文件名称前缀
 * @Return boolean
 */
 public function init($level, $maxNum, $maxSize, $logPath, $file)
 {
 $level = intval($level);
 $maxNum = intval($maxNum);
 $maxSize = intval($maxSize);
 !is_dir($logPath) && mkdir($logPath, 0777, true);
 if(!in_array($level, array(self::LOGS_DEBUG, self::LOGS_MSG, self::LOGS_ERR)) || $maxNum <= 0 || $maxSize <= 0 || !is_dir($logPath))
 {
  return false;
 }
 $this->level = $level;
 $this->maxFileNum = $maxNum;
 $this->maxFileSize = $maxSize;
 $this->logPath = $logPath;
 $this->file = $file;
 return true;
 }

 /**
 * @Desc 获取格式化时间串
 */
 public function formatTime()
 {
    $ustime = explode ( " ", microtime () );
    return "[" . date('Y-m-d H:i:s', time()) .".". ($ustime[0] * 1000) . "]";
 }

 /** 
 * @Desc 滚动方式记录日志文件
 */
 public function log($str)
 {
 $path = $this->logPath.DIRECTORY_SEPARATOR.$this->file.".log";
 clearstatcache();
 if(file_exists($path))
 {
  if(filesize($path) >= $this->maxFileSize)
  {
  $index = 1;
  //获取最大的滚动日志数目
  for(;$index < $this->maxFileNum; $index++)
  {
   if(!file_exists($this->logPath.DIRECTORY_SEPARATOR.$this->file."_".$index.".log"))
   {
   break;
   }
  }
  //已经存在maxFileNum个日志文件了
  if($index == $this->maxFileNum)
  {
   $index--;
  }
  //滚动日志
  for(;$index > 1; $index--)
  {
   $new = $this->logPath.DIRECTORY_SEPARATOR.$this->file."_".$index.".log";
   $old = $this->logPath.DIRECTORY_SEPARATOR.$this->file."_".($index - 1).".log";
   rename($old, $new);
  }

  $newFile = $this->logPath.DIRECTORY_SEPARATOR.$this->file."_1.log";
  rename($path, $newFile);
  }
 }
 $fp = fopen($path, "a+b");
 fwrite($fp, $str, strlen($str));
 fclose($fp);
 return true;
 }

 /**
 * @Desc 记录调试信息
 * @Param string 日志信息
 * @Param string 日志所在文件
 * @Param string 日志所在行
 */
 public function debug($msg, $file, $line)
 {
 if($this->level <= self::LOGS_DEBUG)
 {
  $this->log($this->formatTime()."[{$file}:{$line}]DEBUG: ${msg}\n");
 }
 }

 /**
 * @Desc 记录信息
 * @Param string 日志信息
 * @Param string 日志所在文件
 * @Param string 日志所在行
 */
 public function msg($msg, $file, $line)
 {
 if($this->level <= self::LOGS_MSG)
 {
  $this->log($this->formatTime()."[{$file}:{$line}]MSG: ${msg}\n");
 }
 }

 /**
 * @Desc 记录错误信息
 * @Param string 日志信息
 * @Param string 日志所在文件
 * @Param string 日志所在行
 */
 public function err($msg, $file, $line)
 {
 if($this->level <= self::LOGS_ERR)
 {
  $this->log($this->formatTime()."[{$file}:{$line}]ERR: ${msg}\n");
 }
 }
}

看个例子

#例子中设置记录级别为msg(此时debug信息是不会纪录的),日志文件个数为5,大小为200个字节(测试方便),文件名称为testlog

$logs = LOGS::getInstance();
$logs->init(1, 5, 200, "./", 'testlog');

$logs->msg("YRT", __FILE__, __LINE__);
$logs->debug("YRT", __FILE__, __LINE__);

当我们不停的运行这个例子的时候,会在代码所在文件夹下生成5个文件就像下面这样:

testlog_4.log
testlog_3.log
testlog_2.log
testlog_1.log
testlog.log  #最新的日志在这个文件中

以上所述就是本文的全部内容了,希望大家能够喜欢。

PHP 相关文章推荐
php创建多级目录代码
Jun 05 PHP
一个很不错的PHP翻页类
Jun 01 PHP
php csv操作类代码
Dec 14 PHP
支持中文的php加密解密类代码
Nov 27 PHP
深入探讨<br />和 \r\n两者有什么区别??
Jun 05 PHP
ThinkPHP文件上传实例教程
Aug 22 PHP
ThinkPHP模板替换与系统常量及应用实例教程
Aug 22 PHP
php实现数组中索引关联数据转换成json对象的方法
Jul 08 PHP
php中使用GD库做验证码
Mar 31 PHP
PHP实现将标点符号正则替换为空格的方法
Aug 09 PHP
PHP常见数组排序方法小结
Aug 20 PHP
PHP实现微信提现功能
Sep 30 PHP
简单谈谈favicon
Jun 10 #PHP
简单谈谈php中的unicode和utf8编码
Jun 10 #PHP
PHP中生成UUID自定义函数分享
Jun 10 #PHP
php使用for语句输出三角形的方法
Jun 09 #PHP
php生成图片验证码
Jun 09 #PHP
php判断用户是否手机访问代码
Jun 08 #PHP
浅谈PHP中Stream(流)
Jun 08 #PHP
You might like
php防止sql注入之过滤分页参数实例
2014/11/03 PHP
PHP浮点数的一个常见问题
2016/03/10 PHP
Web版彷 Visual Studio 2003 颜色选择器
2007/01/09 Javascript
比较详细的关于javascript 解析json的代码
2009/12/16 Javascript
2014最热门的JavaScript代码高亮插件推荐
2014/11/25 Javascript
js中document.write的那点事
2014/12/12 Javascript
AngularJS入门教程之数据绑定原理详解
2016/11/02 Javascript
微信小程序  action-sheet详解及实例代码
2016/11/09 Javascript
JavaScript利用闭包实现模块化
2017/01/13 Javascript
bootstrapValidator.min.js表单验证插件
2017/02/09 Javascript
vue项目实现github在线预览功能
2018/06/20 Javascript
Typescript的三种运行方式(小结)
2019/09/18 Javascript
[46:37]LGD vs TNC 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/16 DOTA
[03:18]【TI9纪实】社区大触GL与木木
2019/08/25 DOTA
在python中的socket模块使用代理实例
2014/05/29 Python
Python中给List添加元素的4种方法分享
2014/11/28 Python
python脚本设置超时机制系统时间的方法
2016/02/21 Python
Python selenium 三种等待方式解读
2016/09/15 Python
python安装PIL模块时Unable to find vcvarsall.bat错误的解决方法
2016/09/19 Python
浅谈Django学习migrate和makemigrations的差别
2018/01/18 Python
使用python实现http及ftp服务进行数据传输的方法
2018/10/26 Python
基于Python实现剪切板实时监控方法解析
2019/09/11 Python
flask框架json数据的拿取和返回操作示例
2019/11/28 Python
PyTorch中permute的用法详解
2019/12/30 Python
使用python计算三角形的斜边例子
2020/04/15 Python
全球性的在线鞋类品牌:Public Desire
2019/04/03 全球购物
美国踏板车和轻便摩托车销售网站:Mega Motor Madness
2020/02/26 全球购物
前台接待岗位职责
2013/12/03 职场文书
学习十八大报告感言
2014/02/04 职场文书
影视后期实训报告
2014/11/05 职场文书
2015年“世界无车日”活动方案
2015/05/06 职场文书
地道战观后感
2015/06/04 职场文书
2016秋季幼儿园开学寄语
2015/12/03 职场文书
导游词之贵州百里杜鹃
2019/10/29 职场文书
如何使用PostgreSQL进行中文全文检索
2021/05/27 PostgreSQL
python和C/C++混合编程之使用ctypes调用 C/C++的dll
2022/04/29 Python