php 左右值分类 算法,解析thinkphp的左右值无限分类_PHP教程

解析thinkphp的左右值无限分类_PHP教程

以前一直使用父子无限分类,这种分类结构清晰,使用也简单。但若分类数量很大的话,在查询上性能不佳。比如在做导航菜单中,我要根据某一分类查询出整个分类树的话(祖辈)。

性能消耗是非常大的,要么做递归,要么做多次查询。故,对于分类的数据量很大的情况,我推荐使用左右值,以减少查询上的麻烦。

代码如下:

_id

/**

+----------------------------------------------------------

* 构造函数

* @access public

* @return void

+----------------------------------------------------------

*/

public function __construct($left,$right,$id){

parent::__construct();

$this->_left = $left;

$this->_right = $right;

$this->_id = $id;

}

/**

+----------------------------------------------------------

* 根据node$this->_id得到该node的所有值

* @access public

* @param $nodeId

* @return array

+----------------------------------------------------------

*/

public function getNodeById($nodeId)

{

if($nodeId>0)

{

return $this->getById($nodeId);

}

else

{

throw_exception('未知$this->_id');

return false;

}

}

/**

+----------------------------------------------------------

* 获取父节点,含直属父类(type=1),所有父类:type=0

* @access public

* @param $nodeId int 节点$this->_id

* @return $parentNode array()

+----------------------------------------------------------

*/

public function getParentNode($nodeId,$type = 0)

{

if($nodeId == 0) throw_exception('未知$this->_id');;

$currentNode = $this->getNodeById($nodeId);

if($currentNode)

{

$condition = " ".$this->_left.'_left].' and '.$this->_right.' >'.$currentNode[$this->_right]." ";

if($type ==1) //直属父类

{

return $this->where($condition)->order($this->_left." DESC")->limit(1)->find();

// $sql = "SELECT * FROM ".TABLE_NAME." WHERE {$condition} ORDER BY ".$this->_left." DESC LIMIT 1";

// return mysql_query($sql) or die(mysql_error());

}

else if($type ==0)

{

return $this->where($condition)->findAll();

// $sql = "SELECT * FROM ".TABLE_NAME." WHERE {$condition} ";

// return mysql_query($sql) or die(mysql_error());

}

}

else

{

return false;

}

}

/**

+----------------------------------------------------------

* 当前节点下子孙节点总数.子孙总数=(当前节点的右值 - 当前节点的左值-1)/2

* @access public

* @param $node_id int 节点$this->_id

* @return $amount int 该节点下的子孙总数 *

+----------------------------------------------------------

*/

public function getChildCount($nodeId)

{

$currentNode = $this->getNodeById($nodeId);

if(!empty($currentNode))

{

return (int)($currentNode[$this->_right]-$currentNode[$this->_left] -1)/2;

}

}

/**

+----------------------------------------------------------

* 获取当前节点下所有子节点。 当 A子类的右节点=B子类左节点-1 则 A、B属于同一级别

* @access public

* @param $curentId

* @param $type int 0:当前节点下所有子类,1为当前节点下一级子类

* @return bool

+----------------------------------------------------------

*/

public function getChild($nodeId,$type=0)

