MySQLI 扩展和本机错误处理

发布于 2024-11-24 16:21:49 字数 6204 浏览 2 评论 0原文

我正在尝试扩展 PHP5 MySQLi 库

class mysqli_ls extends mysqli

我扩展该类的原因是添加我需要的默认情况下未包含的功能和特性。对于我创建的函数,我希望以与任何本机函数完全相同的方式处理错误。

我希望使用该类的人不必确定这些函数是本机函数还是定制函数。

普通函数

if ( $_mysqli->query($sql) === false ) die($_mysqli->error);

所需的定制函数示例

if ( $_mysqli->run_config() === false ) die($_mysqli->error);

在我的扩展 mysqli 的类 mysqli_ls 中,我有以下函数,我希望这些函数能够在父 mysqli 类中设置错误消息。

   public function run_config()
   {
      // Open Debug FH
      if ($this->config['debug_log'] === TRUE)
      {
         if ($this->fh_debug = fopen($this->config['debug_log_path'], 'a') === FALSE) 
         {
            $this->handle_error('Could not open Debug file handle','200',__METHOD__);
            return $this->E_ERROR;
         }
      }
  }



private function handle_error($errormsg,$errorno,$method)
   {
      $this->errorno = $errorno;
      $this->error   = $errormsg;

      if ($this->config['debug_log'] === TRUE)
      {
         $msg = "($errorno) $errormsg";                              
         $this->write_log('error', $method, $msg);
      }

      return $this->E_OK;
   }

我已经尝试了上述方法,但出现错误: PHP 致命错误:mysqli_ls::handle_error():无法编写属性

和静态调用语法,这会导致语法错误,如下所示:

parent::error
parent::errorno 

我希望该类的用户能够无缝地拥有默认的 MySQLI 功能和扩展功能使用不同的方法来识别错误。

提前感谢您提供的任何建议或输入..

编辑:添加完整的类

class mysqli_ls extends mysqli
{
   private $config = array
   (
      'config_load'     => FALSE,
      'config_path'     => 'mysqli_ls.database.ini',

      'debug_log'       => FALSE,
      'debug_log_path'  => '/tmp/mysqli_ls.debug.log',

      'query_log'       => FALSE,
      'query_log_path'  => '/tmp/mysqli_ls.debug.log',

      'log_date_format' => 'Y-m-d H:i:s'
   );

   // Expected fields for the autoload file
   private $db_fields = array('hostname','username','password','database','port');

   // File Handles
   private $fh_debug  = FALSE;
   private $fh_query  = FALSE;
   private $fh_config = FALSE;

   // Return presets
   private $E_OK    = TRUE;
   private $E_ERROR = FALSE;

   // Existing database connections
   private $db_existing = array();

   public $error;
   public $errorno;

   ## ----------
   public function __construct()
   {
      if (!function_exists('mysqli_connect')) $this->handle_error('MySQLI is not installed','100');
   }

   ## ----------
   public function run_config()
   {
      // Testing error handling
      $this->handle_error('Could not open Debug file handle','200',__METHOD__);

      // Open Debug FH
      if ($this->config['debug_log'] === TRUE)
      {
         if ($this->fh_debug = fopen($this->config['debug_log_path'], 'a') === FALSE) 
         {
            $this->handle_error('Could not open Debug file handle','200',__METHOD__);
            return $this->E_ERROR;
         }
      }

      // Open Query FH
      if ($this->config['query_log'] === TRUE) 
      {
         if ($this->fh_query = fopen($this->config['query_log_path'], 'a') === FALSE)
         {
            $this->handle_error('Could not open Debug file handle','210');            
            return $this->E_ERROR;
         }
      }

      // Load database INI file
      if ($this->config['config_load'] === TRUE)
      {
         if ( ($this->db_existing = parse_ini_file($this->config['config_path'], TRUE)) === FALSE )
         {
            $this->handle_error('Could not parse the database ini file','220'); 
            return $this->E_ERROR;
         }
      }

      // Check for format of the loaded ini file
      foreach($this->db_existing as $name => $row)
      {
         foreach($this->db_fields as $field)
         {
            if ( !isset($row[$field]) ) 
            {
               $this->handle_error("Missing field ($field) in the config array for element ($name)",'230'); 
               return $this->E_ERROR;
            }
         } // END foreach
      } // END foreach

      return $this->E_OK;
   }

   ## ----------
   public function set_config($key, $value)
   {
      if ( !isset($this->config[$key]) ) 
      {
         $this->handle_error('Configuration variable ($key) does not exist','300');
         return $this->E_ERROR;
      }

      $this->config[$key] = $value;
      return $this->E_OK;
   }

   ## ----------
   public function get_config()
   {
      return array_merge($this->config, $this->db_existing);
   }

