Model类支持关联查询预载入功能 增加__toString方法 Db类增加relation和with方法 修正Model一处BUG

This commit is contained in:
thinkphp
2016-04-10 10:00:46 +08:00
parent 112a59723e
commit 6c406e208e
2 changed files with 175 additions and 36 deletions

View File

@@ -259,33 +259,40 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
break;
}
} elseif (is_null($value) && method_exists($this, $name)) {
// 执行关联定义方法
$db = $this->$name();
// 判断关联类型执行查询
switch ($this->relation) {
case self::HAS_ONE:
$result = $db->find();
break;
case self::HAS_MANY:
$result = $db->select();
break;
case self::BELONGS_TO:
$result = $db->find();
break;
case self::BELONGS_TO_MANY:
$result = $db->select();
break;
default:
// 直接返回
$result = $db;
}
// 避免影响其它操作方法
$this->relation = null;
return $result;
// 获取关联数据
return $this->getRelation($name);
}
return $value;
}
// 获取关联数据
protected function getRelation($relation)
{
// 执行关联定义方法
$db = $this->$relation();
// 判断关联类型执行查询
switch ($this->relation[0]) {
case self::HAS_ONE:
$result = $db->find();
break;
case self::HAS_MANY:
$result = $db->select();
break;
case self::BELONGS_TO:
$result = $db->find();
break;
case self::BELONGS_TO_MANY:
$result = $db->select();
break;
default:
// 直接返回
$result = $db;
}
// 避免影响其它操作方法
$this->relation = [];
return $result;
}
/**
* 检测数据对象的值
* @access public
@@ -688,14 +695,71 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
return $model;
}
// 查询当前模型的关联数据
public function relation($relations)
{
if (is_string($relations)) {
$relations = explode(',', $relations);
}
foreach ($relations as $relation) {
$this->data[$relation] = $this->getRelation($relation);
}
return $this;
}
// 预载入关联查询
public function eagerly($resultSet, $relation)
{
$relations = explode(',', $relation);
// 获取关联外键列表
foreach ($relations as $relation) {
$range = [];
$data = [];
$model = $this->$relation();
list($type, $foreignKey, $localKey) = $this->relation;
foreach ($resultSet as $result) {
$range[] = $result->$localKey;
}
$list = $model::where($foreignKey, 'in', $range)->select();
switch ($type) {
case self::HAS_ONE:
case self::BELONGS_TO:
foreach ($list as $set) {
$data[$set->$foreignKey] = $set;
}
break;
case self::HAS_MANY:
case self::BELONGS_TO_MANY:
foreach ($list as $set) {
$data[$set->$foreignKey][] = $set;
}
break;
}
foreach ($resultSet as &$result) {
if (isset($data[$result->$localKey])) {
$result->$relation = $data[$result->$localKey];
}
}
$this->relation = [];
}
return $resultSet;
}
// HAS ONE
public function hasOne($model, $foreignKey = '', $localKey = '')
{
$model = $this->parseModel($model);
$localKey = $localKey ?: $this->pk;
$foreignKey = $foreignKey ?: Loader::parseName($this->name) . '_id';
$this->relation = self::HAS_ONE;
return $model::where($foreignKey, $this->data[$localKey]);
$this->relation = [self::HAS_ONE, $foreignKey, $localKey];
if (isset($this->data[$localKey])) {
// 关联查询封装
return $model::where($foreignKey, $this->data[$localKey]);
} else {
// 预载入封装
return $model;
}
}
// BELONGS TO
@@ -704,8 +768,13 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
$model = $this->parseModel($model);
$foreignKey = $foreignKey ?: $this->pk;
$localKey = $localKey ?: Loader::parseName(basename(str_replace('\\', '/', $model))) . '_id';
$this->relation = self::BELONGS_TO;
return $model::where($foreignKey, $this->data[$localKey]);
$this->relation = [self::BELONGS_TO, $foreignKey, $localKey];
if (isset($this->data[$localKey])) {
// 关联查询封装
return $model::where($foreignKey, $this->data[$localKey]);
} else {
return $model;
}
}
// HAS MANY
@@ -714,8 +783,14 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
$model = $this->parseModel($model);
$localKey = $localKey ?: $this->pk;
$foreignKey = $foreignKey ?: Loader::parseName($this->name) . '_id';
$this->relation = self::HAS_MANY;
return $model::where($foreignKey, $this->data[$localKey]);
$this->relation = [self::HAS_MANY, $foreignKey, $localKey];
if (isset($this->data[$localKey])) {
// 关联查询封装
return $model::where($foreignKey, $this->data[$localKey]);
} else {
return $model;
}
}
// BELONGS TO MANY
@@ -724,8 +799,14 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
$model = $this->parseModel($model);
$foreignKey = $foreignKey ?: $this->pk;
$localKey = $localKey ?: Loader::parseName(basename(str_replace('\\', '/', $model))) . '_id';
$this->relation = self::BELONGS_TO_MANY;
return $model::where($foreignKey, $this->data[$localKey]);
$this->relation = [self::BELONGS_TO_MANY, $foreignKey, $localKey];
if (isset($this->data[$localKey])) {
// 关联查询封装
return $model::where($foreignKey, $this->data[$localKey]);
} else {
return $model;
}
}
/**
@@ -736,14 +817,17 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
protected static function db()
{
$model = get_called_class();
$name = basename(str_replace('\\', '/', $model));
if (!isset(self::$links[$model])) {
self::$links[$model] = Db::connect(static::$connection);
self::$links[$model]->name($name);
if (isset(static::$tableName)) {
self::$links[$model]->setTable(static::$tableName);
}
}
if (isset(static::$tableName)) {
self::$links[$model]->setTable(static::$tableName);
} else {
$name = basename(str_replace('\\', '/', $model));
self::$links[$model]->name($name);
}
// 设置当前模型 确保查询返回模型对象
self::$links[$model]->model($model);
// 返回当前数据库对象
@@ -769,6 +853,11 @@ abstract class Model implements \JsonSerializable, \ArrayAccess
return call_user_func_array([self::db(), $method], $params);
}
public function __toString()
{
return json_encode($this->data);
}
/**
* 解序列化后处理
*/

