connection = $this->db->getConnect($this->sql); } /** * 重新构建 * @throws */ private function initPDOStatement() { if (empty($this->sql)) { return; } if (($connect = $this->getConnection()) === false) { throw new Exception('数据库繁忙, 请稍后再试!'); } $this->prepare = $connect->prepare($this->sql); if (!($this->prepare instanceof PDOStatement)) { throw new Exception($this->sql . ':' . $this->getError()); } } /** * @return bool|PDOStatement * @throws */ public function incrOrDecr() { return $this->execute(static::EXECUTE); } /** * @param bool $isInsert * @param bool $hasAutoIncrement * @return bool|string * @throws */ public function save($isInsert = TRUE, $hasAutoIncrement = true) { return $this->execute(static::EXECUTE, $isInsert, $hasAutoIncrement); } /** * @param $model * @param $attributes * @param $condition * @param $param * @return Command * @throws Exception */ public function update($model, $attributes, $condition, $param) { $change = $this->db->getSchema()->getChange(); $sql = $change->update($model, $attributes, $condition, $param); return $this->setSql($sql)->bindValues($param); } /** * @param $tableName * @param $attributes * @param $condition * @return Command * @throws Exception */ public function batchUpdate($tableName, $attributes, $condition) { $change = $this->db->getSchema()->getChange(); [$sql, $param] = $change->batchUpdate($tableName, $attributes, $condition); return $this->setSql($sql)->bindValues($param); } /** * @param $tableName * @param $attributes * @return Command * @throws Exception */ public function batchInsert($tableName, $attributes) { $change = $this->db->getSchema()->getChange(); $attribute_key = array_keys(current($attributes)); [$sql, $param] = $change->batchInsert($tableName, $attribute_key, $attributes); return $this->setSql($sql)->bindValues($param); } /** * @param $tableName * @param $attributes * @param $param * @return Command * @throws Exception */ public function insert($tableName, $attributes, $param) { $change = $this->db->getSchema()->getChange(); $sql = $change->insert($tableName, $attributes, $param); return $this->setSql($sql)->bindValues($param); } /** * @return bool|int * @throws Exception */ public function all() { return $this->execute(static::FETCH_ALL); } /** * @param $tableName * @param $param * @param $condition * @return Command * @throws Exception */ public function mathematics($tableName, $param, $condition) { $change = $this->db->getSchema()->getChange(); $sql = $change->mathematics($tableName, $param, $condition); return $this->setSql($sql); } /** * @return array|mixed * @throws Exception */ public function one() { return $this->execute(static::FETCH); } /** * @return bool|int * @throws Exception */ public function fetchColumn() { return $this->execute(static::FETCH_COLUMN); } /** * @return bool|int * @throws Exception */ public function rowCount() { return $this->execute(static::ROW_COUNT); } /** * @return bool|int * @throws Exception */ public function flush() { return $this->execute(static::EXECUTE); } /** * @param $type * @param $isInsert * @param $hasAutoIncrement * @return bool|int * @throws Exception */ private function execute($type, $isInsert = null, $hasAutoIncrement = true) { try { $time = microtime(true); if ($type === static::EXECUTE) { $result = $this->insert_or_change($isInsert, $hasAutoIncrement); } else { $result = $this->search($type); } if ($this->prepare instanceof PDOStatement) { $this->prepare->closeCursor(); } $time = round(microtime(true) - $time, 6); $this->debug($this->sql . ' Run-Time:' . $time); } catch (\Throwable | Exception $exception) { $this->error($this->sql . '. error: ' . $exception->getMessage()); $result = null; } finally { return $result; } } /** * @param $type * @return array|int|mixed * @throws Exception */ private function search($type) { $connect = $this->getConnection()->query($this->sql); if ($type === static::ROW_COUNT) { $result = $connect->rowCount(); } else if ($type === static::FETCH_COLUMN) { $result = $connect->fetchColumn(); } else if ($type === static::FETCH_ALL) { $result = $connect->fetchAll(PDO::FETCH_ASSOC); } else { $result = $connect->fetch(PDO::FETCH_ASSOC); } return $result; } /** * @param $isInsert * @param $hasAutoIncrement * @return bool|string * @throws Exception */ private function insert_or_change($isInsert, $hasAutoIncrement) { $this->initPDOStatement(); $this->bind($this->prepare); if (($result = $this->prepare->execute()) === false) { return $this->addErrorLog(); } if (!$isInsert) { return true; } $result = $this->connection->lastInsertId(); if ($result == 0 && $hasAutoIncrement) { return $this->addErrorLog(); } return $result; } /** * @param $modelName * @return $this */ public function setModelName($modelName) { $this->_modelName = $modelName; return $this; } /** * @return string */ public function getModelName() { return $this->_modelName; } /** * @return bool|int * @throws Exception */ public function delete() { return $this->execute(static::EXECUTE); } /** * @return bool|int * @throws Exception */ public function exec() { return $this->execute(static::EXECUTE); } /** * @param PDOStatement $prepare * @param $name * @param $value */ public function bindParam($prepare, $name, $value) { $prepare->bindParam($name, $value); } /** * @param array|null $data * @return $this */ public function bindValues(array $data = NULL) { if (!is_array($this->params)) { $this->params = []; } if (!empty($data)) { $this->params = array_merge($this->params, $data); } return $this; } /** * @param $sql * @return $this * @throws Exception */ public function setSql($sql) { $this->sql = $sql; return $this; } /** * @return string */ public function getError() { return $this->prepare->errorInfo()[2] ?? 'Db 驱动错误.'; } /** * @return bool * @throws Exception */ public function addErrorLog() { if ($this->prepare->errorCode() === '00000') { return true; } return $this->addError($this->getError(), 'mysql'); } /** * @param PDOStatement $prepare * @return PDOStatement * @throws Exception */ private function bind($prepare) { if (empty($this->params)) { return $prepare; } foreach ($this->params as $key => $val) { if (is_array($val)) { throw new Exception("Save data cannot have array"); } $this->bindParam($prepare, ':' . ltrim($key, ':'), $val); } return $prepare; } }