草庐IT

php - 在自定义记录器中打印变量时排除对象

coder 2024-04-18 原文

我有记录所有内容的 Logger-Class。对象将使用 print_r 记录到人类可读的状态。我的问题是我有一个很大的 MVC 对象。每次发生 Exception 或 Error 时,MVC-Object 也会被 print_r 打印到 Log 中。这会导致非常非常长的日志文件,阅读起来并不友好。

我试图为我的 MVC 类设置一个 __toString() 方法,但这不起作用。我还在日志中获得了完整的 MVC 对象。 MVC 是单例,在每个对象上都被引用。因此,在 Object 进入 print_r 之前简单地排除它并不是那么容易。

有没有办法从 print_r 中排除对象?

我的方法:

日志类错误处理程序方法:

public static function errorHandler($errno, $errstr, $errfile, $errline, $vars) {
                //If @ is set, don't do anything!
                if(error_reporting() === 0) {
                        return;
                }

                //Get StackTrace with low memory usage ;)
                $e = new Exception();
                $stackStr = $e->getTraceAsString();

                //Build ErrorMessage for Log
                $message =      'File: '.$errfile.' - L: '.$errline."\n".
                                'Code: '.$errno."\n".
                                'Message: '.$errstr."\n".
                                'Vars: '.print_r($vars, true)."\n".
                                'Stacktrace: '.$stackStr;
                self::error($message);
        }

日志类异常处理程序方法:

public static function exceptionHandler(Exception $e) {
        $message =      get_class($e).': '.$e->getMessage()."\n".
                        'File: '.$e->getFile().' - L: '.$e->getLine()."\n".
                        'Code: '.$e->getCode()."\n".
                        'Message: '.$e->getMessage()."\n".
                        'Stacktrace: '.$e->getTraceAsString();
        self::error($message);
}

日志类错误方法:

public static function error($data, $file='system.log') {
        $config = Megaira_PropertyConfiguration::getInstance();
        switch($config->get('LOG_MODE')) {
                case'DEEPDEBUG':
                case'ERROR':
                case'WARNING':
                case'INFO':
                case'DEBUG':
                        self::writeToLog('ERROR', $data, $file);
                        break;
        }
}

日志类 writeToLog 方法:

private static function writeToLog($mode='', $text='', $file=''){
        if(!is_string($text) && !is_numeric($text)) {
                $text = print_r($text, true);
        }
        $config = Megaira_PropertyConfiguration::getInstance();
        if(!$config->get('LOGGINGACTIVE')) { return; }
        self::writeLineToFile($mode, $text, $file);
}

设置错误和异常处理程序:

        //Set Error and Exception Handler
        set_error_handler(array(new LOG(), 'errorHandler'));
        set_exception_handler(array(new LOG(), 'exceptionHandler'));

谢谢

一些测试:

public static function print_r_filtered($object, $ret=false) {
                $filtered = array(
                        'Megaira_MVC'
                );
                $text = print_r($object, true);
                foreach($filtered as $filter) {
                        $search = '#('.$filter.'\sObject)\n(\s+)\).*?\n\2\)\n#s';
                        $replace = "$1";
                        $text = preg_replace($search, $replace, $text);
                }
                if($ret)
                        return $text;
                echo $text;
        }

没用。也许 RegEx 失败了?

解决方法:

这是一个设计缺陷。 errorHandler 正在记录在发生错误的地方使用的所有对象。所以在index.php中有如下代码:

$mvc = Megaira_MVC::getInstance();

因此,这种和平的代码通过 LOG-Class 中的 errorHandler 生成了带有 print_r 的 Var $mvc 日志。

我的结论:不要在大型​​单例对象上使用变量,如果不再需要 Var,则不要使用 unset()。

最佳答案

__toString() 在将对象转换为字符串时调用。你可以尝试类似的东西

$objectString = method_exists($object, '__toString')
              ? (string) $object
              : print_r($object, true);

使用 is_object() 判断值是否为对象。

完全

$string = !is_object($value) || method_exists($value, '__toString')
        ? (string) $value
        : print_r($value, true);

关于php - 在自定义记录器中打印变量时排除对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6192292/

有关php - 在自定义记录器中打印变量时排除对象的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  3. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  4. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  5. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  6. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

    我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

  7. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  8. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  9. Ruby 写入和读取对象到文件 - 2

    好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

  10. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

随机推荐