【图书介绍】《ThinkPHP 8高效构建Web应用》-CSDN博客
8.1 模型定义
定义模型非常简单,继承think\Model即可。下面是一个用户模型的示例:
<?php
namespace app\model;
use think\Model;
class User extends Model
{
}
默认情况下,模型类名是去除表前缀的数据表名称,采用大驼峰命名法。比如下面的示例:
- User <-> think_user
- UserWallet <-> think_user_wallet
模型的$table属性可以手动指定数据表名称,下面是一个数据表为users的模型示例:
【示例8-1】
<?php
namespace app\model;
use think\Model;
class User extends Model
{
protected $table='users'; // 手动指定数据表名称
}
其他常用的模型属性包括$pk和$connection。$pk用来指定主键字段,默认为id,如果数据表主键不是id,则需要重新定义。$connection用来指定模型使用的数据库连接。
下面是一个指定主键和数据库连接的示例。
【示例8-2】
<?php
namespace app\model;
use think\Model;
class User extends Model
{
protected $pk = 'user_id'; // 指定主键,多对多关联中必须指定主键
protected $connection = 'user'; // 手动指定数据表名称
}
模型字段用来指定模型属性的数据类型,推荐每个模型类都进行定义,ThinkPHP 8默认会自动获取数据表的字段类型(需要查询一次数据库)。在生产实践中一般会开启字段缓存,以避免频繁获取字段类型的开销。
模型的数据字段和对应数据表的字段是对应的,默认会自动获取(包括字段类型),但自动获取会导致增加一次查询(可以开启字段缓存功能),因此需要在模型中明确定义字段信息以避免多一次查询的开销。下面是一个手动定义模型字段的示例。
【示例8-3】
<?php
namespace app\model;
use think\Model;
class User extends Model
{
// 设置字段信息
protected $schema = [
'id' => 'int',
'username' => 'string',
'password' => 'string',
'age' => 'int',
'balance' => 'float',
'created_at' => 'datetime',
'updated_at' => 'datetime',
];
}
上述代码中,模型字段对应数据库中的User表字段,$schema属性需要定义所有字段。如果只需要手动指定某些字段的数据类型,则可以使用$type属性,示例如下:
【示例8-4】
<?php
namespace app\model;
use think\Model;
class User extends Model
{
// 设置字段信息
protected $type = [
'balance' => 'float',
];
}
8.2 插入数据
使用模型插入数据和查询构造器插入数据,最大的不同是模型会执行修改器、自动完成等逻辑,而数据库操作只是单纯的数据插入。
在调用模型实例的save方法插入数据时,如果插入成功,则返回true,否则会抛出异常。
【示例8-5】
本示例演示每个字段单独赋值。首先在mydb库中创建一张表,语句为:
CREATE TABLE `users` (
`id` int NOT NULL AUTO_INCREMENT,
`name` varchar(45) NOT NULL,
`nickname` varchar(45) NOT NULL,
`status` int DEFAULT '0',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb3;
控制器User代码如下:
<?php
namespace app\controller;
use app\model\UserModel;
class User {
public function create() {
$user = new UserModel();
$data = [
'name' => 'John Doe',
'nickname' => 'John Bull',
'status' => 1
];
$result = $user->save($data);
if ($result) {
return '创建成功';
} else {
return '创建失败';
}
}
}
对应的Model为app\model\UserModel.php,代码如下:
<?php
namespace app\model;
use think\Model;
class UserModel extends Model {
// 设置当前模型对应的完整数据表名称
protected $table = 'users';
// 设置模型名称
protected $name = 'user';
// 设置可以批量赋值的字段列表
protected $field = ['id', 'name', 'nickname', 'status'];
}
启动服务器,在浏览器中访问http://localhost:8000/user/create,页面将提示“创建成功”。当然,ThinkPHP 8也提供批量赋值的方法:
$user = new User;
$user->save([
'name' => 'Bruce Lee',
'nickname' => 'Lee three foots',
'status' => 1
]);
1. REPLACE语句插入
REPLACE语句在数据库中通常用于插入新行或更新现有行。其工作原理类似于INSERT语句,但当新行与一个已有的行(根据PRIMARY KEY或UNIQUE索引)产生唯一性约束冲突时,旧行将被删除,然后新行被插入。因此,它可以视为一个删除(如果有需要)和插入的组合操作。
调用模型的replace方法可以进行REPLACE插入:
$user = new User;
$user->replace()->save([
'username' => 'test',
'password' => password_hash('test', PASSWORD_DEFAULT);
]);
2. 批量插入
调用模型实例的saveAll方法可以批量插入数据。当数据行包含主键时,ThinkPHP 8会执行更新操作,否则执行插入操作,代码如下:
$user = new User;
$list = [
['name'=> 'test','age' => 20], // 识别为插入操作
['id' => 2,'name'=> 'test1','age' => 21], // 识别为更新操作
];
$user->saveAll($list);
3. create方法插入
使用模型类的create方法插入数据是一种常用的方法,它可以返回模型实例,代码如下:
$user = User::create([
'name' => 'test',
'age' => 20
]);
8.3 更新数据
save方法会自动根据是否有主键来决定执行插入和更新操作,无须开发者手动调用更新方法。对于更新操作,一般需要先查询出模型实例,给对应的字段赋值,最后调用save方法更新数据,下面是一个示例:
$user = User::where('username','test')->find();
if(!empty($user)) {
$user->age = 20;
$user->save();
}
还可以使用模型类的update方法更新数据,返回对应的模型实例:
User::update(['age' => 20], ['id' => 1]);
8.4 删除数据
使用模型来删除数据,会执行模型的事件处理逻辑;而直接使用查询构造器删除数据,则不会执行模型的事件处理逻辑。下面是几种删除数据的方法:
$user = User::find(1);
$user->delete(); // 调用模型实例的delete方法
User::destroy(1); // 静态方法删除
User::destroy([1,2,3]); // 支持批量删除多个数据
User::where('age', '<', 20)->delete(); // 基于where条件删除