之前博文

前述博文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

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

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

(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两个字段均关联另外一个表.

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

school表

xueqi表

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

之前博文中:

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

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

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

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

那么问题来了,

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

解决方案:(再次修改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;
}

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

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

THINKPHP_(8)_修改TP源码,支持基于多层关联的任一字段进行排序的更多相关文章

  1. THINKPHP_(1)_修改TP源码,支持对中文字符串按拼音进行排序。

    问题:TP从服务器数据中取出的collection数据,当进一步在网页中进行分页显示时,需要调用order函数,实现类似如下图的排序. 当点击页面中的相关内容时,实现对服务器数据进行重排,就要调用TP ...

  2. 修改json源码支持datetime序列化

    修改json源码支持datetime序列化 import json import datetime now = datetime.datetime.today() json.dumps(now) 抛出 ...

  3. 修改Cosbench源码 支持s3的 http range request 测试场景

    在视频点播的业务应用场景中,用户使用了ffmpeg工具做视频实时转码用. 而ffmpeg使用range 请求.而Cosbench不支持这种测试场景,所以需要修改源码支持这种测试场景. HTTP 协议介 ...

  4. 修改CAS源码是的基于DB的认证方式配置更灵活

    最近在做CAS配置的时候,遇到了数据源不提供密码等数据的情况下,怎样实现密码输入认证呢? 第一步:新建Java项目,根据假面算法生成CAS加密工具 出于保密需要不提供自定义的加密工具,在您的实际项目中 ...

  5. 32.修改IK分词器源码来基于mysql热更新词库

    主要知识点, 修改IK分词器源码来基于mysql热更新词库     一.IK增加新词的原因 在第32小节中学习到了直接在es的词库中增加词语,来扩充自已的词库,但是这样做有以下缺点: (1)每次添加完 ...

  6. 《Flink SQL任务自动生成与提交》后续:修改flink源码实现kafka connector BatchMode

    目录 问题 思路 kafka参数问题 支持batchmode的问题 参数提交至kafkasource的问题 group by支持问题 实现 编译 测试 因为在一篇博文上看到介绍"汽车之家介绍 ...

  7. 修改FFMpeg源码—捕获丢包

    概述 最近我们项目有一个需求就是解决客户端播放RTSP视频流花屏的问题,一般来说丢包就会引起花屏,导致客户端花屏的因素又有很多,比如说: 相机到服务器丢包 服务器到客户端丢包 等等... 其中服务器到 ...

  8. 修改spring源码重写classloader实现项目加密

      (一)操作方法和spring源码添加修改部分 事先说明:spring源码要下载好,会有修改spring的源码操作,本文和本作者所依赖的spring项目的版本是3.1.1,spring4及以上源码对 ...

  9. 修改VCL源码实现自定义输入对话框

    来自:https://yq.aliyun.com/wenji/88428 通过修改VCL源码实现自定义输入对话框 在BCB中有两个函数可以实现输入对话框:InputBox和InputQuery,其实I ...

随机推荐

  1. 【软件工程】《构建之法》 & Git+ & CI/CD

    <构建之法> & Git+ & CI/CD 个人阅读作业#2 项目 内容 本作业所属课程 2020春季软件工程(罗杰 任健) 本作业要求 个人阅读作业#2 我的课程目标 具 ...

  2. Android系统加载Apk文件的时机和流程分析(1)--Android 4.4.4 r1的源码

    本文博客地址:https://blog.csdn.net/QQ1084283172/article/details/80982869 Android系统在启动时安装应用程序的过程,这些应用程序安装好之 ...

  3. 联想R720Y空间问题

    由于之前Y空间在启动项中,所以将他关闭,这次想找到他却找不到 备注:因为在解决问题前,没有把图片保存下来,所以下面用一个颜色框挡住,表示之前的效果 第一个问题 在电脑上找到Y空间 百度上很多说在开始中 ...

  4. PHP逐行解析文件,并写入数据库

    $filePath为文件路径,上传文件则返回文件路径调用下面函数即可public function readText($filePath,&$errorCode,&$errorMess ...

  5. CAS指令

    原文链接:https://www.jianshu.com/p/00edb3d74a33   CAS是CPU的一条指令,其具有原子性,原子性是由CPU硬件层面保证的.   CAS原语有三个操作数--内存 ...

  6. 一文弄懂pytorch搭建网络流程+多分类评价指标

    讲在前面,本来想通过一个简单的多层感知机实验一下不同的优化方法的,结果写着写着就先研究起评价指标来了,之前也写过一篇:https://www.cnblogs.com/xiximayou/p/13700 ...

  7. C++入门教程之一:Hello world

    C++入门教程之一:Hello world C++是各位程序员跳不过的一个坑,也是各位想学编程的人必备的知识,更是各大比赛(如NOI)的官方指定语言. 在TIOBE(一个编程语言社区排行榜)中,截止2 ...

  8. apache common pool2原理与实战

    完整源码,请帮我点个star哦! 原文地址为https://www.cnblogs.com/haixiang/p/14783955.html,转载请注明出处! 简介 对象池顾名思义就是存放对象的池,与 ...

  9. Spring Cloud Alibaba(11)---Sentinel+Nacos持久化

    Sentinel+Nacos持久化 有关Sentinel之前有写过两篇 Spring Cloud Alibaba(9)---Sentinel概述 Spring Cloud Alibaba(10)--- ...

  10. Nacos C++客户端开发文章

    前段时间关注了下阿里巴巴发起的开源项目Nacos,这是一个注册.配置中心(Naming And Config),支持各种语言的客户端,但是唯独没有C++的,考虑到以前做过一段时间的C++程序员,不禁一 ...