在cakephp中的多个模型上的单个分页

时间:2016-05-30 10:27:05

标签: php mysql cakephp view pagination

我的要求是搜索2个不同的表:兽医和诊所。它们之间可能存在关联,即结果应该取得诊所名称中的“a”和兽医中的“a”。兽医可能与诊所有关,也可能不与诊所有关。目前我正在做以下事情。有没有办法避免运行2个查询,这也可以帮助我使用cakephp分页助手?

$this->paginate = array(
        'Vet' => array(
            'conditions' => $conditions,
            'fields' => array('Vet.id', 'Vet.name', 'Vet.professionnal_address', 'phone_number', 'Vet.email', 'Vet.type', 'Vet.latitude', 'Vet.longitude','Vet.city','Vet.clinic_id','Vet.zipcode'),
            'joins' => array( 
                array(
                    'table' => 'vet_appointment_types',
                    'alias' => 'VetAppointmentType',
                    'type' => 'LEFT',
                    'conditions' => array(
                        'Vet.id = VetAppointmentType.vet_id',
                    )
                )
            ),
            'limit' => $limit, 
            'group' => array(
                'Vet.id'
            ), 
            'order' => array(
                'Vet.name' => 'ASC'
            )
        ),
        'Clinic' => array(
            'conditions' => $conditions1,
            'fields' => array('Clinic.*'),
            'limit' => $limit, 
            'order' => array(
                'Clinic.name' => 'ASC'
            )
        )
    );

$results = $this->paginate('Vet');
$results2 = $this->paginate('Clinic');

我尝试通过扩展非数据库模型中的默认分页组件并使用union但数据库结构有点复杂,因此无法使用union。此外,我认为实现一个基于临时表的模型是一个选项,但由于它将用于搜索,所以如何实现它,我无法思考。任何帮助都会+ 1'd;)

1 个答案:

答案 0 :(得分:0)

根据建议,我使用数据库视图完成了它。下面是代码。任何改进建议都受到高度赞赏。

// \app\Model\Search.php
App::uses('ConnectionManager', 'Model');
App::uses('AppModel', 'Model');
App::uses('CakeLog', 'Log');

class Search extends AppModel {

    public function __construct() {
        parent::__construct();

        $connection = ConnectionManager::getDataSource('default');
        $return = $connection->execute("CREATE OR REPLACE VIEW `search` AS
            SELECT Vet.id, (CONCAT( `Vet`.`fname` , ' ', `Vet`.`lname` )) AS name, 'vet' AS TYPE , Vet.latitude, Vet.longitude, Vet.zipcode, Vet.speciality FROM `db`.`vets` AS `Vet` 
            UNION 
            SELECT Clinic.id, Clinic.name, 'clinic' AS TYPE , Clinic.lat, Clinic.long, Clinic.zipcode, Clinic.address FROM `db`.`clinics` AS `Clinic`");
        $return = ($return == true)?'Search View created successfully on '.date('Y-m-d H:i:s'):$return;
        CakeLog::write('search', $return);
    }

    public $useTable = 'search';// This model does not use a database table
    public $primaryKey = 'id'; // Define primary key
    public $useDbConfig = 'default'; // Define db

}

这是model.Everytime用户搜索任何东西,当在控制器中加载模型时,创建/替换视图以便可以获取更新的值。它可以用作cakephp中的任何其他模型。它也支持virtualFields。要构建数据库视图的查询,您还可以使用我使用的查询构建器,如下所示。

$joins = array(
        array(
            'table' => 'vet_appointment_types',
            'alias' => 'VetAppointmentType',
            'type' => 'LEFT',
            'conditions' => array(
                'Vet.id = VetAppointmentType.vet_id',
            )
        )
    );

    $dbo = $this->Vet->getDataSource();
    $subQuery = $dbo->buildStatement(
        array(
            'fields' => array('Vet.id', 'Vet.name'),
            'table' => $dbo->fullTableName($this->Vet),
            'alias' => 'Vet',
            'limit' => $limit, 
            'group' => array(
                'Vet.id'
            ), 
            'order' => array(
                'Vet.name' => 'ASC'
            ),
            'offset' => null,
            'joins' => $joins,
            'conditions' => $conditions
        ),
        $this->Vet
    );
    $query = $subQuery;

    $query .= ' UNION ';

    $dbo = $this->Clinic->getDataSource();
    $subQuery = $dbo->buildStatement(
        array(
            'fields' => array('Clinic.id', 'Clinic.name'),
            'table' => $dbo->fullTableName($this->Clinic),
            'alias' => 'Clinic',
            'limit' => $limit,
            'conditions' => $conditions1,
            'limit' => $limit, 
            'order' => array(
                'Clinic.name' => 'ASC'
            ),
            'offset' => null
        ),
        $this->Clinic
    );

    $query .= $subQuery;

    print_r($query);