{

$currentNode = $this->getNodeById($nodeId);

if($currentNode[$this->_left]-$currentNode[$this->_right] ==1)

{

return false; //当 该节点左值 - 右值=1 时,其下没有子节点。

}

else

{

$condition = $this->_left.'>'.$currentNode[$this->_left].' and '.$this->_right .'_right];

$child = $this->where($condition)->findAll();

if($type == 0)//所有子类

{

return $child;

}

else if($type ==1) //获取当前节点下一级分类

{

$subArr = array(); //一级子类

foreach ($child as $k=>$sub) {

//子类的左节点=父类左节点+1,则子类为第一个子类

if($sub[$this->_left]==$currentNode[$this->_left]+1)

{

//$right = $sub[$k][$this->_right]; //当前节点的右节点

$firstSub = $sub; //当前节点下第一个子类

array_push($subArr,$firstSub); //子类入栈

unset($child[$k]);

}

}

$rightVal = $firstSub[$this->_right]; //第一个子节点为比较标志

$childCount = count($child);//剩余子节点数

for($i=0;$i {

foreach ($child as $key => $sub2) {

if($rightVal == $sub2[$this->_left]-1)

{

$rightVal = $sub2[$this->_right]; //把循环当前的node的右节点当做比较值

array_push($subArr,$sub2);

unset($child[$key]);

}

}

}

return $subArr;

}

}

}

/**

+----------------------------------------------------------

* 返回当前节点的完整路径

* @access public

* @param $nodeId

* @return array

+----------------------------------------------------------

*/

public function getSinglePath($nodeId)

{

$sql = "select parent.* from __TABLE__ as node,__TABLE__ as parent where node.{$this->_left} between parent.{$this->_left}

AND parent.{$this->_right} AND node.{$this->_id} = {$nodeId} order by parent.{$this->_left}";

// echo $sql;

return $this->query($sql);

}

/**

+----------------------------------------------------------

* 添加子节点,分3种:0:在当前节点下最后追加一个子节点;1:在当前节点下追加第一个子节点;

2:在当前节点下的某个子节点后追加

代码如下:

* @access public

* @param $currentId int

* @param $nodeName string 新节点名称

* @param $targetId int 追加到当前节点下子节点的指定节点后

* @return bool

+----------------------------------------------------------

*/

public function addNode($nodeId,$newData,$type=0,$targetId=0)

{

if(empty($newData))

{

throw_exception('新分类不能为空');

}

$currentNode = $this->getNodeById($nodeId);

switch ($type) {

case 0:

$leftNode = $currentNode[$this->_right]; //新节点的左值为父节点的右值

$rightNode = $leftNode+1;

break;

case 1:

$leftNode = $currentNode[$this->_left]+1; //新节点的左值为父节点的左值+1

$rightNode = $leftNode+1;

break;

case 2:

$otherNode = $this->getNodeById($targetId);

$leftNode = $otherNode[$this->_right]+1;

$rightNode = $leftNode+1;

default:

break;

}

// $sql = "UPDATE ".TABLE_NAME." SET ".$this->_right."=".$this->_right."+2 WHERE ".$this->_right." >= ".$leftNode;

// $sql2 = "UPDATE ".TABLE_NAME." SET ".$this->_left."=".$this->_left."+2 WHERE ".$this->_left.">".$leftNode;

$this->setInc($this->_right,$this->_right.">=".$leftNode,2); //把所有右值大于新节点左值的节点的右值+2,注意效率

$this->setInc($this->_left,$this->_left.">".$leftNode,2); //把所有大于新节点的左值+2

$newData[$this->_left] = (int)$leftNode;

$newData[$this->_right] =(int) $rightNode;

return $this->add($newData);

}

/**

+----------------------------------------------------------

* 删除节点

* @access public

* @param type 操作类型,默认为0删除当前节点下的所有子节点,1为删除包括自身的节点

* @param $nodeId int 要删除的$this->_id

* @return bool

+----------------------------------------------------------

*/

public function rmNode($nodeId,$type =1)

{

$currentNode = $this->getNodeById($nodeId);

if($type == 1) //删除包含自身的节点

{

$sql = "DELETE FROM __TABLE__ WHERE ".$this->_left.">= {$currentNode[$this->_left]} AND ".$this->_right."<= {$currentNode[$this->_right]}";

$childCount = ($this->getChildCount($nodeId)+1)*2; //要更新的值

$sql2 = "UPDATE __TABLE__ SET ".$this->_right."=".$this->_right."-".$childCount." WHERE ".$this->_right.">".$currentNode[$this->_right];

$sql3 = "UPDATE __TABLE__ SET ".$this->_left."=".$this->_left."-".$childCount." WHERE ".$this->_left.">".$currentNode[$this->_left];

}

else //删除当前节点下的所有节点

{

$sql ="DELETE FROM __TABLE__ WHERE ".$this->_left."> {$currentNode[$this->_left]} AND ".$this->_right."< {$currentNode[$this->_right]}";

$childCount = $this->getChildCount($nodeId)*2; //要更新的值

$sql2 = "UPDATE __TABLE__ SET ".$this->_right."=".$this->_right ."-".$childCount." WHERE ".$this->_right.">=".$currentNode[$this->_right];

$sql3 = "UPDATE __TABLE__ SET ".$this->_left."=".$this->_left."-".$childCount." WHERE ".$this->_left.">".$currentNode[$this->_left];

}

$this->execute($sql);

$this->execute($sql2);

$this->execute($sql3);

return true;

}

/**

+----------------------------------------------------------

* 修改节点,名称等

* @access public

* @param $newData array()必须含有 要修改的$this->_id,k-v必须对齐,如arr['node_name'] = '商品'

* @return bool

+----------------------------------------------------------

*/

public function modiNode($newData)

{

if(!empty($newData))

{

$id = $newData[$this->_id];

unset($newData[$this->_id]);

return $this->save($newData,$this->_id.'='.$id);

}

}

}

?>

http://www.bkjia.com/PHPjc/327715.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/327715.htmlTechArticle以前一直使用父子无限分类,这种分类结构清晰,使用也简单。但若分类数量很大的话,在查询上性能不佳。比如在做导航菜单中,我要根...

本条技术文章来源于互联网,如果无意侵犯您的权益请点击此处反馈版权投诉

本文系统来源:php中文网

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值