数据查询助手

快捷查询助手是一项旨在简化数据查询及分页操作的功能。通过该功能,用户只需输入相关变量,系统便能自动组装数据搜索条件,从而快速获取所需数据。这一优化不仅提高了查询效率,还降低了查询操作的复杂度,使得用户能够更便捷地进行数据检索。

在实际应用中,快捷查询助手能够根据用户输入的关键词、时间范围、数据范围等变量,自动生成相应的查询语句,并自动应用到数据查询过程中。同时,该功能还支持分页显示,用户可以根据需要设置每页显示的数据量,以便更好地浏览和管理数据。

通过使用快捷查询助手,用户可以更加高效地获取所需数据,减少了手动编写查询语句的繁琐过程,提高了工作效率。同时,由于查询条件的自动生成,也减少了因手动输入错误而导致的查询结果不准确的问题,提高了数据查询的准确性和可靠性。

创建查询实例

创建快捷查询助手目前有两种方式,最为方便快捷的方式为模型创建法。

若在模型中需要使用到 QueryHelper 快捷查询助手,模型需要继承 \think\admin\Model 类;

若在控制器需要使用到 QueryHelper 快捷查询助手,控制器需要继承 \think\admin\Controller 类;

// 1. 控制器中,创建查询器(查询器支持链式操作)
// 控制器需要继承 \think\admin\Controller 类;
// 参数:$dbQuery 数据模型或者数据表名
// 参数:$input 为输入对象,如:post、get、request 或 数组
$query = $this->_query();
$query = $this->_query($dbQuery, $input);

// 2. 新版本中,模型也支持直接创建查询器
// 模型需要继承基础模型 \think\admin\Model
// 参数:$input 为输入对象,如:post、get、request 或 数组
// 参数:$callable 为回调用处理,回调参数1为对象本身`QueryHelper`对象
$query = MyModel::mQuery();
$query = MyModel::mQuery($data, function(QueryHelper $helper){});
$query = MyModel::mQuery('post', function(QueryHelper $helper){});
$query = MyModel::mQuery(null, function(QueryHelper $helper){
    $helper->with("数据模型关联");
    $helper->where("查询条件"); 
    // $helper->like()->equal() ...
});

原生表格输出

调用 page 方法后,启用输出模板时会使用 HttpResponseException 响应直接输出机制,后续代码不会再操作。

对象快捷查询进行分页并输出,前端会有两个变量 $list$pagehtml,其中 $list 为当前数据列表数组,$pagehtml 为当前数据列表分页导航条。 前端 html -> table 可以通过使用 foreach 循环输出 Table 数据,生成原生 table 结构的标签代码。

// 快捷查询助手分页参数:
// 参数:$page bool 是否启用分页,默认分页
// 参数:$display bool 是否直接显示模板,默认显示模板
// 参数:$total int|false 分页模式,如果为数字则为列表统计数,false 自动统计,true 使用简化分页
// 参数:$limit int 每页数量,分页时每页数据记录数量
// 参数:$template string 模板名称,当启用显示模板时,会加载指定模板 
$query->page($page, $display, $total, $limit, $template);

layTable 输出

新版本推荐使用 layui.table 组件显示表格数据,目前有针对 Layui 表格组件进行封装 layTable 插件,可动态填充页面高度,支持搜索表单绑定及混合模式调度。


<div class="think-box-shadow">
    <table id="OplogData" data-url="{:sysuri()}" data-target-search="form.form-search"></table>
</div>
<script>
    $(function () {

        // 动态创建 layui.table 表格
        $('#OplogData').layTable({
            even: true, height: 'full',
            sort: {field: 'id', type: 'desc'},
            cols: [[
                {checkbox: true},
                {field: 'id', title: 'ID', width: 80, sort: true, align: 'center'},
                {field: 'username', title: '操作账号', minWidth: 100, sort: true, align: 'center'},
                {field: 'node', title: '操作节点', minWidth: 120},
                {field: 'action', title: '操作行为', minWidth: 120},
                {field: 'content', title: '操作描述', minWidth: 120},
                {field: 'geoip', title: '访问地址', minWidth: 100},
                {field: 'geoisp', title: '网络服务商', minWidth: 100},
                {field: 'create_at', title: '操作时间', minWidth: 170, align: 'center', sort: true},
                {toolbar: '#toolbar', title: '操作面板', align: 'center', fixed: 'right'}
            ]]
        });

        //  监听原 layui.table 的 tool 事件
        $('#OplogData').trigger('tool', function (item) {
            // 对应 layui.table 写法 layui.table.on('tool(OplogData)',function(){ })
            if (item.event === 'EventName') {
                // 判断 lay-event 名称处理
            }
        })

        //  监听原 layui.table 的 toolbar 事件
        $('#OplogData').trigger('toolbar', function (item) {
            // 对应 layui.table 写法 layui.table.on('toolbar(OplogData)',function(){ })
        })

        // 以上为 layTable 兼容 layui.table 用法
        // 更多事件绑定与 layui.table 的事件对应 https://layui.dev/docs/2.8/table/#table.on

        // 原有 jQuery 写法需要使用 off + on 的方式实现 Element 对象查找
        // 由于后台是单页程序,绑定事件需要先 off 再 on,否则容易重复绑定事件或无法触发事件
        $('body').off('click', 'jQ选择器').of('click', 'jQ选择器', function () {
            // 点击事件,可以用 this 读取 Element 对象属性内容
        })
    });
