検索
ホームページデータベースmysql チュートリアル独自のデータベース パッケージを作成する方法 (3)

独自のデータベース パッケージを作成する方法 (3)

Apr 04, 2017 pm 02:25 PM
データベースのカプセル化


この問題のキーポイント

php関数の様々な補助関数を深く理解する PHPコア構文: function

変数パラメータfunction、...$var、新機能の紹介とは何かを理解するPHP5.6の

compact関数PHPの使い方:compact - Manual

list関数PHPの使い方: list - Manual

PHPMagicメソッド

始める前に

今回は、PHP5.6のクエリ文についてです。 SQL は複雑さや複数のクラス間の相関関係が Connector の記事ほど単純ではないため、今回は Builder クラス、Grammar クラス、Model クラスを一度に説明する必要があります。クエリ部分


Builder.php

  • リクエストビルダー、すべてのクラス間のブリッジ

<?php /**
* 请求构建器
*/
class Builder {
    // 连接数据库, Connector类
    protected $connector;

    // 生成SQL语法,Grammar类
    protected $grammar;

    // 连接的Model, Model类
    protected $model;

    // SQL查询语句中条件的值
    // 虽然列出了全部, 但本教程只实现了where,其余的因为懒(理直气壮),请自行实现, 逻辑和where函数大致
    protected $bindings = [
        &#39;select&#39; => [],
        'join'   => [],
        'where'  => [],
        'having' => [],
        'order'  => [],
        'union'  => [],
    ];

    // select 语法想要查看的字段
    public $columns;

    // 过滤重复值
    public $distinct = false;

    // 需要查询的表
    public $from;

    // 所有join 语法
    public $joins;

    // 所有where 语法
    public $wheres;

    // group 语法
    public $groups;

    // having 语法
    public $havings;

    // order by 语法
    public $orders;

    // 限制数据库返回的数据量, limit语法
    public $limit;

    // 需要略过的数据量, offset语法
    public $offset;

    // 数据写保护, 开启后该条数据无法删除或改写
    public $writeLock = false;

次は関数

  • _construct - インスタンスを生成した後の最初のステップは何ですか?

    select - 表示したい列を選択します。デフォルトはすべて、つまり '*'です
  function construct() {
      // 新建两个实例
      // 如果已经理解Connector的原理后自然明白这个Connector实例已经联通了数据库
      $this->connector = new Connector;
      $this->grammar = new Grammar;
  }
  • distinct - 重複する値をフィルターします
  • public function select($columns = ['*']) {
          // $columns只能存入数组, 所以需要判定, 如果不是, 将所有参数合成一个数组再存入
          // 这是一个更人性化的设定, 用户可以选择以下两种调用方式
          // select(['first_name', 'last_name']), 以数组的方式
          // select('first_name', 'last_name'), 以参数的方式
          // 最后一点, 你可以发现所有函数最后都会存入对应的Builder属性中
          // 这和你在做饭前先处理材料是同一个道理, 也就是预处理
          $this->columns = is_array($columns) ? $columns : func_get_args();
          return $this;
      }
  • from - テーブル名を設定します
  • public function distinct() {
          // 开启过滤
          $this->distinct = true;
          return $this;
      }
  • join - 2 つのテーブルを接続する結合構文、デフォルトは内部結合です
  •   public function from($table) {
          $this->from = $table;
          return $this;
      }
  • join のさまざまな変形は 3 つの一般的な結合のみを実装します
  • /**
       * @param  string $table   需要连接的副表名
       * 为什么主键和外键可以单个或数组呢
       * 原因是join语法可以on多个键
       * @param  string/array $foregin 外键
       * @param  string/array $primary 主键
       * @param  string $type    连接方式, 默认inner
       * @return Builder          返回Builder实例
       */
      public function join($table, $foregin , $primary, $type = 'inner') {
          // 判定外键变量的数据类型
          if(is_array($foregin)) {
              // 如果是数组, 循环加上副表名在前头
              foreach($foregin as &$f)
                  $f = $table.".".$f;
          }else {
              // 反之, 不需循环直接加
              $foregin = $table.".".$foregin;
          }
    
          // 与$foreign的逻辑同理
          if(is_array($primary)) {
              foreach($primary as &$p)
                  $p = $this->from.".".$p;
          }else {
              $primary = $this->from.".".$primary;
          }
    
          // 将所有经过处理的参数收入$joins待用
          $this->joins[] = (object)[
              'from' => $this->from,
              'table' => $table,
              'foregin' => $foregin,
              'primary' => $primary,
              'type' => $type
          ];
    
          // 返回Builder实例
          return $this;
      }
  • addBinding - に添付された値を保存します。さまざまな SQL 条件
  • // 所有逻辑同join(), 不过这是left join
      public function leftJoin($table, $foregin , $primary) {
          return $this->join($table, $foregin , $primary, 'left');
      }
    
