백엔드 개발 PHP 튜토리얼 关于Thinkphp框架视图模型调用的一些有关问题总结

关于Thinkphp框架视图模型调用的一些有关问题总结

Jun 13, 2016 am 11:01 AM
gt this topic

关于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的确是国内一款少有的成熟的框架~~但有些地方还是要用些另类的技巧来使用,自连接表的视图模型使用总觉得还是比较尴尬的,期待有更好的方式可以达到自连接的目的

?

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

화웨이 GT3 Pro와 GT4의 차이점은 무엇입니까? 화웨이 GT3 Pro와 GT4의 차이점은 무엇입니까? Dec 29, 2023 pm 02:27 PM

많은 사용자들이 스마트 시계를 선택할 때 Huawei 브랜드를 선택하게 됩니다. 그 중 Huawei GT3pro와 GT4가 가장 인기 있는 선택입니다. 두 제품의 차이점을 궁금해하는 사용자가 많습니다. Huawei GT3pro와 GT4의 차이점은 무엇입니까? 1. 외관 GT4: 46mm와 41mm, 재질은 유리 거울 + 스테인레스 스틸 본체 + 고해상도 섬유 후면 쉘입니다. GT3pro: 46.6mm 및 42.9mm, 재질은 사파이어 유리 + 티타늄 본체/세라믹 본체 + 세라믹 백 쉘입니다. 2. 건강한 GT4: 최신 Huawei Truseen5.5+ 알고리즘을 사용하면 결과가 더 정확해집니다. GT3pro: ECG 심전도, 혈관 및 안전성 추가

수정: Windows 11에서 캡처 도구가 작동하지 않음 수정: Windows 11에서 캡처 도구가 작동하지 않음 Aug 24, 2023 am 09:48 AM

Windows 11에서 캡처 도구가 작동하지 않는 이유 문제의 근본 원인을 이해하면 올바른 솔루션을 찾는 데 도움이 될 수 있습니다. 캡처 도구가 제대로 작동하지 않는 주요 이유는 다음과 같습니다. 초점 도우미가 켜져 있습니다. 이렇게 하면 캡처 도구가 열리지 않습니다. 손상된 응용 프로그램: 캡처 도구가 실행 시 충돌하는 경우 응용 프로그램이 손상되었을 수 있습니다. 오래된 그래픽 드라이버: 호환되지 않는 드라이버가 캡처 도구를 방해할 수 있습니다. 다른 응용 프로그램의 간섭: 실행 중인 다른 응용 프로그램이 캡처 도구와 충돌할 수 있습니다. 인증서가 만료되었습니다. 업그레이드 프로세스 중 오류로 인해 이 문제가 발생할 수 있습니다. 이 문제는 대부분의 사용자에게 적합하며 특별한 기술 지식이 필요하지 않습니다. 1. Windows 및 Microsoft Store 앱 업데이트

iPhone에서 App Store 오류에 연결할 수 없는 문제를 해결하는 방법 iPhone에서 App Store 오류에 연결할 수 없는 문제를 해결하는 방법 Jul 29, 2023 am 08:22 AM

1부: 초기 문제 해결 단계 Apple 시스템 상태 확인: 복잡한 솔루션을 살펴보기 전에 기본 사항부터 시작해 보겠습니다. 문제는 귀하의 기기에 있는 것이 아닐 수도 있습니다. Apple 서버가 다운되었을 수도 있습니다. Apple의 시스템 상태 페이지를 방문하여 AppStore가 제대로 작동하는지 확인하세요. 문제가 있는 경우 Apple이 문제를 해결하기를 기다리는 것뿐입니다. 인터넷 연결 확인: "AppStore에 연결할 수 없음" 문제는 때때로 연결 불량으로 인해 발생할 수 있으므로 인터넷 연결이 안정적인지 확인하십시오. Wi-Fi와 모바일 데이터 간을 전환하거나 네트워크 설정을 재설정해 보세요(일반 > 재설정 > 네트워크 설정 재설정 > 설정). iOS 버전을 업데이트하세요.

php提交表单通过后,弹出的对话框怎样在当前页弹出,该如何解决 php提交表单通过后,弹出的对话框怎样在当前页弹出,该如何解决 Jun 13, 2016 am 10:23 AM

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

Kafka에서 주제를 만드는 방법 Kafka에서 주제를 만드는 방법 Jan 17, 2024 pm 04:56 PM

Kafka가 주제를 생성하는 단계: 1. Kafka 설치 및 구성 2. 주제 생성 확인 4. 주제 매개변수 구성 5. Kafka Manager 또는 Confluent Control Center 사용을 고려합니다. 자세한 소개: 1. Kafka 설치 및 구성 먼저 Kafka가 올바르게 설치되었는지, 필요와 환경에 따라 Kafka 매개변수 등을 구성합니다.

이 점을 이해하고 프론트엔드 70%를 따라잡는 글 이 점을 이해하고 프론트엔드 70%를 따라잡는 글 Sep 06, 2022 pm 05:03 PM

Vue2의 이 포인팅 문제로 인해 동료가 버그로 인해 화살표 기능이 사용되어 해당 소품을 얻을 수 없게 되었습니다. 제가 그에게 소개했을 때 그는 그것을 몰랐고, 그래서 저는 일부러 프론트엔드 커뮤니케이션 그룹을 살펴보았습니다. 지금까지 적어도 70%의 프론트엔드 프로그래머들은 오늘 그것을 이해하지 못하고 있습니다. 모든 것이 불분명하다면 이 링크를 아직 배우지 않았다면 큰 소리로 말해주세요.

Vue2가 이를 통해 다양한 옵션의 속성에 접근할 수 있는 이유에 대해 이야기해보겠습니다. Vue2가 이를 통해 다양한 옵션의 속성에 접근할 수 있는 이유에 대해 이야기해보겠습니다. Dec 08, 2022 pm 08:22 PM

이 글은 Vue 소스 코드를 해석하는 데 도움이 될 것이며 이를 사용하여 Vue2의 다양한 옵션에서 속성에 액세스할 수 있는 이유를 소개하는 것이 모든 사람에게 도움이 되기를 바랍니다!

watch4pro가 더 나은가요, 아니면 gt인가요? watch4pro가 더 나은가요, 아니면 gt인가요? Sep 26, 2023 pm 02:45 PM

Watch4pro와 gt는 각각 서로 다른 기능과 적용 가능한 시나리오를 가지고 있습니다. 포괄적인 기능, 고성능, 세련된 외관에 중점을 두고 더 높은 가격을 감수할 의향이 있다면 Watch 4 Pro가 더 적합할 수 있습니다. 높은 기능 요구 사항이 없고 배터리 수명과 합리적인 가격에 더 많은 관심을 기울이는 경우 GT 시리즈가 더 적합할 수 있습니다. 최종 선택은 개인의 필요와 예산, 선호도에 따라 결정되어야 합니다. 자신의 필요를 잘 고려한 후 구매하고, 다양한 제품에 대한 리뷰와 비교를 참고하여 보다 현명한 선택을 하는 것이 좋습니다.

See all articles