</script>

<script type="text/html" id="toolbar">
    <!--{if auth('remove')}-->
    <a data-action='{:url("remove")}' data-value="id#{{d.id}}" data-confirm="确认要删除这条记录吗?" class="layui-btn layui-btn-sm layui-btn-danger">删 除</a>
    <!--{/if}-->
</script>

后端QueryHelper实现表格数据自动处理,前置操作可以对html模板进行操作,后置操作可以对数据进行过滤及关联操作。

调用layTable方法后,默认为使用HttpResponseException响应直接输出机制,后面的代码不会再执行。

// 新版本中 QueryHelper 对象支持 Layui 表格数据加载
$query->layTable(function(QueryHelper $query){
    // 前置操作,处理 HTML 模型
},function(QueryHelper $query){
    // 后置操作,处理表格数据
});

查询终态操作

特别注意: 下面操作的查询对象 QueryHelper 对象并不是 ThinkORMQuery 对象,如果获取结果操作,需要从 QueryHelper 中获取 ThinkORMQuery 对象。

// 1. 从 QueryHelper 对象中获取 ThinkORM 的 Query 对象
$db = $query->db(); // 获取 ThinkORM 的查询对象

// 2. 使用 ThinkORM 的 Query 对象进行操作
$count = $db->count(); // 获取记录数量
$total = $db->sum('字段'); // 统计字段的求和
$mysql = $db->buildSql(); // 获取生成的 SQL 查询语句

// 3. 更新多数据聚合及查询请查询 ThinkORM 文档

查询条件处理

另外对于 url 输入的变量也可以快速输入到查询器条件中,如:

以下示例代码中的 SystemUser 为数据模型名称或表名的驼峰名称。

// 1.0. 控制器中使用原生查询器(指定表名)
$query = $this->_query('SystemUser');

// 1.1. 使用模型创建查询器。
// SystemUser 是已继承基础模型 \think\admin\Model 的模型类,mQuery() 方法是自定义的查询构建器。
$query = SystemUser::mQuery();

// 2.0. 使用传入的 URL 参数进行查询条件设置。
// 这里使用了 `like` 和 `equal` 方法分别处理模糊查询和精确查询。
$query->like('username,sex')->equal('status');

// 3.0. 执行分页查询,分页结果会自动返回,便于前端展示。
$query->page();

// 4.0. 使用别名来处理 URL 参数与数据库字段不匹配的情况。
// 例如,URL 参数 'username_alias' 对应数据库中的 'username' 字段,
// 'user_status' 对应数据库中的 'status' 字段。
$get = ['username_alias' => 'admin', 'user_status' => 1];
$query = $this->_query('SystemUser', $get)->like('username#username_alias,sex')->equal('status#user_status');

// 5.0. 执行分页查询。
$query->page();

// 6.0. 额外的列表数据赋值到模板:
// 这里查询所有状态为 1 的用户列表,并将结果传递给模板变量 $userList。
$this->userList = $this->_query('SystemUser')->where(['status' => 1])->select();

// 7.0. 接收 URL 参数,进行查询条件设置。
// 假设 URL 参数 'username' 和 'sex' 使用 `like` 模糊查询,'status' 使用 `equal` 精确查询。
$get = ['status' => 1, 'username' => 'admin'];
$query = $this->_query('SystemUser', $get)->like('username,sex')->equal('status');

// 8.0. 执行分页查询并返回结果。
$query->page();

原生 SQL 子查询

使用原生SQL子查询做为了数据表,Sql 语句需要带别名,自 2025/02/16 之后才开始支持。