      // 所有逻辑同join(), 不过这是right join
      public function rightJoin($table, $foregin , $primary) {
          return $this->join($table, $foregin , $primary, 'right');
      }
  • where - SQL の条件、where 構文
  • where() には 2 つの異なる呼び出しメソッドがあります
  • 以下の関数には 2 つの呼び出しメソッドがあります。以下のコードロジックから理解してくださいパラメータ、デフォルトは '=' です

    /**
       * @param string/array $value 字段匹配的值
       * @param string $type  条件类型, 默认为where, 具体查看$bindings
       */
      public function addBinding($value, $type = 'where') {
          // 如果$type并不是$bindings的键, 跳出错误
          if (!array_key_exists($type, $this->bindings)) 
              throw new InvalidArgumentException("Invalid binding type: {$type}.");
    
          // 如果$value是数组,将其与之前存入的值整合为一维数组
          if (is_array($value)) 
              $this->bindings[$type] = array_values(array_merge($this->bindings[$type], $value));
          // 反之, 直接存入最后一位即可
          else
              $this->bindings[$type][] = $value;
    
          // 返回Builder实例
          return $this;
      }

    配列の形式では、このメソッドはフィールドが値と等しい状況にのみ一致します
    Actor::select('first_name', 'last_name')
      ->where('first_name', 'NICK')
      ->where('last_name','!=', 'WAHLBERG')
      ->first()
    次に、 のさまざまな変換を見てみましょう。 code

    Actor::select('first_name', 'last_name')
      ->where(['first_name'=> 'NICK', 'last_name'=> 'WAHLBERG'])
      ->first()

    where
      public function where($column, $operator = null, $value = null, $boolean = 'and') {
              // 判定$column是否为数组
              if (is_array($column)) {
                  // 如果是数组, 循环再调用本函数,where()
                  foreach ($column as $key => $value) 
                      $this->where($key, "=", $value, $boolean);
              }else {
                  // 反之, 判定参数数量和$value是否为空, 如果为真,这意味着用户省略了'=',自动添加
                  if(func_num_args() == 2 || is_null($value)) list($operator, $value) = ['=', $operator];
      
                  // 最简单原始的条件查询, 所以$type值为Basic
                  $type = "Basic";
                  // 将处理过的条件存入$wheres
                  $this->wheres[] = compact('type', 'column', 'operator', 'value', 'boolean');
                  // 将字段需要匹配的值存入$bindings中的where
                  $this->addBinding($value, 'where');
              }
      
              // 返回Builder实例
              return $this;
          }
    • orderBy - ステートメントごとに並べ替え、決定 返されるデータの配置
    • // 所有逻辑同where(), 不过这是or where
        public function orWhere($column, $operator = null, $value = null) {
            return $this->where($column, $operator, $value, 'or');
        }
      
        /**
         * where in 语法, 字段匹配多个值
         * @param  string  $column  字段
         * @param  array  $values  一组字段需匹配的值
         * @param  string  $boolean 默认为and
         * @param  boolean $not     默认为假, 真为排除所有$value里的数据
         * @return Builder           返回Builder实例
         */
        public function whereIn($column, $values, $boolean = 'and', $not = false) {
            // 判定条件查询的类型, false = where in ($value),true = where not in ($value)
            $type = $not ? 'NotIn' : 'In';
            // 将条件存入$wheres
            $this->wheres[] = compact('type', 'column', 'values', 'boolean');
            // 循环将字段需要匹配的值存入$bindings中的where
            foreach ($values as $value) 
                $this->addBinding($value, 'where');
      
            // 返回Builder实例
            return $this;
        }
      
        // 所有逻辑同whereIn(), 不过这是or where in
        public function orWhereIn($column, $values) {
            return $this->whereIn($column, $values, 'or');
        }
      
        // 所有逻辑同whereIn(), 不过这是and where not in
        public function whereNotIn($column, $values, $boolean = 'and') {
            return $this->whereIn($column, $values, $boolean, true);
        }
      
        // 所有逻辑同whereNotIn(), 不过这是or where not in
        public function orWhereNotIn($column, $values) {
            return $this->whereNotIn($column, $values, 'or');
        }
      
