标签:esc 提高 了解 code dex offset with 函数 操作
继上次写的一篇文章《CodeIgniter Doctrine2基本使用(一)》写到操作实体的之通过Channel这个实体向数据库表插入一条数据,那么今天要写的就是通过实体获取数据,当然查询这一块比较多,可能也会分好几篇讲。
上一篇文章讲到插入一条数据,插入很简单只要设置实体的成员属性的值就能完成,当然如果有多个实体的话可以多次使用 $this->em->persist();
这个方法。然后只要再执行一次 $this->em->flush();
这个方法就行了,它会进行一次数据库连接,然后执行上面实体生成的SQL语句。
通过Entity查询的方式 有很多,我们先从最简单的开始,还是以Entity\Channel
这个实体为例,以下面一段代码为例:
public function index()
{
$id = 1;
$limit = 20;
$offset = 1;
/**
* @var $channelInfo Entity\Channel
* @var $channelOneBy Entity\Channel
*/
$channelRepository = $this->em->getRepository(‘Entity\Channel‘);
$channelAll = $channelRepository->findAll();
$channelInfo = $channelRepository->find( $id );
$channelOneBy = $channelRepository->findOneBy([
‘channelName‘ => ‘百度‘
], [
‘channelId‘ => ‘DESC‘
]);
$channelBy = $channelRepository->findBy([
‘channelName‘ => ‘阿里‘
], [
‘channelId‘ => ‘DESC‘
], $limit, $offset);
/** @var $value Entity\Channel */
foreach( $channelBy as $value )
{
echo $value->getChannelName();
}
echo $channelInfo->getChannelName();
var_dumt( $channelInfo );
}
看上面一段代码,分别写了4个咱们比较常用的并且简单的查询方式,下面对这些方法进行一个简单的解释:
findAll
这个方法表示查询这个实体的所有数据,也就是查询相应表的所有数据,它返回的是一个多维数组 对象,可以通过 foreach
进行遍历。find
这个是按照ID查询数据,返回一条结果,也就是一个 Channel
实体对象。findOneBy
这个是根据条件查询一条数据,可传入两个数组参数,第一个参数是简单的查询数组,可以写多个它们是 AND
的关系。第二个参数是排序,上面我是channelId
这个字段排序。findBy
这个与上面那个findOneBy
方法类似,前两个参数都一样,第三个参数表示查询的数量,第四表参数表示从第几行开始,返回的是多维数组对象,可以用foreach
遍历。为什么我要写 /** @var $value Entity\Channel */
这样的一些注释? 关于这点,我上篇文章已经讲过了一些,可远远不只是那样。这样写表示 $value
这个变量它指向了实体对象 Entity\Channel
,因为查询到结果后把结果设置到Entity\Channel
这个实体上并且返回这个实体。然后咱们就可以使用这个实体里的一些方法了,比如当你输入 $value->
的时候IDE应该提示这个实体上有哪些方法,然后你找到自己需要的照着敲就是了,这种写法对编辑器是比较友好的。如果对返回的东西不熟悉的话使用var_dump()
这个函数把它打印出来看一下就知道了。
当然这里只是最简单的几种用法如果想了解更多的用法可以去看 Doctrine\ORM\EntityRepository
这个类或继续看我下面的讲解。
往往有时间就上面这些查询是无法满我们的需求的,我们还需要更多的更加复杂的查询功能,当然 Doctrine\ORM\EntityRepository
也提供了给开发人员自己定更复杂的语句的方法,比如:
createQueryBuilder
createNamedQuery
createResultSetMappingBuilder
createNativeNamedQuery
想知道更多可以去doctrine的官网查询更多的相关资料(前提是你英文足够好),当然它提供的这些方法我不建议在控制器上使用,这个时候我们最好把它单独出来,减少代码冗余提高复用性。这个时候我们就要使用到 Repository
了,还记得上篇文章我对Repository的一些简单的介绍吗?什么?不知道?回去把我写的文章抄写三遍...
咱们上一篇文章提到过Repository
这个东西,如果有不清楚的可以点这里《CodeIgniter Doctrine2基本使用(一)》 其实写在Repository里就是为了同样的东西可以重复调用,提高代码的复用性。
那天下面我就举几个例子吧,说这么还不如直接看代码。
/**
* 根据渠道tag查询渠道信息
* @param $tag
* @return array
*/
public function findChannelByTag( $tag )
{
return $this->_em->createQueryBuilder()
->select("c")
->from(‘Entity\Channel‘, ‘c‘)
->where(‘c.channelTag = :tag‘)
->setParameter( ‘tag‘, $tag )
->getQuery()->getSingleResult();
}
/**
* 分页查找所有渠道
* @param $page
* @return mixed
*/
public function findChannelPage( $page )
{
return $this->_em->createQueryBuilder()
->select("c")
->from(‘Entity\Channel‘, ‘c‘)
->orderBy(‘c.channelId‘, ‘desc‘)
->setMaxResults( 10 )
->setFirstResult( (($page -1 ) * 10 ) )
->getQuery()->getResult();
}
/**
* 统计渠道数量
* @return mixed
*/
public function countChannelAll()
{
return $this->_em->createQueryBuilder()
->select("COUNT(c.channelId)")
->from(‘Entity\Channel‘, ‘c‘)
->getQuery()->getSingleScalarResult();
}
如以上写的这些方法,相信大多数人都能看得懂吧,基本每个方法都是这样,更多的方法请查看Doctrine\ORM\QueryBuilder这个类。
Entity\Channel
里面成员属性所对应的字段,可以自定义查询的字段。比如select(‘c.channelId,c.channelTag‘)
它只会返回两个字段数据(理论上是这样)。如果有关联查询如innerJoin
,leftJoin
等关联查询那么可以这样写select(‘c‘, ‘u‘)
当然我这样写的比较少,基本都是使用 ManyToOne 、 OneToOne等形式来进行关联查询,这个我后面再讲,当然也可以通过addSelect()这个方法来设置多个查询实体 $this->_em->createQueryBuilder()
->select(‘u‘)
->addSelect(‘c‘)
->from(‘Entity\User‘, ‘u‘)
->leftJoin(‘Entity\Channel‘, ‘c‘);
where(‘c.channelTag = :tag AND c.channelId = :id‘)
当然,如果不想这样写的话还可以通过andWhere()
方法来设置条件。$this_em->createQueryBuilder()
->select(‘c‘)
->from(‘Entity\Channel‘, ‘c‘)
->where(‘c.channelId = :id‘)
->orWhere(‘c.channelTag = :tag‘)
$this->_em->createQueryBuilder()
->delete(‘Entity\Channel‘, ‘c‘)
->where(‘c.channelId = :id‘)
->setParameter(‘id‘, 1);
$this->_em->createQueryBuilder()
->update(‘Entity\Channel‘, ‘c‘)
->set(‘c.channelName‘, "{$channelName}")
->where(‘c.channelId = 1‘);
use Doctrine\ORM\Query\Expr\Join
因为到现在为止我项目用得比较少,非常少基本都是用ManyToOne()做的联查关于这个联查我将在下一篇文章进行讲解$this_em->createQueryBuilder()
->select(‘u‘)
->from(‘Entity\User‘, ‘u‘)
->innerJoin(‘Entity\Channel‘, ‘c‘, Join::WITH, ‘c.channelId = u.channelId‘);
:id
与:tag
这两个flag,这个方法就是对它们进行传参的,这个上面也有例子,我就不再这里凑字数了 setParameters(new ArrayCollection(array(
new Parameter(‘id‘, 1),
new Parameter(‘tag‘, 2)
)));
$this->_em->createQueryBuilder()
->select(‘u‘)
->from(‘Entity\User‘, ‘u‘)
->groupBy(‘u.channelId‘);
foreach
进行遍历,遍历出来的value就是它的实体上面写了辣么多,我怎么感觉自己好像是在讲怎么操作数据库呢?总结就是需要复用的稍微复杂一点的查询就放在Repository里吧,方便你我他。
好了Repository就暂时先讲到这里吧,下一节我们讲 ManyToOne、OneToMany、OneToOne、ManyToMany的查询。
原文目录:https://lattecake.com/post/20045
CodeIgniter Doctrine2基本使用(二)(转)
标签:esc 提高 了解 code dex offset with 函数 操作
原文地址:https://www.cnblogs.com/codeisfun/p/9001346.html