View File

@@ -1190,12 +1190,24 @@ abstract class Driver
return $tableName;
}
/**
* 设置当前name
* @access public
* @param string $name
* @return Db
*/
public function name($name)
{
$this->name = $name;
return $this;
}
/**
* 得到完整的数据表名
* @access public
* @param array $options 表达式参数
* @return string
*/
public function options(array $options)
{
$this->options = $options;
@@ -1927,19 +1939,53 @@ abstract class Driver
// 数据列表读取后的处理
if (!empty($options['model'])) {
foreach ($resultSet as $key => $result) {
if (!empty($options['model'])) {
// 返回模型对象
$result = new $options['model']($result);
$result->isUpdate(true);
// 关联查询
if (!empty($options['relation'])) {
$result->relation($options['relation']);
}
}
$resultSet[$key] = $result;
}
if (!empty($options['with'])) {
// 预载入
$result = new $options['model']();
return $result->eagerly($resultSet, $options['with']);
}
}
}
return $resultSet;
}
/**
* 设置关联查询预载入
* @access public
* @param string $with 关联名称
* @return Db
*/
public function with($with)
{
$this->options['with'] = $with;
return $this;
}
/**
* 设置关联查询
* @access public
* @param string $relation 关联名称
* @return Db
*/
public function relation($relation)
{
$this->options['relation'] = $relation;
return $this;
}
/**
* 把主键值转换为查询条件 支持复合主键
* @access public
@@ -2006,6 +2052,10 @@ abstract class Driver
// 返回模型对象
$data = new $options['model']($data);
$data->isUpdate(true);
// 关联查询
if (!empty($options['relation'])) {
$data->relation($options['relation']);
}
}
} else {
$data = false;