        /**
         * where $coulmn is null 语法, 搜索字段为空的数据
         * @param  string  $column  字段
         * @param  string  $boolean 默认为and
         * @param  boolean $not     默认为假, 真为排除所有字段为空的数据
         * @return Builder          返回Builder实例
         */
        public function whereNull($column, $boolean = 'and', $not = false) {
            // 判定条件查询的类型, false = where $column is null,true = where $column is not null
            $type = $not ? 'NotNull' : 'Null';
            // 将条件存入$wheres
            $this->wheres[] = compact('type', 'column', 'boolean');
            // 返回Builder实例
            return $this;
        }
      
        // 所有逻辑同whereNull(), 不过这是or where $column is null
        public function orWhereNull($column) {
            return $this->whereNull($column, 'or');
        }
      
        // 所有逻辑同whereNull(), 不过这是and where $column is not null
        public function whereNotNull($column, $boolean = 'and') {
            return $this->whereNull($column, $boolean, true);
        }
      
        // 所有逻辑同whereNotNull(), 不过这是or where $column is not null
        public function orWhereNotNull($column) {
            return $this->whereNotNull($column, 'or');
        }
    • 補助関数 - array_ flatten

      これは私が自分で書いた関数で、

      多次元配列を平坦化するために使用されます

      それは何ですか多次元配列を 1 次元配列にフラット化することです

      /**
         * @param  string/array $column    字段
         * @param  string $direction 排序,默认为asc, 顺序
         * @return Builder            返回Builder实例
         */
        public function orderBy($column, $direction = 'asc') {
            // 局限性在于必须声明顺序或逆序
            if(is_array($column)) {
                foreach ($column as $key => $value) 
                    $this->orderBy($key, $value);
            }else {
                // 简单判定后直接存入$orders, $direction输入错误不跳错误直接选择顺序
                $this->orders[] = [
                    'column' => $column,
                    'direction' => strtolower($direction) == 'desc' ? 'desc' : 'asc',
                ];
            }
      
            // 返回Builder实例
            return $this;
        }

      function array_flatten(array $array) {
        $return = array();
        array_walk_recursive($array, function($a) use (&$return) { 
            $return[] = $a; 
        });
        return $return;
      }
      Return

      $a = array('this', 'is', array('a', 'multidimentional', array('array') ), 'to', 'make', 'the', 'tutotal', array('more', 'easier', 'to', 'understand') );
      dd(array_flatten($a));

      groupBy - ステートメントごとにグループ化し、データを統合します
    • array (size=13)
      0 => string 'this' (length=4)
      1 => string 'is' (length=2)
      2 => string 'a' (length=1)
      3 => string 'multidimentional' (length=16)
      4 => string 'array' (length=5)
      5 => string 'to' (length=2)
      6 => string 'make' (length=4)
      7 => string 'the' (length=3)
      8 => string 'tutotal' (length=7)
      9 => string 'more' (length=4)
      10 => string 'easier' (length=6)
      11 => string 'to' (length=2)
      12 => string 'understand' (length=10)
    • limit - 返されるデータの量を制限しますsqlsrv の記述方法が異なります。興味がある場合は、メッセージを残してください。
    • /**
         * @param  string/array $groups 字段
         * @return Builder        返回Builder实例
         */
        public function groupBy(...$groups) {
            if(empty($this->groups)) $this->groups = [];
            $this->groups = array_merge($this->groups, array_flatten($groups));
            // 返回Builder实例
            return $this;
        }
    • offset - 指定されたデータ量をスキップします。sqlsrv の記述方法が異なります。知りたい場合は、メッセージを残すことができます
    •   public function limit($value) {
            // 如果$value大于零这条函数才生效
            if ($value >= 0) $this->limit = $value;
            return $this;
        }
      
        // limit函数的别名, 增加函数链的可读性
        public function take($value) {
            return $this->limit($value);
        }
    • get - データベース データの読み取り
    • ここでハイライトします。これまでのすべての関数は Builder インスタンスを返します。つまり、編集可能です。これは、要約すると、フロントエンド関数チェーンのリクエストがまとめて処理されるということです
    •   public function offset($value) {
            // 如果$value大于零这条函数才生效
            if ($value >= 0) $this->offset = $value;
            return $this;
        }
      
        // offset函数的别名, 增加函数链的可读性
        public function skip($value) {
            return $this->offset($value);
        }



      getBindings - すべての $bindings の値を返します