// 1. 模拟通过 GET 请求传递的查询条件,这里可以根据实际情况动态生成或获取条件
$get = ['username' => 'admin']; 

// 2. 构建查询规则,创建 Query 查询对象。请注意:SQL 语句中必须包含别名以避免歧义。
$query = $this->_query('(SELECT * FROM system_user) AS a', $get);

// 3. 设置查询字段及条件。在使用别名时,条件也必须使用相应的别名。使用 like() 等 QueryHelper 方法进行模糊查询,确保条件正确应用。
$query->like('a.username#username'); // 假设查询用户名为 'admin'

// 4. 额外的过滤条件(可选)。可以根据需求继续链式添加查询条件。
$query->where('a.status', 1); // 假设 1 表示启用状态的用户

// 5. 使用 page() 分页输出结果,也可以直接调用 layTable() 或 select() 方法。
$result = $query->page(true, false);

使用 JOIN 查询

使用 ThinkORM 的 join 操作,一定要设置好 field 以及查询条件必需指定别名。

// 1. 模拟通过 GET 请求传递的查询条件,这里可以根据实际情况动态生成或获取条件
$get = ['username' => 'admin']; 

// 2. 构建查询规则,创建 Query 查询对象。设置 join 相关的参数与 ThinkORM 操作一样。
$query = $this->_query('system_user', $get)->alias('a')->join('system_oplog b', 'a.username=b.username');

// 3. 设置查询字段及条件。在使用别名时,条件也必须使用相应的别名。
//    使用 join 时,输出的结果字段可能会出现重复,这里需要指定 field 输出的字段
$query->like('a.username#username')->field('a.username,b.action');

// 4. 使用 page() 分页输出结果,也可以直接调用 layTable() 或 select() 方法。
$result = $query->page(true, false);
dump($result);

查询回调处理

对于通过 page 方法实现的即将显示到模板的列表,还可以进行引用二次处理,如:

protected function _page_filter(&$data){
  // 这里可以对 $data 进行二次处理,注意是引用
}

### 当一个控制器存在多个page操作时,可以指定回调前缀
protected function _index_page_filter(&$data){
  // 精准回调对 $data 进行二次处理,注意是引用
}

查询使用案例

/**
 * 系统操作日志
 * @auth true
 * @menu true
 * @throws \think\Exception
 * @throws \think\db\exception\DataNotFoundException
 * @throws \think\db\exception\ModelNotFoundException
 * @throws \think\exception\DbException
 */
public function index()
{
    $this->title = '系统操作日志';
    $query = $this->_query($this->table)->like('action,node,content,username,geoip');
    $query->dateBetween('create_at')->order('id desc')->page();
}

/**
 * 列表数据处理
 * @param array $data
 * @throws \Exception
 */
protected function _index_page_filter(&$data)
{
    $ip = new \Ip2Region();
    foreach ($data as &$vo) {
        $result = $ip->btreeSearch($vo['geoip']);
        $vo['isp'] = isset($result['region']) ? $result['region'] : '';
        $vo['isp'] = str_replace(['内网IP', '0', '|'], '', $vo['isp']);
    }
}

快捷查询助手同样保留原 ThinkPHP Query 的功能,也可以使用 JOIN 进行关联查询。

// 在这里对 Db 进行操作,如果 $db->where()->buildSql() 等等
$db = $this->_query('表名')->db();
// or 模型调用法
$db = MyModel::mQuery()->db();

// 输出 Sql 语句
var_dump($db->buildSql());

// 对于联表操作,需要指定 field,join 如果用到了条件查询,一定要用上别名查询
$query = $this->_query('table_a')->alias('a')->field('a.id,a.name,b.pid,b.nickname');
$query->join('table_b b','a.id=b.pid')->like('a.name#a_name')->page();

// 另外,对于 TP 原有的用法也支持如 | 和 & 用法
$map = ['a&b' => 1]; // 生成的条件是: where a = 1 and b = 1
$map = ['a|b' => 1]; // 生成的条件是: where b = 1 or b = 1

// 以上两种情况配合别名使用
$this->_query('table_a')->alias('a')->equal('a|b#name')...
// or 模型调用法
MyModel::mQuery()->alias('a')->equal('a|b#name')...
// 当前请求带上 name 值时,会生成 where a=值 or b=值

$this->_query('table_a')->alias('a')->equal('a&b#name')...
// or 模型调用法
MyModel::mQuery()->alias('a')->equal('a&b#name')...
// 当前请求带上 name 值时,会生成 where a=值 and b=值
Last Updated:
Contributors: 邹景立