   ## ----------
   private function write_log($type,$method,$msg)
   {
      $msg = date($this->config['log_date_format']) ."\t". $type ."\t". $method ."\t". $msg ."\n";

      switch($type)
      {
         case 'error':
            fwrite($this->fh_debug, $msg);
         break;

         case 'debug':
            fwrite($this->fh_debug, $msg);
         break;

         case 'query':
            fwrite($this->fh_query, $msg);
         break;

         default:
            return $this->E_ERROR;
         break;      
      }
   }

   ## ----------
   private function handle_error($errormsg,$errorno,$method)
   {
      $this->errorno = $errorno;
      $this->error   = $errormsg;

      if ($this->config['debug_log'] === TRUE)
      {
         $msg = "($errorno) $errormsg";                              
         $this->write_log('error', $method, $msg);
      }

      return $this->E_OK;
   }

   ## ----------
   ## ----------
   ## ----------
   ## ----------      

} // END Class

测试脚本调用类

#!/usr/bin/php
<?php

require('mysqli_ls.class.php');

try
{
   $_mysqli = new mysqli_ls;
}
catch (Exception $e)
{
  print "\n\n". $e->getMessage() ."\n\n";
   print_r($e);
}

$_mysqli->set_config('debug_log',TRUE);
$_mysqli->set_config('query_log',TRUE);
$_mysqli->set_config('config_load',TRUE);

$_mysqli->run_config();
print_r($_mysqli->get_config());

?>

编辑:所以看起来 MySQLi 类中的错误变量可能是只读的..还有其他方法解决这个问题吗?我想也许有我自己的错误函数来检查父类和我自己的变量。

I am trying to extend the PHP5 MySQLi Library

class mysqli_ls extends mysqli

The reason I am extending the class is to add functionality and features I require that are not included by default. For the functions I create I would like to handle errors in exactly the same way as i would any of the native functions..

I would like the person using the class not have to determine if the functions are native or bespoke.

Normal Function

if ( $_mysqli->query($sql) === false ) die($_mysqli->error);

Required Bespoke Function Example

if ( $_mysqli->run_config() === false ) die($_mysqli->error);

Class

Within my class mysqli_ls that is extending mysqli, i have the following functions which i hoped would set the error message within the parent mysqli class..

   public function run_config()
   {
      // Open Debug FH
      if ($this->config['debug_log'] === TRUE)
      {
         if ($this->fh_debug = fopen($this->config['debug_log_path'], 'a') === FALSE) 
         {
            $this->handle_error('Could not open Debug file handle','200',__METHOD__);
            return $this->E_ERROR;
         }
      }
  }



private function handle_error($errormsg,$errorno,$method)
   {
      $this->errorno = $errorno;
      $this->error   = $errormsg;

      if ($this->config['debug_log'] === TRUE)
      {
         $msg = "($errorno) $errormsg";                              
         $this->write_log('error', $method, $msg);
      }

      return $this->E_OK;
   }

I have tried the above method which errors out with:
PHP Fatal error: mysqli_ls::handle_error(): Cannot write property

and the static call syntax which results in a syntax error as seen below:

parent::error
parent::errorno 

I would like the users of the class to have the default MySQLI functionality and the extended functionality seamlessly without having to use different methods to identify errors.

Thanks in advance for any advise or input given..

EDIT: Adding the full class

class mysqli_ls extends mysqli
{
   private $config = array
   (
      'config_load'     => FALSE,
      'config_path'     => 'mysqli_ls.database.ini',

      'debug_log'       => FALSE,
      'debug_log_path'  => '/tmp/mysqli_ls.debug.log',

      'query_log'       => FALSE,
      'query_log_path'  => '/tmp/mysqli_ls.debug.log',

      'log_date_format' => 'Y-m-d H:i:s'
   );

   // Expected fields for the autoload file
   private $db_fields = array('hostname','username','password','database','port');

   // File Handles
   private $fh_debug  = FALSE;
   private $fh_query  = FALSE;
   private $fh_config = FALSE;

   // Return presets
   private $E_OK    = TRUE;
   private $E_ERROR = FALSE;

   // Existing database connections
   private $db_existing = array();

   public $error;
   public $errorno;

   ## ----------
   public function __construct()
   {
      if (!function_exists('mysqli_connect')) $this->handle_error('MySQLI is not installed','100');
   }