      // 返回一组数据库数据, 可以在这里设定想返回的字段, 但是select()的优先度最高
        public function get($columns = ['*']) {
            // 如果Builder的$columns依然为空, 那么就用该函数的$columns, 反之则使用select()所声明的字段
            if (is_null($this->columns)) $this->columns = $columns;
            // 如果Builder的$orders依然为空, 那么就默认第一个字段顺序
            // 发现一个莫名的bug, 可能是我理解错了, 不加 order by 1数据返回竟然不是按照主键(第一个字段)排序
            // 所以以防万一加一个默认
            if (is_null($this->orders)) $this->orderBy(1);
            // 将Grammar类生成的语句,和处理过的字段所对应的值,都交给Connector类, 让它与数据库进行通信,返回数据
            // 注意这里的三个函数
            // read() 不用说Connector篇介绍过了
            // compileSelect()是用来编译生成查询语句
            // getBindings()用来获取收在$bindings中条件的值, 下方会有说明
            $results = $this->connector->read($this->grammar->compileSelect($this), $this->getBindings());
            // 返回一组数据库数据,如果查询为空,返回空数组
            // cast()下方会有说明
            return $this->cast($results);
        }
      
        // get函数的别名, 增加函数链的可读性
        public function all($columns = ['*']) {
            return $this->get($columns);
        }
    • cast - 返されたデータ型を独自の Model サブクラスに変換します

      基本的な考え方の章で、データの操作性については、最後のエフェクトのセクションで言及されています。コードはここにあります。理解できなくても問題ありません。Model.php を読めば理解できます (そうですよね?)
    •   public function getBindings() {
            // 抚平多维数组成一维数组后再返回
            return array_flatten($this->bindings);
        }

    • first - 最初のデータのみです。がフェッチされるのでオブジェクトを返しますが、get は複数のオブジェクトを含む配列を返します

        public function cast($results){
            // 获取Model子类的名称
            $class = get_class($this->model);
            // 新建一个Model子类
            $model = new $class();
            // 如果获得的数据库数据是数组
            if (gettype($results)=="array") {
                $arr = [];
                // 循环数据
                foreach ($results as $result) 
                    // 再调用本函数
                    $arr[] = $this->cast($result);
                // 返回经过转化的数据数组
                return $arr;
            // 如果获得的数据库数据是对象
            }elseif(gettype($results)=="object"){
                // 存入数据对象
                $model->setData($results);
                // 加入主键或unique key以实现数据的可操作性
                // 如果表存在主键和返回的数据中有主键的字段
                if($model->getIdentity() && isset($results->{$model->getIdentity()})) {
                    $model->where($model->getIdentity(), $results->{$model->getIdentity()});
                // 如果表存在unique key和返回的数据中有unique key的字段
                }elseif($model->getUnique() && array_check($model->getUnique(),$results)) {
                    foreach ($model->getUnique() as $key) 
                        $model->where($key, $results->$key);
                // 改写和删除操作仅仅在符合以上两种条件其中之一的时候
                // 反之, 开启写保护不允许改写
                }else {
                    // 其实还可以考虑直接复制query
                    // 但变数太多干脆直接一棍子打死
                    $model->getBuilder()->writeLock = true;
                }
                // 返回该实例
                return $model;
            }
            // 如果转化失败返回false
            return false;
        }
    • setModel - Model インスタンスの設定

      /**
         * @param  array  $columns 如果Builder的$columns依然为空, 那么就用该函数的$columns, 反之则使用select()所声明的字段
         * @return boolean/Model          查询为空返回false, 反之则返回附带数据的表类
         */
        public function first($columns = ['*']) {
            $results = $this->take(1)->get($columns);
            return empty($results) ? false : $results[0];
        }
      が終了したら、次に SQL ステートメント、文法クラスをコンパイルします
    • Grammar.php


    • は、処理後にBuilderインスタンス属性に格納された値に従ってコンパイルされます

      • コンパイルは文法の一部ずつゆっくりとコンパイルされ、最終的に要約されます

        public function setModel(Model $model) {
            $this->model = $model;
            return $this;
        }
    • concatenate - exclude コンパイル後に空の値が存在する可能性があり、SQL ステートメント全体が接続されます
    • <?php /**
      * 数据库语法生成    
      */
      class Grammar {
          // 构建查询语句所可能出现的各种SQL语法
          // 注意, 它们的顺序是对应着各自在SQL语句中合法的位置
          // sqlsrv略微不同
          protected $selectComponents = [
              &#39;distinct&#39;,
              &#39;columns&#39;,
              &#39;from&#39;,
              &#39;joins&#39;,
              &#39;wheres&#39;,
              &#39;groups&#39;,
              &#39;orders&#39;,
              &#39;limit&#39;,
              &#39;offset&#39;,
          ];
      • compileSelect - SQL クエリ ステートメントをコンパイルします

          protected function concatenate($segments) {
              return implode(' ', array_filter($segments, function ($value) {
                  return (string) $value !== '';
              }));
          }
      • compileComponents - $selectComponents をループし、さまざまな条件に従って対応するステートメントをローカルでコンパイルします構文

