2012-06-25 27 views
4

是否可以在Codeigniter中实现抽象基础模型?我创建了一个抽象类,它具有基本的CRUD功能,但我收到“无法实例化抽象类”错误。Codeigniter抽象基础模型

抽象模型位于/application/core/my_model.php和根本

abstract class MY_Model extends CI_Model 
{ 

将无法​​实例抽象类”是/system/core/Common.php

174线

它看起来像Codeigniter试图实例化my_model.php时,它被加载,我猜是由于/ core文件夹中的文件被用作扩展核心系统文件,如控制器和模型的地方。停止这个?我试图自动加载模型使用Phil Sturgeon's native autoload但它没有帮助。

/* 
| ------------------------------------------------------------------- 
| Native Auto-load 
| ------------------------------------------------------------------- 
| 
| Nothing to do with cnfig/autoload.php, this allows PHP autoload to work 
| for base controllers and some third-party libraries. 
| 
*/ 

function __autoload($class) 
{ 
    if(strpos($class, 'CI_') !== 0) 
    { 
     @include_once(APPPATH . 'core/'. $class . EXT); 
    } 
} 

我意识到一个简单的方法来做到这一点只是将文件包括在我想使用它的每个模型的顶部,但显然这不会是最佳的。

回答

3

为什么让MY_Model抽象?您只需在MY_Model中放入所有的CRUD函数,并从MY_Model而不是CI_Model扩展模型。无需使用自动加载,因为CodeIgniter允许您只要放入核心文件夹就可以使用MY_Model扩展CI_Model。

一个很好的MY_Model例子是来自Jamie Rumbelow的例子。你可以在这里找到它:https://github.com/jamierumbelow/codeigniter-base-model/

祝你好运!

干杯

巴特

2

我试图调整Loader类为接受摘要和接口。

首先,我把我所有的编辑到我/application/core/MY_Loader.php:

<?php if (! defined('BASEPATH')) exit('No direct script access allowed'); 
/** 
* CodeIgniter 
* 
* An open source application development framework for PHP 5.1.6 or newer 
* 
* @package  CodeIgniter 
* @author  ExpressionEngine Dev Team 
* @copyright Copyright (c) 2008 - 2011, EllisLab, Inc. 
* @license  http://codeigniter.com/user_guide/license.html 
* @link  http://codeigniter.com 
* @since  Version 1.0 
* @filesource 
*/ 

// ------------------------------------------------------------------------ 
// Joseff Betancourt - 11/14/2012 modified code by adding a autoloader and interface loader copied from model loader. 

/** 
* Loader Class 
* 
* Loads views and files 
* 
* @package  CodeIgniter 
* @subpackage Libraries 
* @author  ExpressionEngine Dev Team 
* @category Loader 
* @link  http://codeigniter.com/user_guide/libraries/loader.html 
*/ 
class MY_Loader extends CI_Loader { 

    // All these are set automatically. Don't mess with them. 
    /** 
    * Nesting level of the output buffering mechanism 
    * 
    * @var int 
    * @access protected 
    */ 

    protected $_ci_abstracts_paths  = array(); 
    /** 
    * List of paths to load models from 
    * 
    * @var array 
    * @access protected 
    */ 
    protected $_ci_interfaces_paths  = array(); 
    /** 
    * List of paths to load helpers from 
    * 
    * @var array 
    * @access protected 
    */ 

    protected $_ci_abstracts   = array(); 
    /** 
    * List of loaded interfaces 
    * 
    * @var array 
    * @access protected 
    */ 
    protected $_ci_interfaces   = array(); 
    /** 
    * List of loaded helpers 
    * 
    * @var array 
    * @access protected 
    */ 

    /** 
    * Constructor 
    * 
    * Sets the path to the view files and gets the initial output buffering level 
    */ 

    function __construct() 
    { 
     parent::__construct(); 
     $this->_ci_abstracts_paths = array(APPPATH); 
     $this->_ci_interfaces_paths = array(APPPATH); 
     log_message('debug', "Loader Class Initialized"); 
    } 

    // -------------------------------------------------------------------- 

    /** 
    * Initialize the Loader 
    * 
    * This method is called once in CI_Controller. 
    * 
    * @param array 
    * @return object 
    */ 
    public function initialize() 
    { 

     $this->_ci_abstracts = array(); 
     $this->_ci_interfaces = array(); 
     $this->_ci_autoloader(); 

     return $this; 
    } 

    // -------------------------------------------------------------------- 