   ## ----------
   public function run_config()
   {
      // Testing error handling
      $this->handle_error('Could not open Debug file handle','200',__METHOD__);

      // Open Debug FH
      if ($this->config['debug_log'] === TRUE)
      {
         if ($this->fh_debug = fopen($this->config['debug_log_path'], 'a') === FALSE) 
         {
            $this->handle_error('Could not open Debug file handle','200',__METHOD__);
            return $this->E_ERROR;
         }
      }

      // Open Query FH
      if ($this->config['query_log'] === TRUE) 
      {
         if ($this->fh_query = fopen($this->config['query_log_path'], 'a') === FALSE)
         {
            $this->handle_error('Could not open Debug file handle','210');            
            return $this->E_ERROR;
         }
      }

      // Load database INI file
      if ($this->config['config_load'] === TRUE)
      {
         if ( ($this->db_existing = parse_ini_file($this->config['config_path'], TRUE)) === FALSE )
         {
            $this->handle_error('Could not parse the database ini file','220'); 
            return $this->E_ERROR;
         }
      }

      // Check for format of the loaded ini file
      foreach($this->db_existing as $name => $row)
      {
         foreach($this->db_fields as $field)
         {
            if ( !isset($row[$field]) ) 
            {
               $this->handle_error("Missing field ($field) in the config array for element ($name)",'230'); 
               return $this->E_ERROR;
            }
         } // END foreach
      } // END foreach

      return $this->E_OK;
   }

   ## ----------
   public function set_config($key, $value)
   {
      if ( !isset($this->config[$key]) ) 
      {
         $this->handle_error('Configuration variable ($key) does not exist','300');
         return $this->E_ERROR;
      }

      $this->config[$key] = $value;
      return $this->E_OK;
   }

   ## ----------
   public function get_config()
   {
      return array_merge($this->config, $this->db_existing);
   }

   ## ----------
   private function write_log($type,$method,$msg)
   {
      $msg = date($this->config['log_date_format']) ."\t". $type ."\t". $method ."\t". $msg ."\n";

      switch($type)
      {
         case 'error':
            fwrite($this->fh_debug, $msg);
         break;

         case 'debug':
            fwrite($this->fh_debug, $msg);
         break;

         case 'query':
            fwrite($this->fh_query, $msg);
         break;

         default:
            return $this->E_ERROR;
         break;      
      }
   }

   ## ----------
   private function handle_error($errormsg,$errorno,$method)
   {
      $this->errorno = $errorno;
      $this->error   = $errormsg;

      if ($this->config['debug_log'] === TRUE)
      {
         $msg = "($errorno) $errormsg";                              
         $this->write_log('error', $method, $msg);
      }

      return $this->E_OK;
   }

   ## ----------
   ## ----------
   ## ----------
   ## ----------      

} // END Class

TEST Script Calling the Class

#!/usr/bin/php
<?php

require('mysqli_ls.class.php');

try
{
   $_mysqli = new mysqli_ls;
}
catch (Exception $e)
{
  print "\n\n". $e->getMessage() ."\n\n";
   print_r($e);
}

$_mysqli->set_config('debug_log',TRUE);
$_mysqli->set_config('query_log',TRUE);
$_mysqli->set_config('config_load',TRUE);

$_mysqli->run_config();
print_r($_mysqli->get_config());

?>

EDIT: So it looks like the error variables in the MySQLi class might be readonly..Is there another way around this issue? Im thinking maybe have my own error function that checks the parent class AND my own variables.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

荒路情人 2024-12-01 16:21:49

使用 __get 魔术函数,您可以执行一些操作来检查父级的错误变量是否已设置。像这样的东西:

class mysqli_ls extends mysqli{
  public $error;
  public $errorno;
  public $connect_error;
  public $connect_errorno;

  function __get($key){
    if(in_array($key, array('error', 'errorno', 'connect_error', 'connect_errorno'))){
      $parentValue = parent::$key;
      if(empty($parentValue)){
        return $this->$key;
      }
      return $parentValue;
    }
  }
}

小测试脚本:

<?php
$test = @new mysqli2('localhost', 'badlogin', 'whatisapassword', 'anonexistentdb');
echo 'test : '.$test->connect_error;

Using the __get magic function, you can do something which will check if the parent's error variable is set. Something like that:

class mysqli_ls extends mysqli{
  public $error;
  public $errorno;
  public $connect_error;
  public $connect_errorno;

  function __get($key){
    if(in_array($key, array('error', 'errorno', 'connect_error', 'connect_errorno'))){
      $parentValue = parent::$key;
      if(empty($parentValue)){
        return $this->$key;
      }
      return $parentValue;
    }
  }
}

Little test script:

<?php
$test = @new mysqli2('localhost', 'badlogin', 'whatisapassword', 'anonexistentdb');
echo 'test : '.$test->connect_error;
少女净妖师 2024-12-01 16:21:49

调用父类方法使用两个冒号:

parent::error();
parent::errorno();

Calls to parent class methods use two colons:

parent::error();
parent::errorno();
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文