          // 还记得Builder->get()中的compileSelect()吗?
          public function compileSelect(Builder $query) {
              // concatenate()排除编译后可能存在空的值,然后连接整句SQL语句
              // 去掉可能存在的前后端空格再返回
              return trim($this->concatenate($this->compileComponents($query)));
          }
      • applyDistinct - 個別のステートメントをコンパイルします

          protected function compileComponents(Builder $query) {
              $sql = [];
              // 循环$selectComponents
              foreach ($this->selectComponents as $component) {
                  // 如果Builder实例中对应的函数曾经被调用,那意味着对应的语法非空
                  if (!is_null($query->$component)) {
                      $method = 'compile'.ucfirst($component);
                      // 编译该语法并将之收入$sql
                      $sql[$component] = $this->$method($query, $query->$component);
                  }
              }
              // 返回$sql数组
              return $sql;
          }
      • compileLimit - 制限ステートメントをコンパイルします

          protected function compileDistinct(Builder $query, $distinct) {
              return $distinct ? 'select distinct' : 'select';
          }
      • compileOffset - オフセットステートメントをコンパイルします

          protected function compileLimit(Builder $query, $limit) {
              return "limit $limit";
          }
      • compileColumns - クエリされるフィールドをコンパイルします

        rree

      • コンパイルから -テーブル名をコンパイルして生成

          protected function compileOffset(Builder $query, $offset) {
              return "offset $offset";
          }
      • compileJoins - joinステートメントをコンパイルします

          protected function compileColumns(Builder $query, $columns) {
              return implode(', ', $columns);
          }
      • compileWheres - whereステートメントをコンパイルします

          protected function compileFrom(Builder $query, $table) {
              return 'from '.$table;
          }
      • whereBasic - 最も基本的なwhereの使用法をコンパイルします

          protected function compileJoins(Builder $query, $joins) {
              $sql = [];
              foreach ($joins as $join) {
                  // 如果存在多个副键和主键
                  if(is_array($join->foregin) && is_array($join->primary)) {
                      $on = [];
                      // 循环键的数量, 将之与对应的主键组合
                      for($i=0; $i<sizeof>foregin); $i++)
                          $on[] = $join->foregin[$i]." = ".$join->primary[$i];
                      // 最后再将所有句子用and连接
                      $on = implode(' and ', $on);
                  } else {
                  //反之, 直接连接即可
                      $on = "$join->foregin = $join->primary";
                  }
                  // 附上join的类型和副表
                  $sql[] = trim("{$join->type} join {$join->table} on $on");
              }
        
              // 连接再返回
              return implode(' ', $sql);
          }</sizeof>
      • whereIn - where inの使用法をコンパイルします

          protected function compileWheres(Builder $query) {
              $sql = [];
              // 类似与compileComponents(), 循环Builder实例中的$wheres
              foreach ($query->wheres as $where) {
                  // 根据不同的$type来进行编译
                  $method = "where{$where['type']}";
                  // 返回的部分语句先收入数组
                  $sql[] = $where['boolean'].' '.$this->$method($query, $where);
              }
              // 最后将$sql数组连接起来, 删掉最前面的and或or在返回
              return 'where '.preg_replace('/and |or /i', '', implode(" ", $sql), 1);
          }
      • whereNotIn - 使用法ではない場所でコンパイルします

      • whereNull - 使用法が null の場所でコンパイルします
      •   protected function whereBasic(Builder $query, $where) {
              // 检测$where[column]是否存在小数点
              // 是, 就意味着前缀已经附带了表名
              // 否, 为之后的字段添上表名
              // 因为join存在副表, 所以部分$where可能有附带表名, 这时候就不用添加了
              $table = !preg_match('/\./', $where['column']) ? $query->from."." : '';
              // 返回添上表名的字段,和表达式, 再一个问号
              // 为何使用问号而不是:变量名? 因为:变量名存在太多局限性,不能标点符号,不能数字开头
              return $table.$where['column'].' '.$where['operator'].' ?';
          }

      • whereNotNull - 使用法が null でない場所でコンパイルします
      •   protected function whereIn(Builder $query, $where) {
              // 检测$where[column]是否存在小数点, 同理whereBasic()
              $table = !preg_match('/\./', $where['column']) ? $query->from."." : '';
              // 有多少个匹配值那就连接多少个问号
              $values = implode(', ', array_fill(0, sizeof($where['values']), '?'));
              // 返回where in 语句
              return $table.$where['column'].' in ('.$values.')';
          }

