热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析

之前博文前述博文THINKPHP_(1)_修改TP源码,支持对中文字符串按拼音进行排序,其解决的主要问题是,对于查询出的think\collection数据,按指定字段对数据进行排
之前博文

前述博文THINKPHP_(1)_修改TP源码,支持对中文字符串按拼音进行排序,其解决的主要问题是,对于查询出的think\collection数据,按指定字段对数据进行排序,从而在页面上进行重排。

基本原理

(1)前端使用layui框架进行字段显示,比如:

options.cols=[[ //表头

{field: "canxunDanweiSchool", title: '单位名称', sort: true, minWidth:150, templet:function(d){
var str='';
if(d.canxunDanweiSchool)
{
if(d.canxunDanweiSchool.title)
{
str
= d.canxunDanweiSchool.title;
}
}
return str;
}}
,
...

详细内容查阅layui的options设置表格显示,其中sort设置为true

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

(2)后端查询的think\collection数据按指定字段进行重排。当点击对应的上下三角排序角标的时候,会返回到thinkphp的后端controller中。

(3)后端从request中取出对应的field值和order值。

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

 

 (4)对model返回的think\collection数据,按页面点击的排序要求(field和order,即按什么字段,按正序还是倒序进行排序)进行重排。执行的是如下代码:

public function reSetObject($obj, $srcfrom)
{
// 整理变量
$src = [
'field' => ''
,'order' => 'asc'
,'page' => 1
,'limit' => 10
];
$src = array_cover($srcfrom, $src) ;
$data = [
'code' => 0 // ajax请求次数,作为标识符
,'msg' => "" // 获取到的结果数(每页显示数量)
,'count' => 0 // 符合条件的总数据量
,'data' => '' //获取到的数据结果
];
// 整理数据
$cnt = $obj->count();
if($cnt > 0)
{
if($src['field'] != '') # 排序
{
$obj = $obj->order($src['field'], $src['order']);
}
$limit_start = $src['page'] * $src['limit'] - $src['limit'];
$limit_length = $src['limit'] * 1;
$obj = $obj->slice($limit_start, $limit_length);
$data = [
'code' => 0 // ajax请求次数,作为标识符
,'msg' => "" // 获取到的结果数(每页显示数量)
,'count' => $cnt // 符合条件的总数据量
,'data' => $obj //获取到的数据结果
];
}
return $data;
}

上述的核心代码是

$obj->order

即,对think\collection调用order方法。order执行的具体内容位于thinkphp的源代码Collection.php文件中。

具体可以见我之前的博文:THINKPHP_(1)_修改TP源码,支持对中文字符串按拼音进行排序

新的问题

对于多层关联的字段(位于多个不同的表中),如何进一步修正TP源码,以支持排序。

表的关联关系是:

(1)本表,其中school_id和xueqi_id两个字段均关联另外一个表.

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

 

(2)第一层关联的两个表:school表和xueqi表。

school表

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

 

 xueqi表

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

 

 (3)关联第一层表后,再关联第二层的表,即school表的parent_id关联school表自身,表示上级单位。然后xueqi表的peiyang_category_id和xuenian等均关联category表。

之前博文中:

TP模型的多表关联查询和多表字段的关键字搜索

TP6中实现多层关联,第一个表关联第二个表查询出的数据,再关联第三个表

THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析

中能够查询多层关联数据。通过将位于本表,多层关联的表的字段取出来显示在layui的同一个表格中去后。

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

 

 那么问题来了,

如何实现,按位于多个表的字段进行排序呢?

解决方案:(再次修改TP源码)

首先,在layui的options表头配置中,field字段,填入如下内容:

canxunDanweiSchool.xiaojieShangJiDanwei.jiancheng

对应的就是表中的“上级单位”那一列。当点击文字旁边的上下三角角标进行排序,就能将这个field内容传回后台。

然后我们进一步修改order函数,改为:

public function order(string $field, string $order = 'asc')
{
return $this->sort(function ($a, $b) use ($field, $order) {
//添加对中文的支持。
//xiaojie add代码,添加对中文的支持。
//检测field是否为.区分开的多层。
//$pos = strpos($field, '.');
// $field='canxunDanweiSchool.xiaojieShangJiDanwei.jiancheng';

$pos = strpos($field, '.');
if($pos!=false){//存在点号
$k=explode('.',$field);
$fieldA=$a;
$fieldB=$b;
foreach($k as $value){
$fieldA=$fieldA[$value];
$fieldB=$fieldB[$value];
}
$fieldA = $fieldA ?? null;
$fieldB = $fieldB ?? null;
}
else{
$fieldA = $a[$field] ?? null;
$fieldB = $b[$field] ?? null;
}
//注意,如果取出的$fieldA是数字,就不能用preg_match。所以,应该加上一个字符串判断类型。因为有些时候会对数字进行排序。
/*添加对多层关联的支持。*///如果是多层关联数据,比如field传入的数据是canxunDanweiSchool.xiaojieShangJiDanwei.jiancheng
//我们就可以用类似下述的代码进行强制关联。

if (isset($fieldA) && isset($fieldB) && is_string($fieldA) &&is_string($fieldB) && preg_match("/[\x7f-\xff]/", $fieldA)){ //如果字段内容中有中文。
$coll = collator_create( 'zh-CN' );
$res = collator_compare( $coll, $fieldA, $fieldB );
return 'desc' == strtolower($order) ? $res<0 : $res>0;
}
else{
// $fieldA = $a[$field] ?? null;
// $fieldB = $b[$field] ?? null;

return 'desc' == strtolower($order) ? $fieldB > $fieldA : $fieldA > $fieldB;
}
});
}

上述代码,相较于之前博文THINKPHP_(1)_修改TP源码,支持对中文字符串按拼音进行排序中的内容,又主要添加了如下代码:

$pos = strpos($field, '.');
if($pos!=false){//存在点号
$k=explode('.',$field);
$fieldA=$a;
$fieldB=$b;
foreach($k as $value){
$fieldA=$fieldA[$value];
$fieldB=$fieldB[$value];
}
$fieldA = $fieldA ?? null;
$fieldB = $fieldB ?? null;
}

即对多层关联的字段进行识别。如果是多层关联的数据,我们就一直定位到最后的那个表的那个字段,然后按那个字段进行排序。最后的排序结果如下:

 

《THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序,THINKPHP_(4)_TP模型中with、withJoin和多层关联的深入分析》

 

 可以看到,实现了按两层关联的那个表的jiancheng字段进行了中文拼音的排序。


推荐阅读
  • python模块之正则
    re模块可以读懂你写的正则表达式根据你写的表达式去执行任务用re去操作正则正则表达式使用一些规则来检测一些字符串是否符合个人要求,从一段字符串中找到符合要求的内容。在 ... [详细]
  • 本文节选自《NLTK基础教程——用NLTK和Python库构建机器学习应用》一书的第1章第1.2节,作者Nitin Hardeniya。本文将带领读者快速了解Python的基础知识,为后续的机器学习应用打下坚实的基础。 ... [详细]
  • 本文详细介绍了 PHP 中对象的生命周期、内存管理和魔术方法的使用,包括对象的自动销毁、析构函数的作用以及各种魔术方法的具体应用场景。 ... [详细]
  • DVWA学习笔记系列:深入理解CSRF攻击机制
    DVWA学习笔记系列:深入理解CSRF攻击机制 ... [详细]
  • 深入探索HTTP协议的学习与实践
    在初次访问某个网站时,由于本地没有缓存,服务器会返回一个200状态码的响应,并在响应头中设置Etag和Last-Modified等缓存控制字段。这些字段用于后续请求时验证资源是否已更新,从而提高页面加载速度和减少带宽消耗。本文将深入探讨HTTP缓存机制及其在实际应用中的优化策略,帮助读者更好地理解和运用HTTP协议。 ... [详细]
  • Cookie学习小结
    Cookie学习小结 ... [详细]
  • 2020年9月15日,Oracle正式发布了最新的JDK 15版本。本次更新带来了许多新特性,包括隐藏类、EdDSA签名算法、模式匹配、记录类、封闭类和文本块等。 ... [详细]
  • javax.mail.search.BodyTerm.matchPart()方法的使用及代码示例 ... [详细]
  • 本文将带你快速了解 SpringMVC 框架的基本使用方法,通过实现一个简单的 Controller 并在浏览器中访问,展示 SpringMVC 的强大与简便。 ... [详细]
  • Tornado框架中模块与静态文件的应用
    本文详细介绍了Tornado框架中模块和静态文件的使用方法。首先明确模块与模板的区别,然后通过具体的代码示例展示如何在HTML文档中使用模块,并配置模块的路由。最后,提供了模块类中参数获取的示例。 ... [详细]
  • 原文网址:https:www.cnblogs.comysoceanp7476379.html目录1、AOP什么?2、需求3、解决办法1:使用静态代理4 ... [详细]
  • 大类|电阻器_使用Requests、Etree、BeautifulSoup、Pandas和Path库进行数据抓取与处理 | 将指定区域内容保存为HTML和Excel格式
    大类|电阻器_使用Requests、Etree、BeautifulSoup、Pandas和Path库进行数据抓取与处理 | 将指定区域内容保存为HTML和Excel格式 ... [详细]
  • 本文是Java并发编程系列的开篇之作,将详细解析Java 1.5及以上版本中提供的并发工具。文章假设读者已经具备同步和易失性关键字的基本知识,重点介绍信号量机制的内部工作原理及其在实际开发中的应用。 ... [详细]
  • 本文全面解析了 Python 中字符串处理的常用操作与技巧。首先介绍了如何通过 `s.strip()`, `s.lstrip()` 和 `s.rstrip()` 方法去除字符串中的空格和特殊符号。接着,详细讲解了字符串复制的方法,包括使用 `sStr1 = sStr2` 进行简单的赋值复制。此外,还探讨了字符串连接、分割、替换等高级操作,并提供了丰富的示例代码,帮助读者深入理解和掌握这些实用技巧。 ... [详细]
  • Python 程序转换为 EXE 文件:详细解析 .py 脚本打包成独立可执行文件的方法与技巧
    在开发了几个简单的爬虫 Python 程序后,我决定将其封装成独立的可执行文件以便于分发和使用。为了实现这一目标,首先需要解决的是如何将 Python 脚本转换为 EXE 文件。在这个过程中,我选择了 Qt 作为 GUI 框架,因为之前对此并不熟悉,希望通过这个项目进一步学习和掌握 Qt 的基本用法。本文将详细介绍从 .py 脚本到 EXE 文件的整个过程,包括所需工具、具体步骤以及常见问题的解决方案。 ... [详细]
author-avatar
sprout--_557
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有