关于Thinkphp框架视图模型调用的一些有关问题总结
关于Thinkphp框架视图模型调用的一些问题总结
对于tp框架中视图模型的调用在项目中比较常用,这次的毕业设计中对于碰到了一些问题写下总结:
?
一、自定义DAO的ORM的Model定义在视图调用时出现异常
?
当把Model定义成形如
class UserModel extends Model { private $ormObj; /** * * 构造函数 */ function __construct(){ $this->ormObj=M('User'); }}
然后是视图模型的定义:
class MentionviewModel extends ViewModel { public $viewFields = array( 'Mention'=>array('id','tid','uid'), 'Topic' => array('create_time','from'=>'topic_from','content','status','_on'=>'Mention.tid=Topic.id'), 'User' => array('nickname','homepage','avatar','_on'=>'Topic.uid=User.id') ); }
而输出的SQL语句确是:
SELECT Mention.id AS id,Mention.uid AS uid,Mention.tid AS tid,Topic.create_time AS create_time,Topic.from AS topic_from,Topic.content AS content,User.avatar AS avatar,User.nickname AS nickname,User.homepage AS homepage FROM fl_mention Mention JOIN Topic ON Mention.tid=Topic.id JOIN User ON Topic.uid=User.id WHERE Topic.status=1 and Mention.uid=1 ORDER BY Mention.id desc LIMIT 0,20
?对于要取别名的表名丢失了,这种DAO式的ORM调用时在视图模型中会出现找不到表名的情况。
?
在父类ViewModel中有这样一个函数,是提取表名的:
public function getTableName() { if(empty($this->trueTableName)) { $tableName = ''; foreach ($this->viewFields as $key=>$view){ // 获取数据表名称 $class = $key.'Model'; $Model = class_exists($class)?new $class():M($key); $tableName .= $Model->getTableName(); // 表别名定义 $tableName .= !empty($view['_as'])?' '.$view['_as']:' '.$key; // 支持ON 条件定义 $tableName .= !empty($view['_on'])?' ON '.$view['_on']:''; // 指定JOIN类型 例如 RIGHT INNER LEFT 下一个表有效 $type = !empty($view['_type'])?$view['_type']:''; $tableName .= ' '.strtoupper($type).' JOIN '; $len = strlen($type.'_JOIN '); } $tableName = substr($tableName,0,-$len); $this->trueTableName = $tableName; } return $this->trueTableName; }
这里函数看到会去调用该类的超类Model的 $Model->getTableName() 这个函数:
?
public function getTableName() { if(empty($this->trueTableName)) { $tableName = !empty($this->tablePrefix) ? $this->tablePrefix : ''; if(!empty($this->tableName)) { $tableName .= $this->tableName; }else{ $tableName .= parse_name($this->name); } $tableName .= !empty($this->tableSuffix) ? $this->tableSuffix : ''; if(!empty($this->dbName)) $tableName = $this->dbName.'.'.$tableName; $this->trueTableName = strtolower($tableName); } return $this->trueTableName; }
这个函数中这个语句empty($this->trueTableName)便是提取表名的
下面给出解决方案一:
class UserModel extends Model { protected $trueTableName = 'fl_user'; private $ormObj; /** * * 构造函数 */ function __construct(){ $this->ormObj=M('User'); }}
?在该类中加入protected $trueTableName = 'fl_user'; 这个属性使触发getTablename函数时可以找到对应的表名;
解决方案二:
function __construct(){ parent::__construct(); $this->ormObj=M('User'); }
?在子类中将覆盖掉的父类构造函数重新引入
?
?
?
?
二、自连接表的视图模型调用:
?
形如:
class TopicviewModel extends ViewModel { public $viewFields = array( 'topic'=>array('id'=>'root_id','content'=>'root_content'), 'Topic' =>array('id'=>'topic_id','create_time','from'=>'topic_from','content','status','_on'=>'topic.id=topic.rootid') ); }
上面的语句执行的sql如下:
SELECT topic.id AS root_id,Topic.id AS topic_id FROM fl_topic topic JOIN fl_topic Topic ON topic.id=Topic.rootid
这句sql语句的错误很明显:因为sql是不区分大小写的,所以会直接报??Not unique table/alias: 'Topic'的错误。
?
继续来看这个函数:
public function getTableName() { if(empty($this->trueTableName)) { $tableName = ''; foreach ($this->viewFields as $key=>$view){ // 获取数据表名称 $class = $key.'Model'; $Model = class_exists($class)?new $class():M($key); $tableName .= $Model->getTableName(); // 表别名定义 $tableName .= !empty($view['_as'])?' '.$view['_as']:' '.$key; // 支持ON 条件定义 $tableName .= !empty($view['_on'])?' ON '.$view['_on']:''; // 指定JOIN类型 例如 RIGHT INNER LEFT 下一个表有效 $type = !empty($view['_type'])?$view['_type']:''; $tableName .= ' '.strtoupper($type).' JOIN '; $len = strlen($type.'_JOIN '); } $tableName = substr($tableName,0,-$len); $this->trueTableName = $tableName; } return $this->trueTableName; }
上面的函数对于前面的‘topic’索引只是做了工厂方式的模型生成。而其中的class_exists是不区分大小写的,而数组的索引是区分大小写的,我们可以利用这个特性,把同一个表的引用用仅大小写不同的索引表示
再在后面跟上_as属性即可:
?
下面代码:
public $viewFields = array( 'topic'=>array('id'=>'root_id','content'=>'root_content','_as'=>'root_topic'), 'Topic' => array('id'=>'topic_id','create_time','from'=>'topic_from','content','status','_on'=>'root_topic.id=topic.rootid') );
?这样便可得到正确的sql语句了
SELECT root_topic.id AS root_id,Topic.id AS topic_id FROM fl_topic root_topic JOIN fl_topic Topic ON root_topic.id=topic.rootid
?
tp的确是国内一款少有的成熟的框架~~但有些地方还是要用些另类的技巧来使用,自连接表的视图模型使用总觉得还是比较尴尬的,期待有更好的方式可以达到自连接的目的
?

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

许多用户在选择智能手表的时候都会选择的华为的品牌,其中华为GT3pro和GT4都是非常热门的选择,不少用户都很好奇华为GT3pro和GT4有什么区别,下面就就给大家介绍一下二者。华为GT3pro和GT4有什么区别一、外观GT4:46mm和41mm,材质是玻璃表镜+不锈钢机身+高分纤维后壳。GT3pro:46.6mm和42.9mm,材质是蓝宝石玻璃表镜+钛金属机身/陶瓷机身+陶瓷后壳二、健康GT4:采用最新的华为Truseen5.5+算法,结果会更加的精准。GT3pro:多了ECG心电图和血管及安

为什么截图工具在Windows11上不起作用了解问题的根本原因有助于找到正确的解决方案。以下是截图工具可能无法正常工作的主要原因:对焦助手已打开:这可以防止截图工具打开。应用程序损坏:如果截图工具在启动时崩溃,则可能已损坏。过时的图形驱动程序:不兼容的驱动程序可能会干扰截图工具。来自其他应用程序的干扰:其他正在运行的应用程序可能与截图工具冲突。证书已过期:升级过程中的错误可能会导致此issu简单的解决方案这些适合大多数用户,不需要任何特殊的技术知识。1.更新窗口和Microsoft应用商店应用程

第1部分:初始故障排除步骤检查苹果的系统状态:在深入研究复杂的解决方案之前,让我们从基础知识开始。问题可能不在于您的设备;苹果的服务器可能会关闭。访问Apple的系统状态页面,查看AppStore是否正常工作。如果有问题,您所能做的就是等待Apple修复它。检查您的互联网连接:确保您拥有稳定的互联网连接,因为“无法连接到AppStore”问题有时可归因于连接不良。尝试在Wi-Fi和移动数据之间切换或重置网络设置(“常规”>“重置”>“重置网络设置”>设置)。更新您的iOS版本:

php提交表单通过后,弹出的对话框怎样在当前页弹出php提交表单通过后,弹出的对话框怎样在当前页弹出而不是在空白页弹出?想实现这样的效果:而不是空白页弹出:------解决方案--------------------如果你的验证用PHP在后端,那么就用Ajax;仅供参考:HTML code

kafka创建topic的步骤:1、安装和配置Kafka;2、创建Topic;3、验证Topic创建;4、配置Topic参数;5、考虑使用Kafka Manager或Confluent Control Center;6、注意事项。详细介绍:1、安装和配置Kafka,首先确保已经正确安装了Kafka,并且它正在运行,根据需求和环境,配置Kafka的参数等等。

同事因为this指向的问题卡住的bug,vue2的this指向问题,使用了箭头函数,导致拿不到对应的props。当我给他介绍的时候他竟然不知道,随后也刻意的看了一下前端交流群,至今最起码还有70%以上的前端程序员搞不明白,今天给大家分享一下this指向,如果啥都没学会,请给我一个大嘴巴子。

本篇文章带大家解读vue源码,来介绍一下Vue2中为什么可以使用 this 访问各种选项中的属性,希望对大家有所帮助!

watch4pro和gt各自具有不用的特点和适用场景,如果注重功能的全面性、高性能和时尚外观,同时愿意承担较高的价格,那么Watch 4 Pro可能更适合。如果对功能要求不高,更注重电池续航和价格的合理性,那么GT系列可能更适合。最终的选择应根据个人需求、预算和喜好来决定,建议在购买前仔细考虑自己的需求,并参考各种产品的评测和比较,以做出更明智的选择。