    /** 
    * Abstracts Loader 
    * 
    * This function lets users load and instantiate models. 
    * 
    * 11/14/2012 - Joseff Betancourt - Cloned from Models 
    * 
    * @param string the name of the class 
    * @param string name for the abstract 
    * @param bool database connection 
    * @return void 
    */ 
    public function abstracts($abstracts, $name = '', $db_conn = FALSE) 
    { 
     if (is_array($abstracts)) 
     { 
      foreach ($abstracts as $babe) 
      { 
       $this->abstracts($babe); 
      } 
      return; 
     } 

     if ($abstracts == '') 
     { 
      return; 
     } 

     $path = ''; 

     // Is the abstracts in a sub-folder? If so, parse out the filename and path. 
     if (($last_slash = strrpos($abstracts, '/')) !== FALSE) 
     { 
      // The path is in front of the last slash 
      $path = substr($abstracts, 0, $last_slash + 1); 

      // And the model name behind it 
      $abstracts = substr($abstracts, $last_slash + 1); 
     } 

     if ($name == '') 
     { 
      $name = $abstracts; 
     } 

     if (in_array($name, $this->_ci_abstracts, TRUE)) 
     { 
      return; 
     } 

     $CI =& get_instance(); 
     if (isset($CI->$name)) 
     { 
      show_error('The model name you are loading is the name of a resource that is already being used: '.$name); 
     } 

     $abstracts = strtolower($abstracts); 

     foreach ($this->_ci_abstracts_paths as $mod_path) 
     { 
      if (! file_exists($mod_path.'abstracts/'.$path.$abstracts.'.php')) 
      { 
       continue; 
      } 

      if ($db_conn !== FALSE AND ! class_exists('CI_DB')) 
      { 
       if ($db_conn === TRUE) 
       { 
        $db_conn = ''; 
       } 

       $CI->load->database($db_conn, FALSE, TRUE); 
      } 

      if (! class_exists('CI_Abstracts')) 
      { 
       load_class('Abstracts', 'core'); 
      } 

      require_once($mod_path.'abstracts/'.$path.$abstracts.'.php'); 

      $abstracts = ucfirst($abstracts); 

      $CI->$name = new $abstracts(); 

      $this->_ci_abstracts[] = $name; 
      return; 
     } 

     // couldn't find the abstracts 
     show_error('Unable to locate the abstracts you have specified: '.$abstracts); 
    } 

    // -------------------------------------------------------------------- 

    /** 
    * Interface Loader 
    * 
    * This function lets users load and instantiate interfaces. 
    * 
    * 11/14/2012 - Joseff Betancourt - Cloned from Models 
    * 
    * @param string the name of the class 
    * @param string name for the interface 
    * @param bool database connection 
    * @return void 
    */ 
    public function interfaces($interfaces, $name = '', $db_conn = FALSE) 
    { 
     if (is_array($interfaces)) 
     { 
      foreach ($interfaces as $babe) 
      { 
       $this->interfaces($babe); 
      } 
      return; 
     } 

     if ($interfaces == '') 
     { 
      return; 
     } 

     $path = ''; 

     // Is the abstracts in a sub-folder? If so, parse out the filename and path. 
     if (($last_slash = strrpos($interfaces, '/')) !== FALSE) 
     { 
      // The path is in front of the last slash 
      $path = substr($interfaces, 0, $last_slash + 1); 

      // And the model name behind it 
      $interfaces = substr($interfaces, $last_slash + 1); 
     } 

     if ($name == '') 
     { 
      $name = $interfaces; 
     } 

     if (in_array($name, $this->_ci_interfaces, TRUE)) 
     { 
      return; 
     } 

     $CI =& get_instance(); 
     if (isset($CI->$name)) 
     { 
      show_error('The interface name you are loading is the name of a resource that is already being used: '.$name); 
     } 

     $interfaces = strtolower($interfaces); 

     foreach ($this->_ci_interfaces_paths as $mod_path) 
     { 
      if (! file_exists($mod_path.'interfaces/'.$path.$interfaces.'.php')) 
      { 
       continue; 
      } 

      if ($db_conn !== FALSE AND ! class_exists('CI_DB')) 
      { 
       if ($db_conn === TRUE) 
       { 
        $db_conn = ''; 
       } 

       $CI->load->database($db_conn, FALSE, TRUE); 
      } 

      if (! class_exists('CI_Interfaces')) 
      { 
       load_class('Interfaces', 'core'); 
      } 

      require_once($mod_path.'interfaces/'.$path.$interfaces.'.php'); 

      $interfaces = ucfirst($interfaces); 

      $CI->$name = new $interfaces(); 

      $this->_ci_interfaces[] = $name; 
      return; 
     } 

     // couldn't find the interfaces 
     show_error('Unable to locate the interfaces you have specified: '.$interfaces); 
    } 

    // -------------------------------------------------------------------- 


    /** 
    * Autoloader 
    * 
    * The config/autoload.php file contains an array that permits sub-systems, 
    * libraries, and helpers to be loaded automatically. 
    * 
    * @param array 
    * @return void 
    */ 
    private function _ci_autoloader() 
    { 
     // Abstracts models 
     if (isset($autoload['abstracts'])) 
     { 
      $this->model($autoload['abstracts']); 
     } 

     // Interfaces models 
     if (isset($autoload['interfaces'])) 
     { 
      $this->model($autoload['interfaces']); 
     } 

    } 