      • compileGroups - ステートメントごとにグループをコンパイルします
      •   protected function whereNotIn(Builder $query, $where) {
              // 检测$where[column]是否存在小数点, 同理whereBasic()
              $table = !preg_match('/\./', $where['column']) ? $query->from."." : '';
              // 有多少个匹配值那就连接多少个问号
              $values = implode(', ', array_fill(0, sizeof($where['values']), '?'));
              // 返回where not in 语句
              return $table.$where['column'].' not in ('.$values.')';
          }

      • 注文をコンパイルする -ステートメントごとに順序をコンパイルします
      •   protected function whereNull(Builder $query, $where) {
              // 检测$where[column]是否存在小数点, 同理whereBasic()
              $table = !preg_match('/\./', $where['column']) ? $query->from."." : '';
              // 返回where is null 语句
              return $table.$where['column'].' is null';
          }

        文法は終わりです、次は

        エントリファイル
      • 、モデルクラス、魔法の世界です

      Model.php

      • 数据库表的依赖对象

      • 作为数据的出口, 数据就在这里进行修饰

      • 各种魔术方法用得飞起, 使用之前请先理解魔术方法是什么

      <?php /**
      * 入口文件, 数据库表的父类
      */
      class Model {
          // SQL命令构建器, Builder类
          protected $builder;
      
          // 数据库返回的数据存在这里
          protected $data;
      
          // 数据库表名, 选填, 默认为类名
          protected $table;
      
          // 主键, 二选一($unique)
          protected $identity;
      
          // unique key, 二选一($identity)
          protected $unique;
      • getTable - 获取数据库表名, 没有设置返回false

          public function getTable() {
              return isset($this->table) ? $this->table : false;
          }
      • getIdentity - 获取主键名, 没有返回假

          public function getIdentity() {
              return isset($this->identity) ? $this->identity : false;
          }
      • getUnique - 获取unique key名, 没有返回假

          public function getUnique() {
              // 检测是否存在unique key, 不存在返回假, 存在就在检查是否数组, 不是就装入数组再返回
              return isset($this->unique) ? is_array($this->unique) ? $this->unique : [$this->unique] : false;
          }
      • check - 检查必须预设的实例属性

          public function check() {
              // 如果数据库表的名称和Model的子类相同,可以选择不填,默认直接取类的名称
              if(!$this->getTable()) 
                  $this->table = get_class($this);
        
              // 跳出提醒必须设置$identity或$unique其中一项
              if(!$this->getIdentity() && !$this->getUnique())
                  throw new Exception('One of $identity or $unique should be assigned in Model "'.get_called_class().'"');
        
          }
      • set/getBuilder - 设置或读取Builder实例

        // 设置Builder实例
          public function setBuilder(Builder $builder) {
              $this->builder = $builder;
              return $this;
          }
        
          // 获取Builder实例
          public function getBuilder() {
              return $this->builder;
          }
      • setData - 设置数据库数据

          public function setData($data) {
              $this->data = $data;
              return $this;
          }
      • construct - 创建实例后的第一步

          function construct() {
              // 检查设定是否正确
              $this->check();
              // 新建一个Builder实例
              $this->setBuilder(new Builder);
              // 设置构建器的主表名称
              $this->getBuilder()->from($this->table);
              // 将Model实例带入Builder
              $this->getBuilder()->setModel($this);
          }

        魔术方法

      • callStatic - 如果找不到静态函数的时候自动运行下面的逻辑

          static public function callStatic($method, $args = null) {
              // 这是一个伪静态, 创建一个实例
              $instance = new static;
              // 在$instance->builder之中, 寻找函数$method, 并附上参数$args
              return call_user_func_array([$instance->builder, $method], $args);
          }
      • call - 如果找不到函数的时候自动运行下面的逻辑

          public function call($method, $args) {
              // 在$this->builder之中, 寻找函数$method, 并附上参数$args
              return call_user_func_array([$this->builder, $method], $args);
          }
      • debugInfo - 在调试的时候隐藏多余的信息, 只留下数据库返回的数据

          public function debugInfo() {
              // 也不懂算不算bug, 该方法强制要求返回的数据类型必须是array数组
              // 但是就算我强行转换(casting)后返回的数据依然是对象(object)
              return (array)$this->data;
          }
      • get - 当调用对象的属性时, 强制调用这个魔术方法

          // 为了避免局外人可以访问Model类的属性
          // 为了避免对象属性和表的字段名字相同
          public function get($field) {
              // 如果调用的属性是Model类内的逻辑
              // 直接返回该属性的值
              if(get_called_class()==="Model") 
                  return $this->$field;
              // 反之, 则检查$data内是否存在该属性, 没有的话跳出错误
              if(!isset($this->data->$field)) 
                  throw new Exception("column '$field' is not exists in table '$this->table'");
              // 如果存在,由于返回的数据都是存在$data里, 所以要这样调用
              return $this->data->$field;
          }
      • set - 当想修改的对象属性时, 强制调用这个魔术方法

