目录
$title Error
首页 php教程 php手册 PHP系统异常处理类程序

PHP系统异常处理类程序

May 25, 2016 pm 04:47 PM
catch explode fopen foreach substr

以前我们用过的异常处理函数都是单个的,下面我找到一个非常的不错的异常处理类系统,不但可以控制错误还能给出好的界面哦。

<?php
// 自定义异常函数
set_exception_handler(&#39;handle_exception&#39;);
// 自定义错误函数
set_error_handler(&#39;handle_error&#39;);
/**
 * 异常处理
 *
 * @param mixed $exception 异常对象
 * @author www.phprm.com
 */
function handle_exception($exception) {
    Error::exceptionError($exception);
}
/**
 * 错误处理
 *
 * @param string $errNo 错误代码
 * @param string $errStr 错误信息
 * @param string $errFile 出错文件
 * @param string $errLine 出错行
 * @author www.phprm.com
 */
function handle_error($errNo, $errStr, $errFile, $errLine) {
    if ($errNo) {
        Error::systemError($errStr, false, true, false);
    }
}
/**
 * 系统错误处理
 *
 * @author http://www.phprm.com
 */
class Error {
    public static function systemError($message, $show = true, $save = true, $halt = true) {
        list($showTrace, $logTrace) = self::debugBacktrace();
        if ($save) {
            $messageSave = &#39;<b>&#39; . $message . &#39;</b><br /><b>PHP:</b>&#39; . $logTrace;
            self::writeErrorLog($messageSave);
        }
        if ($show) {
            self::showError(&#39;system&#39;, "<li>$message</li>", $showTrace, 0);
        }
        if ($halt) {
            exit();
        } else {
            return $message;
        }
    }
    /**
     * 代码执行过程回溯信息
     *
     * @static
     * @access public
     */
    public static function debugBacktrace() {
        $skipFunc[] = &#39;Error->debugBacktrace&#39;;
        $show = $log = &#39;&#39;;
        $debugBacktrace = debug_backtrace();
        ksort($debugBacktrace);
        foreach ($debugBacktrace as $k => $error) {
            if (!isset($error[&#39;file&#39;])) {
                // 利用反射API来获取方法/函数所在的文件和行数
                try {
                    if (isset($error[&#39;class&#39;])) {
                        $reflection = new ReflectionMethod($error[&#39;class&#39;], $error[&#39;function&#39;]);
                    } else {
                        $reflection = new ReflectionFunction($error[&#39;function&#39;]);
                    }
                    $error[&#39;file&#39;] = $reflection->getFileName();
                    $error[&#39;line&#39;] = $reflection->getStartLine();
                }
                catch(Exception $e) {
                    continue;
                }
            }
            $file = str_replace(SITE_PATH, &#39;&#39;, $error[&#39;file&#39;]);
            $func = isset($error[&#39;class&#39;]) ? $error[&#39;class&#39;] : &#39;&#39;;
            $func.= isset($error[&#39;type&#39;]) ? $error[&#39;type&#39;] : &#39;&#39;;
            $func.= isset($error[&#39;function&#39;]) ? $error[&#39;function&#39;] : &#39;&#39;;
            if (in_array($func, $skipFunc)) {
                break;
            }
            $error[&#39;line&#39;] = sprintf(&#39;%04d&#39;, $error[&#39;line&#39;]);
            $show.= &#39;<li>[Line: &#39; . $error[&#39;line&#39;] . &#39;]&#39; . $file . &#39;(&#39; . $func . &#39;)</li>&#39;;
            $log.= !empty($log) ? &#39; -> &#39; : &#39;&#39;;
            $log.= $file . &#39;:&#39; . $error[&#39;line&#39;];
        }
        return array(
            $show,
            $log
        );
    }
    /**
     * 异常处理
     *
     * @static
     * @access public
     * @param mixed $exception
     */
    public static function exceptionError($exception) {
        if ($exception instanceof DbException) {
            $type = &#39;db&#39;;
        } else {
            $type = &#39;system&#39;;
        }
        if ($type == &#39;db&#39;) {
            $errorMsg = &#39;(&#39; . $exception->getCode() . &#39;) &#39;;
            $errorMsg.= self::sqlClear($exception->getMessage() , $exception->getDbConfig());
            if ($exception->getSql()) {
                $errorMsg.= &#39;<div class="sql">&#39;;
                $errorMsg.= self::sqlClear($exception->getSql() , $exception->getDbConfig());
                $errorMsg.= &#39;</div>&#39;;
            }
        } else {
            $errorMsg = $exception->getMessage();
        }
        $trace = $exception->getTrace();
        krsort($trace);
        $trace[] = array(
            &#39;file&#39; => $exception->getFile() ,
            &#39;line&#39; => $exception->getLine() ,
            &#39;function&#39; => &#39;break&#39;
        );
        $phpMsg = array();
        foreach ($trace as $error) {
            if (!empty($error[&#39;function&#39;])) {
                $fun = &#39;&#39;;
                if (!empty($error[&#39;class&#39;])) {
                    $fun.= $error[&#39;class&#39;] . $error[&#39;type&#39;];
                }
                $fun.= $error[&#39;function&#39;] . &#39;(&#39;;
                if (!empty($error[&#39;args&#39;])) {
                    $mark = &#39;&#39;;
                    foreach ($error[&#39;args&#39;] as $arg) {
                        $fun.= $mark;
                        if (is_array($arg)) {
                            $fun.= &#39;Array&#39;;
                        } elseif (is_bool($arg)) {
                            $fun.= $arg ? &#39;true&#39; : &#39;false&#39;;
                        } elseif (is_int($arg)) {
                            $fun.= (defined(&#39;SITE_DEBUG&#39;) && SITE_DEBUG) ? $arg : &#39;%d&#39;;
                        } elseif (is_float($arg)) {
                            $fun.= (defined(&#39;SITE_DEBUG&#39;) && SITE_DEBUG) ? $arg : &#39;%f&#39;;
                        } else {
                            $fun.= (defined(&#39;SITE_DEBUG&#39;) && SITE_DEBUG) ? &#39;&#39;&#39; . htmlspecialchars(substr(self::clear($arg), 0, 10)) . (strlen($arg) > 10 ? &#39; . . . &#39; : &#39;&#39;) . &#39;&#39;&#39; : &#39;%s&#39;;
                        }
                        $mark = &#39;, &#39;;
                    }
                }
                $fun.= &#39;)&#39;;
                $error[&#39;function&#39;] = $fun;
            }
            if (!isset($error[&#39;line&#39;])) {
                continue;
            }
            $phpMsg[] = array(
                &#39;file&#39; => str_replace(array(
                    SITE_PATH,
                    &#39;&#39;
                ) , array(
                    &#39;&#39;,
                    &#39;/&#39;
                ) , $error[&#39;file&#39;]) ,
                &#39;line&#39; => $error[&#39;line&#39;],
                &#39;function&#39; => $error[&#39;function&#39;]
            );
        }
        self::showError($type, $errorMsg, $phpMsg);
        exit();
    }
    /**
     * 记录错误日志
     *
     * @static
     * @access public
     * @param string $message
     */
    public static function writeErrorLog($message) {
        return false; // 暂时不写入 http://www.phprm.com
        $message = self::clear($message);
        $time = time();
        $file = LOG_PATH . &#39;/&#39; . date(&#39;Y.m.d&#39;) . &#39;_errorlog.php&#39;;
        $hash = md5($message);
        $userId = 0;
        $ip = get_client_ip();
        $user = &#39;<b>User:</b> userId=&#39; . intval($userId) . &#39;; IP=&#39; . $ip . &#39;; RIP:&#39; . $_SERVER[&#39;REMOTE_ADDR&#39;];
        $uri = &#39;Request: &#39; . htmlspecialchars(self::clear($_SERVER[&#39;REQUEST_URI&#39;]));
        $message = "<?php exit;?> {$time} $message $hash $user $uri ";
        // 判断该$message是否在时间间隔$maxtime内已记录过,有,则不用再记录了
        if (is_file($file)) {
            $fp = @fopen($file, &#39;rb&#39;);
            $lastlen = 50000; // 读取最后的 $lastlen 长度字节内容
            $maxtime = 60 * 10; // 时间间隔:10分钟
            $offset = filesize($file) - $lastlen;
            if ($offset > 0) {
                fseek($fp, $offset);
            }
            if ($data = fread($fp, $lastlen)) {
                $array = explode(" ", $data);
                if (is_array($array)) foreach ($array as $key => $val) {
                    $row = explode(" ", $val);
                    if ($row[0] != &#39;<?php exit;?>&#39;) {
                        continue;
                    }
                    if ($row[3] == $hash && ($row[1] > $time - $maxtime)) {
                        return;
                    }
                }
            }
        }
        error_log($message, 3, $file);
    }
    /**
     * 清除文本部分字符
     *
     * @param string $message
     */
    public static function clear($message) {
        return str_replace(array(
            " ",
            " ",
            " "
        ) , " ", $message);
    }
    /**
     * sql语句字符清理
     *
     * @static
     * @access public
     * @param string $message
     * @param string $dbConfig
     */
    public static function sqlClear($message, $dbConfig) {
        $message = self::clear($message);
        if (!(defined(&#39;SITE_DEBUG&#39;) && SITE_DEBUG)) {
            $message = str_replace($dbConfig[&#39;database&#39;], &#39;***&#39;, $message);
            //$message = str_replace($dbConfig[&#39;prefix&#39;], &#39;***&#39;, $message);
            $message = str_replace(C(&#39;DB_PREFIX&#39;) , &#39;***&#39;, $message);
        }
        $message = htmlspecialchars($message);
        return $message;
    }
    /**
     * 显示错误
     *
     * @static
     * @access public
     * @param string $type 错误类型 db,system
     * @param string $errorMsg
     * @param string $phpMsg
     */
    public static function showError($type, $errorMsg, $phpMsg = &#39;&#39;) {
        global $_G;
        $errorMsg = str_replace(SITE_PATH, &#39;&#39;, $errorMsg);
        ob_end_clean();
        $host = $_SERVER[&#39;HTTP_HOST&#39;];
        $title = $type == &#39;db&#39; ? &#39;Database&#39; : &#39;System&#39;;
        echo <<<EOT
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html>
<head>
 <title>$host - $title Error</title>
 <meta http-equiv="Content-Type" content="text/html; charset={$_G[&#39;config&#39;][&#39;output&#39;][&#39;charset&#39;]}" />
 <meta name="ROBOTS" content="NOINDEX,NOFOLLOW,NOARCHIVE" />
 <style type="text/css">
 <!--
 body { background-color: white; color: black; font: 9pt/11pt verdana, arial, sans-serif;}
 #container {margin: 10px;}
 #message {width: 1024px; color: black;}
 .red {color: red;}
 a:link {font: 9pt/11pt verdana, arial, sans-serif; color: red;}
 a:visited {font: 9pt/11pt verdana, arial, sans-serif; color: #4e4e4e;}
 h1 {color: #FF0000; font: 18pt "Verdana"; margin-bottom: 0.5em;}
 .bg1 {background-color: #FFFFCC;}
 .bg2 {background-color: #EEEEEE;}
 .table {background: #AAAAAA; font: 11pt Menlo,Consolas,"Lucida Console"}
 .info {
  background: none repeat scroll 0 0 #F3F3F3;
  border: 0px solid #aaaaaa;
  border-radius: 10px 10px 10px 10px;
  color: #000000;
  font-size: 11pt;
  line-height: 160%;
  margin-bottom: 1em;
  padding: 1em;
 }
 .help {
  background: #F3F3F3;
  border-radius: 10px 10px 10px 10px;
  font: 12px verdana, arial, sans-serif;
  text-align: center;
  line-height: 160%;
  padding: 1em;
 }
 .sql {
  background: none repeat scroll 0 0 #FFFFCC;
  border: 1px solid #aaaaaa;
  color: #000000;
  font: arial, sans-serif;
  font-size: 9pt;
  line-height: 160%;
  margin-top: 1em;
  padding: 4px;
 }
 -->
 </style>
</head>
<body>
<div id="container">
<h1 id="title-nbsp-Error">$title Error</h1>
<div class=&#39;info&#39;>$errorMsg</div>
EOT;
        if (!empty($phpMsg)) {
            echo &#39;<div class="info">&#39;;
            echo &#39;<p><strong>PHP Debug</strong></p>&#39;;
            echo &#39;<table cellpadding="5" cellspacing="1" width="100%" class="table"><tbody>&#39;;
            if (is_array($phpMsg)) {
                echo &#39;<tr class="bg2"><td>No.</td><td>File</td><td>Line</td><td>Code</td></tr>&#39;;
                foreach ($phpMsg as $k => $msg) {
                    $k++;
                    echo &#39;<tr class="bg1">&#39;;
                    echo &#39;<td>&#39; . $k . &#39;</td>&#39;;
                    echo &#39;<td>&#39; . $msg[&#39;file&#39;] . &#39;</td>&#39;;
                    echo &#39;<td>&#39; . $msg[&#39;line&#39;] . &#39;</td>&#39;;
                    echo &#39;<td>&#39; . $msg[&#39;function&#39;] . &#39;</td>&#39;;
                    echo &#39;</tr>&#39;;
                }
            } else {
                echo &#39;<tr><td><ul>&#39; . $phpMsg . &#39;</ul></td></tr>&#39;;
            }
            echo &#39;</tbody></table></div>&#39;;
        }
        echo <<<EOT
</div>
</body>
</html>
EOT;
        exit();
    }
}
/**
 * DB异常类
 *
 * @author www.phprm.com
 */
class DbException extends Exception {
    protected $sql;
    protected $dbConfig; // 当前数据库配置信息
    public function __construct($message, $code = 0, $sql = &#39;&#39;, $dbConfig = array()) {
        $this->sql = $sql;
        $this->dbConfig = $dbConfig;
        parent::__construct($message, $code);
    }
    public function getSql() {
        return $this->sql;
    }
    public function getDbConfig() {
        return $this->dbConfig;
    }
}
?>
登录后复制

效果图:PHP系统异常处理类程序


本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

<🎜>:泡泡胶模拟器无穷大 - 如何获取和使用皇家钥匙
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系统,解释
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆树的耳语 - 如何解锁抓钩
3 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

热门话题

Java教程
1667
14
CakePHP 教程
1426
52
Laravel 教程
1328
25
PHP教程
1273
29
C# 教程
1255
24
Java ArrayList遍历时使用foreach和iterator删除元素的区别是什么? Java ArrayList遍历时使用foreach和iterator删除元素的区别是什么? Apr 27, 2023 pm 03:40 PM

一、Iterator和foreach的区别多态差别(foreach底层就是Iterator)Iterator是一个接口类型,他不关心集合或者数组的类型;for和foreach都需要先知道集合的类型,甚至是集合内元素的类型;1.为啥说foreach底层就是Iterator编写的代码:反编译代码:二、foreach与iterator时remove的区别先来看阿里java开发手册但1的时候不会报错,2的时候就会报错(java.util.ConcurrentModificationException)首

php如何判断foreach循环到第几个 php如何判断foreach循环到第几个 Jul 10, 2023 pm 02:18 PM

​php判断foreach循环到第几个的步骤:1、创建一个“$fruits”的数组;2、创建一个计数器变量“$counter”初始值为0;3、使用“foreach”循环遍历数组,并在循环体中增加计数器变量的值,再输出每个元素和它们的索引;4、在“foreach”循环体外输出计数器变量的值,以确认循环到了第几个元素。

如何解决PHP Warning: fopen(): SSL operation failed in file.php on line X 如何解决PHP Warning: fopen(): SSL operation failed in file.php on line X Aug 25, 2023 am 09:22 AM

如何解决PHPWarning:fopen():SSLoperationfailedinfile.phponlineX在PHP编程中,我们经常使用fopen函数来打开文件或者URL,并进行相关操作。然而,在使用fopen函数时,有时候会遇到类似于Warning:fopen():SSLoperationfailedinfile.p

如何解决PHP Warning: fopen(): failed to open stream: Permission denied 如何解决PHP Warning: fopen(): failed to open stream: Permission denied Aug 20, 2023 pm 01:45 PM

如何解决PHPWarning:fopen():failedtoopenstream:Permissiondenied在开发PHP程序的过程中,我们常常会遇到一些报错信息,比如PHPWarning:fopen():failedtoopenstream:Permissiondenied。这个错误通常是由于文件或目录权限不正

如何解决PHP Warning: fopen(): failed to open stream: No such file or directory 如何解决PHP Warning: fopen(): failed to open stream: No such file or directory Aug 19, 2023 am 10:44 AM

如何解决PHPWarning:fopen():failedtoopenstream:Nosuchfileordirectory在使用PHP开发过程中,我们经常会遇到一些文件操作的问题,其中之一就是"PHPWarning:fopen():failedtoopenstream:Nosuchfileordirectory

PHP返回一个键值翻转后的数组 PHP返回一个键值翻转后的数组 Mar 21, 2024 pm 02:10 PM

这篇文章将为大家详细讲解有关PHP返回一个键值翻转后的数组,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。PHP键值翻转数组键值翻转是一种对数组进行的操作,它将数组中的键和值进行交换,生成一个新的数组,其中原始键作为值,原始值作为键。实现方法在php中,可以通过以下方法对数组进行键值翻转:array_flip()函数:array_flip()函数专门用于键值翻转操作。它接收一个数组作为参数,并返回一个新的数组,其中键和值已交换。$original_array=[

Matlab中fopen函数用法 Matlab中fopen函数用法 Nov 28, 2023 am 11:03 AM

在Matlab中,fopen函数用于打开文件并返回文件标识符,以便后续对文件进行读取或写入操作。根据需要选择适当的权限选项来打开文件,并在操作完成后及时关闭文件。需要注意的是,打开文件后需要确保在不再需要文件时及时关闭文件,以释放系统资源。另外,如果文件打开失败或操作出错,可以通过错误处理机制进行相应的处理。

PHP explode函数使用方法与报错解决 PHP explode函数使用方法与报错解决 Mar 10, 2024 am 09:18 AM

PHP中的explode函数是一种用来将字符串分割成数组的函数,它非常常用且灵活。在使用explode函数的过程中,常常会遇到一些报错和问题,本文将介绍explode函数的基本用法并提供一些解决报错的方法。一、explode函数基本用法在PHP中,explode函数的基本语法如下:explode(string$separator,string$stri

See all articles