    // -------------------------------------------------------------------- 

} 

/* End of file Loader.php */ 
/* Location: ./system/core/Loader.php */ 

然后我复制的核心Model.php到Abstracts.php和Interfaces.php(更换用or或者word模型)并将它们放在application/core /文件夹中。

在自动加载我加

/* 
| ------------------------------------------------------------------- 
| Auto-load Interfaces 
| ------------------------------------------------------------------- 
| Prototype: 
| 
| $autoload['interfaces'] = array('interface1', 'interface2'); 
| 
*/ 

$autoload['interfaces'] = array(); 


/* 
| ------------------------------------------------------------------- 
| Auto-load Abstracts 
| ------------------------------------------------------------------- 
| Prototype: 
| 
| $autoload['abstracts'] = array('abstract1', 'abstract2'); 
| 
*/ 

$autoload['abstracts'] = array(); 

,最后我在论文摘要和界面的应用程序文件夹中添加了目录。目前尚未充分证明,但我认为这是一个更全面的方法,允许一个人在需要引用时加载摘要。

而且你创建抽象文件夹中的抽象是这样的:

abstract class MY_Model extends CI_Model 
{ 

blah 

} 
+0

https://github.com/Joseffb/CIAI.git是我放在一起的代码。希望它可以帮助和人们可以为我搞砸了任何东西做出贡献。 – Joseff

+0

只是为了澄清...你真的必须破解(或扩展)CORE才能使其在CI上工作? FML,为什么不使用FuelPHP ......?或者实际上支持常见面向对象操作的东西?为什么要经过这么多的努力? – rckehoe

0
在/application/core/MY_model.php

保存文件:

<?php if (! defined('BASEPATH')) exit('No direct script access allowed'); 

class MY_model extends CI_Model 
{ 
    public function __construct() 
    { 
     parent::__construct(); 
    } 

    // Return all records in the table 
    public function get_all($table) 
    { 
     $q = $this->db->get($table); 
     if($q->num_rows() > 0) 
     { 
      return $q->result(); 
     } 
     return array(); 
    } 

    // Return only one row 
    public function get_row($table,$primaryfield,$id) 
    { 
     $this->db->where($primaryfield,$id); 
     $q = $this->db->get($table); 
     if($q->num_rows() > 0) 
     { 
      return $q->row(); 
     } 
     return false; 
    } 

    // Return one only field value 
    public function get_data($table,$primaryfield,$fieldname,$id) 
    { 
     $this->db->select($fieldname); 
     $this->db->where($primaryfield,$id); 
     $q = $this->db->get($table); 
     if($q->num_rows() > 0) 
     { 
      return $q->result(); 
     } 
     return array(); 
    } 

    // Insert into table 
    public function add($table,$data) 
    { 
     return $this->db->insert($table, $data); 
    } 

    // Update data to table 
    public function update($table,$data,$primaryfield,$id) 
    { 
     $this->db->where($primaryfield, $id); 
     $q = $this->db->update($table, $data); 
     return $q; 
    } 

    // Delete record from table 
    public function delete($table,$primaryfield,$id) 
    { 
     $this->db->where($primaryfield,$id); 
     $this->db->delete($table); 
    } 

    // Check whether a value has duplicates in the database 
    public function has_duplicate($value, $tabletocheck, $fieldtocheck) 
    { 
     $this->db->select($fieldtocheck); 
     $this->db->where($fieldtocheck,$value); 
     $result = $this->db->get($tabletocheck); 

     if($result->num_rows() > 0) { 
      return true; 
     } 
     else { 
      return false; 
     } 
    } 

    // Check whether the field has any reference from other table 
    // Normally to check before delete a value that is a foreign key in another table 
    public function has_child($value, $tabletocheck, $fieldtocheck) 
    { 
     $this->db->select($fieldtocheck); 
     $this->db->where($fieldtocheck,$value); 
     $result = $this->db->get($tabletocheck); 

     if($result->num_rows() > 0) { 
      return true; 
     } 
     else { 
      return false; 
     } 
    } 

    // Return an array to use as reference or dropdown selection 
    public function get_ref($table,$key,$value,$dropdown=false) 
    { 
     $this->db->from($table); 
     $this->db->order_by($value); 
     $result = $this->db->get(); 

     $array = array(); 
     if ($dropdown) 
      $array = array("" => "Please Select"); 

     if($result->num_rows() > 0) { 
      foreach($result->result_array() as $row) { 
      $array[$row[$key]] = $row[$value]; 
      } 
     } 
     return $array; 
    } 
} 

然后扩展,像这样

class any_model extends MY_Model 
{ 
    public function __construct() 
    { 
     parent::__construct(); 
    } 
}