          public function set($field, $value) {
              // 如果调用的属性是Model类内的逻辑
              // 直接赋值该属性
              if(get_called_class()==="Model")
                  return $this->$field = $value;
              // 反之, 则检查$data内是否存在该属性, 没有的话跳出错误
              if(!isset($this->data->$field)) 
                  throw new Exception("column '$field' is not exists in table '$this->table'");
        
              // 如果存在,由于返回的数据都是存在$data里, 所以要这样赋值
              return $this->data->$field = $value;
          }

        进阶的Buider类封装没错, 我又把Builder(微)封装了

      • find - 使用主键查寻数据

        /**
           * @param  int $id 主键
           * @return Model subclass     返回一个Model的子类数据
           */
          public static function find($id) {
              // 这是一个伪静态, 创建一个实例
              $self = new static;
        
              // 该函数只适用于设置了主键的表, 如果没有设置, 跳出错误
              if(!$self->getIdentity()) 
                  throw new Exception("Table's identity key should be assgined");
        
              return $self->where($self->identity, $id)->first();
          }

        还有更多, 看心情更...


      本期疑问

      1.) 缺少的Builder函数如果有人愿意提供例子就好了, 进阶复杂的语句那就更好了, 直接写然后再分享给我那就最好了
      2.) 有些函数或结构可能没有效率或者白白添加服务器压力, 但我写的顺了可能没看见, 请指出
      3.) 有人能告诉我laravel是怎么解决pdo 2100个最大参数(bind)的问题吗? 源代码把我看蒙了


      以上が独自のデータベース パッケージを作成する方法 (3)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

      声明
      この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
      複数の単一列インデックスに対して複合インデックスをいつ使用する必要がありますか?複数の単一列インデックスに対して複合インデックスをいつ使用する必要がありますか?Apr 11, 2025 am 12:06 AM

      データベースの最適化では、クエリ要件に従ってインデックス作成戦略を選択する必要があります。1。クエリに複数の列が含まれ、条件の順序が固定されている場合、複合インデックスを使用します。 2。クエリに複数の列が含まれているが、条件の順序が修正されていない場合、複数の単一列インデックスを使用します。複合インデックスは、マルチコラムクエリの最適化に適していますが、単一列インデックスは単一列クエリに適しています。

      MySQLでスロークエリを識別して最適化する方法は? (スロークエリログ、Performance_schema)MySQLでスロークエリを識別して最適化する方法は? (スロークエリログ、Performance_schema)Apr 10, 2025 am 09:36 AM

      MySQLスロークエリを最適化するには、slowquerylogとperformance_schemaを使用する必要があります。1。LowerQueryLogを有効にし、しきい値を設定して、スロークエリを記録します。 2。performance_schemaを使用してクエリの実行の詳細を分析し、パフォーマンスのボトルネックを見つけて最適化します。

      MySQLおよびSQL:開発者にとって不可欠なスキルMySQLおよびSQL:開発者にとって不可欠なスキルApr 10, 2025 am 09:30 AM

      MySQLとSQLは、開発者にとって不可欠なスキルです。 1.MYSQLはオープンソースのリレーショナルデータベース管理システムであり、SQLはデータベースの管理と操作に使用される標準言語です。 2.MYSQLは、効率的なデータストレージと検索機能を介して複数のストレージエンジンをサポートし、SQLは簡単なステートメントを通じて複雑なデータ操作を完了します。 3.使用の例には、条件によるフィルタリングやソートなどの基本的なクエリと高度なクエリが含まれます。 4.一般的なエラーには、SQLステートメントをチェックして説明コマンドを使用することで最適化できる構文エラーとパフォーマンスの問題が含まれます。 5.パフォーマンス最適化手法には、インデックスの使用、フルテーブルスキャンの回避、参加操作の最適化、コードの読み取り可能性の向上が含まれます。

      MySQL非同期マスタースレーブレプリケーションプロセスを説明してください。MySQL非同期マスタースレーブレプリケーションプロセスを説明してください。Apr 10, 2025 am 09:30 AM

      MySQL非同期マスタースレーブレプリケーションにより、BINLOGを介したデータの同期が可能になり、読み取りパフォーマンスと高可用性が向上します。 1)マスターサーバーレコードはBinlogに変更されます。 2)スレーブサーバーは、I/Oスレッドを介してBINLOGを読み取ります。 3)サーバーSQLスレッドは、BINLOGを適用してデータを同期させます。

      MySQL:簡単な学習のためのシンプルな概念MySQL:簡単な学習のためのシンプルな概念Apr 10, 2025 am 09:29 AM

      MySQLは、オープンソースのリレーショナルデータベース管理システムです。 1)データベースとテーブルの作成:createdatabaseおよびcreateTableコマンドを使用します。 2)基本操作:挿入、更新、削除、選択。 3)高度な操作:参加、サブクエリ、トランザクション処理。 4)デバッグスキル:構文、データ型、およびアクセス許可を確認します。 5)最適化の提案:インデックスを使用し、選択*を避け、トランザクションを使用します。

      MySQL:ユーザーフレンドリーなデータベースの紹介MySQL:ユーザーフレンドリーなデータベースの紹介Apr 10, 2025 am 09:27 AM

      MySQLのインストールと基本操作には、次のものが含まれます。1。mysqlをダウンロードしてインストールし、ルートユーザーパスワードを設定します。 2。sqlコマンドを使用して、createdatabaseやcreateTableなどのデータベースとテーブルを作成します。 3. CRUD操作を実行し、挿入、選択、更新、コマンドを削除します。 4.パフォーマンスを最適化し、複雑なロジックを実装するためのインデックスとストアドプロシージャを作成します。これらの手順を使用すると、MySQLデータベースをゼロから構築および管理できます。

      InnoDBバッファープールはどのように機能し、なぜパフォーマンスに不可欠なのですか?InnoDBバッファープールはどのように機能し、なぜパフォーマンスに不可欠なのですか?Apr 09, 2025 am 12:12 AM

      Innodbbufferpoolは、データとインデックスページをメモリにロードすることにより、MySQLデータベースのパフォーマンスを向上させます。 1)データページは、ディスクI/Oを削減するためにBufferPoolにロードされます。 2)汚れたページは、定期的にディスクにマークされ、リフレッシュされます。 3)LRUアルゴリズム管理データページの排除。 4)読み出しメカニズムは、可能なデータページを事前にロードします。

      MySQL:初心者向けのデータ管理の容易さMySQL:初心者向けのデータ管理の容易さApr 09, 2025 am 12:07 AM

      MySQLは、インストールが簡単で、強力で管理しやすいため、初心者に適しています。 1.さまざまなオペレーティングシステムに適した、単純なインストールと構成。 2。データベースとテーブルの作成、挿入、クエリ、更新、削除などの基本操作をサポートします。 3.参加オペレーションやサブクエリなどの高度な機能を提供します。 4.インデックス、クエリの最適化、テーブルパーティション化により、パフォーマンスを改善できます。 5。データのセキュリティと一貫性を確保するために、バックアップ、リカバリ、セキュリティ対策をサポートします。

      See all articles

      ホットAIツール

      Undresser.AI Undress

      Undresser.AI Undress

      リアルなヌード写真を作成する AI 搭載アプリ

      AI Clothes Remover

      AI Clothes Remover

      写真から衣服を削除するオンライン AI ツール。

      Undress AI Tool

      Undress AI Tool

      脱衣画像を無料で

      Clothoff.io

      Clothoff.io

      AI衣類リムーバー

      AI Hentai Generator

      AI Hentai Generator

      AIヘンタイを無料で生成します。

      ホットツール

      DVWA

      DVWA

      Damn Vulnerable Web App (DVWA) は、非常に脆弱な PHP/MySQL Web アプリケーションです。その主な目的は、セキュリティ専門家が法的環境でスキルとツールをテストするのに役立ち、Web 開発者が Web アプリケーションを保護するプロセスをより深く理解できるようにし、教師/生徒が教室環境で Web アプリケーションを教え/学習できるようにすることです。安全。 DVWA の目標は、シンプルでわかりやすいインターフェイスを通じて、さまざまな難易度で最も一般的な Web 脆弱性のいくつかを実践することです。このソフトウェアは、

      Safe Exam Browser

      Safe Exam Browser

      Safe Exam Browser は、オンライン試験を安全に受験するための安全なブラウザ環境です。このソフトウェアは、あらゆるコンピュータを安全なワークステーションに変えます。あらゆるユーティリティへのアクセスを制御し、学生が無許可のリソースを使用するのを防ぎます。

      メモ帳++7.3.1

      メモ帳++7.3.1

      使いやすく無料のコードエディター

      SublimeText3 Mac版

      SublimeText3 Mac版

      神レベルのコード編集ソフト(SublimeText3)

      MantisBT

      MantisBT

      Mantis は、製品の欠陥追跡を支援するために設計された、導入が簡単な Web ベースの欠陥追跡ツールです。 PHP、MySQL、Web サーバーが必要です。デモおよびホスティング サービスをチェックしてください。