php+mysql实现无限分类实例详解


Posted in PHP onJanuary 15, 2015

本文实例讲述了php+mysql实现无限分类的方法。分享给大家供大家参考。具体分析如下:

1、数据库通过设置父类ID来进行唯一索引,然后使用函数的递归调用实现无限分类;

2、数据库设计通过特定格式进行排列,然后使用mysql查询关键函数:concat,程序实现比较简单,首先我们假设有这样的一个三级分类,新闻→PHP新闻→PHP6.0出来了.

如果我们要查找“PHP6.0出来了”这条新闻,我们先点击新闻,然后再点击PHP新闻,就可以查出来了,也就是说我们可以通过祖父类一级一级地往下找,反过来我们只要知道一个子类的父类,就可以把它查找出来了,这样我们在设计数据库时就可以多设计一个父类id的字段就可以实现无限分类的功能了.

数据库代码如下:

这里我们建一个表"class"

CREATE TABLE `class` ( 

  `id` int(11) NOT NULL auto_increment COMMENT '分类id', 

  `f_id` int(11) NOT NULL COMMENT '父id', 

  `name` varchar(25) collate gbk_bin NOT NULL COMMENT '分类名称', 

  PRIMARY KEY  (`id`) 

) ENGINE=MyISAM  DEFAULT CHARSET=gbk COLLATE=gbk_bin AUTO_INCREMENT=1 ;

首先我们往数据库里插入'新闻'这个大分类,因为'新闻'是最大分类,上面没有父类了,所以我把它的f_id设置为0.
INSERT INTO `class` (`id`, `f_id`, `name`) VALUES(1, 0, '新闻');//id这个字段是自动增长的,可以不写值.

然后我们再往数据库里插入'PHP新闻'这个分类,它的父类'新闻'的id是1,所以它的f_id设置为1。

INSERT INTO `class` (`id`, `f_id`, `name`) VALUES(2, 1, 'PHP新闻');

然后我们再往数据库里插入'PHP6.0出来了'这个分类,它的父类'PHP新闻'的id是2,所以它的f_id设置为2。

INSERT INTO `class` (`id`, `f_id`, `name`) VALUES(3, 2, 'PHP6.0出来了');

同理,我们可以这样一直往下插入分类,也就达到了无限分类.

我们可以发现插入一个分类的原则关键是找到这个分类的父类的id,然后作为这个分类的f_id字段的值.

假设要插入跟'新闻'同一个级别的分类'技术',也就是说它也是最大分类,上面没有父类了,那么它的f_id也设置为0;

INSERT INTO `class` (`id`, `f_id`, `name`) VALUES(4, 0, '技术');

在'技术'下面又有一个分类'PHP技术',那么我们怎么插入呢,首先找到'PHP技术'的父类'技术'的id,然后作为自己的f_id字段的值.

INSERT INTO `class` (`id`, `f_id`, `name`) VALUES(5, 4, 'PHP技术');

看到这里,想必大家应该都明白怎么往数据库里插入各个分类了,就不再举例了,我们已经知道如何往数据库里插入各个分类了,那又如何把各个分类罗列出来呢?

php实例代码如下:

<?php 

header("Content-type:text/html;charset=utf-8");  

$db=new mysqli("localhost","root","","news_php100") ; //实例化一个数据库连接。使用这个前一定要确保已经加载了mysqli类库,或者用mysql_connect这个方式连接。  

if(mysqli_connect_errno()){ 

  echo "链接失败:".mysqli_connect_error(); 

  exit(); }  

$db->query("set names utf8"); 

$result=$db->query("select name from class where f_id=0"); //查找f_id=0的分类,也就是查找每一个大类。 

while($row=$result->fetch_assoc()){ 

      echo $row['name']."<br>";        //这样就把每个大类循环出来了。 

} 

//同样我们可以把新闻的子类循环出来。 

$result=$db->query("select * from class where f_id=1"); //查找f_id=1的分类,也就是查找'新闻'的子类。 

while($row=$result->fetch_assoc()){ 

      echo $row['name']." 

";        //这样就把'新闻'的子类循环出来了。注意:只是子类,不包括孙子类。 

} 

//写到这里,我们会发现一个问题,如果这个分类是10级分类,难道我们要写10个循环把它每个子类循环出来?如果是更多级分类呢,这样写显然是不现实的。 

//那又有什么办法解决呢?我们可以写一个递归的函数,把f_id作为参数传入,不断循环每一个f_id的值,也就是说把每一个f_id值的子类循环出来。 

//首先我们把各个分类的值保存在一个二维数组中,在下面的递归函数里有用。 

$result=$db->query("select * from class"); 

while($row=$result->fetch_assoc()){ 

     $arr[]=array($row[id],$row[f_id],$row[name]);    //每一行保存一个分类的id,f_id,name的信息。 

} 

function fenlei($f_id=0){     //$f_id初始化为0,也就是从最大分类开始循环. 

    global $arr;   //声明$arr为全局变量才可在函数里引用。 

    for($i=0;$i<count($arr);$i++){       //对每个分类进行循环。 

           if($arr[$i][1]==$f_id){         //$arr[$i][1]表示第$i+1个分类的f_id的值。开始$f_id=0,也就是把f_id=0的分类输出来。 

                 echo $arr[$i][2]."<br>"; //$arr[$i][1]表示第$i+1个分类的name的值。 

            fenlei($arr[$i][0]);   //$arr[$i][1]表示第$i+1个分类的id的值。进行递归,也就是把自己的id作为f_id参数把自己的子类再循环出来。 

} 

} 

} 

?>

三个字段id,parentid,name,算法也很简单递归,以前用递归的时候很傻,应该说极傻,因为在递归中通过查询数据表来获得子类的所有,最近开窍了,想到了一个地球人都能想得到的方法,下面是代码,一个class,代码如下:
<?php 

class Tree { 

 

 /** 

  * 从数据库查询出的所有分类信息 

  * @var array 

  */ 

 var $arr; 

/** 

 * 如下格式 

 *  var $arr = array( 

 1 => array('id'=>'1','parentid'=>0,'name'=>'一级栏目一'), 

 2 => array('id'=>'2','parentid'=>0,'name'=>'一级栏目二'), 

 3 => array('id'=>'3','parentid'=>1,'name'=>'二级栏目一'), 

 );*/ 

 

 /** 

  * 输出结构 

  * @var array 

  */ 

 var $tree = array(); 

 /** 

  * 树形递归的深度 

  * @var int 

  */ 

 var $deep = 1; 

 

 /** 

  * 生成树形的修饰符号 

  * @var array 

  */ 

 var $icon = array('│','├','└'); 

 /** 

  * 生成指定id的下级树形结构 

  * @param int $rootid 要获取树形结构的id 

  * @param string $add 递归中使用的前缀 

  * @param bool $parent_end 标识上级分类是否是最后一个 

  */ 

 function getTree($rootid = 0,$add = ”,$parent_end =true){ 

  $is_top = 1; 

  $child_arr = $this->getChild($rootid); 

  if(is_array($child_arr)){ 

   $cnt = count($child_arr); 

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

    $cid = $child['id']; 

    $child_child = $this->getChild($cid); 

    if($this->deep >1){ 

     if($is_top == 1 && $this->deep > 1){ 

      $space = $this->icon[1]; 

      if(!$parent_end) 

      $add .=  $this->icon[0]; 

      else $add .= '  '; 

     } 

 

     if($is_top == $cnt){ 

      $space = $this->icon[2]; 

      $parent_end = true; 

     }else { 

      $space = $this->icon[1]; 

      $parent_end = false; 

     } 

    } 

    $this->tree[] = array('spacer'=>$add.$k.$space, 

           'name'=>$child['name'], 

           'id'=>$cid 

    ); 

    $is_top++; 

 

    $this->deep++; 

    if($this->getChild($cid)) 

    $this->getTree($cid,$add,$parent_end); 

    $this->deep?; 

   } 

  } 

  return $this->tree; 

 } 

 

 /** 

  * 获取下级分类数组 

  * @param int $root 

  */ 

 function getChild($root = 0){ 

 

  $a = $child = array(); 

  foreach($this->arr as $id=>$a){ 

   if($a['parentid'] == $root){ 

    $child[$a['id']] = $a; 

   } 

  } 

  return $child?$child:false; 

 } 

 /** 

  * 设置源数组 

  * @param $arr 

  */ 

 function setArr($arr = array()){ 

  $this->arr = $arr; 

 } 

} 

?>

通过一次查询把结构保存进一个数组,再数组进行递归运算,无疑极大的提高了程序运行效率,使用代码很简单.

希望本文所述对大家的php程序设计有所帮助。

PHP 相关文章推荐
基于PHP+MySQL的聊天室设计
Oct 09 PHP
PHP生成网页快照 不用COM不用扩展.
Feb 11 PHP
PHP缩略图等比例无损压缩,可填充空白区域补充色
Jun 10 PHP
完美的2个php检测字符串是否是utf-8编码函数分享
Jul 28 PHP
使用YUI+Ant 实现JS CSS压缩
Sep 02 PHP
开启PHP Static 关键字之旅模式
Nov 13 PHP
基于PHP如何把汉字转化为拼音
Dec 11 PHP
Yii2分页的使用及其扩展方法详解
May 23 PHP
PHP实现上传图片到 zimg 服务器
Oct 19 PHP
php分享朋友圈的实现代码
Feb 18 PHP
php实现的生成排列算法示例
Jul 25 PHP
Laravel中GraphQL接口请求频率实战记录
Sep 01 PHP
php截取html字符串及自动补全html标签的方法
Jan 15 #PHP
php在linux下检测mysql同步状态的方法
Jan 15 #PHP
php中静态类与静态变量用法的区别分析
Jan 15 #PHP
将FCKeditor导入PHP+SMARTY的实现方法
Jan 15 #PHP
php通过Chianz.com获取IP地址与地区的方法
Jan 14 #PHP
php中JSON的使用与转换
Jan 14 #PHP
php rsa加密解密使用详解
Jan 14 #PHP
You might like
浅析php中常量,变量的作用域和生存周期
2013/08/10 PHP
php调用C代码的实现方法
2014/03/11 PHP
PHP生成等比缩略图类和自定义函数分享
2014/06/25 PHP
PHP仿微信多图片预览上传实例代码
2016/09/13 PHP
PHP进制转换实例分析(2,8,16,36,64进制至10进制相互转换)
2017/02/04 PHP
PHP静态延迟绑定和普通静态效率的对比
2017/10/20 PHP
PHP实现的解汉诺塔问题算法示例
2018/08/06 PHP
二级域名转向类
2006/11/09 Javascript
js电信网通双线自动选择技巧
2008/11/18 Javascript
JavaScript中的prototype.bind()方法介绍
2014/04/04 Javascript
利用JavaScript的AngularJS库制作电子名片的方法
2015/06/18 Javascript
jQuery实现美观的多级动画效果菜单代码
2015/09/06 Javascript
MUI顶部选项卡的用法(tab-top-webview-main)详解
2017/10/08 Javascript
详解用webpack的CommonsChunkPlugin提取公共代码的3种方式
2017/11/09 Javascript
JS设计模式之访问者模式定义与用法分析
2018/02/05 Javascript
JavaScript高级函数应用之分时函数实例分析
2018/08/03 Javascript
Next.js项目实战踩坑指南(笔记)
2018/11/29 Javascript
微信公众号网页分享功能开发的示例代码
2020/05/27 Javascript
Vue指令实现OutClick的示例
2020/11/16 Javascript
Python爬虫实现爬取京东手机页面的图片(实例代码)
2017/11/30 Python
Python数据可视化:顶级绘图库plotly详解
2019/12/07 Python
利用Pytorch实现简单的线性回归算法
2020/01/15 Python
python3+opencv 使用灰度直方图来判断图片的亮暗操作
2020/06/02 Python
百思买加拿大:Best Buy Canada
2018/03/20 全球购物
女孩每月服装订阅盒:kidpik
2019/04/17 全球购物
学生自我鉴定模板
2013/12/30 职场文书
信息技术专业大学生职业生涯规划书
2014/01/24 职场文书
三年级音乐教学反思
2014/01/28 职场文书
大学生毕业自我鉴定范文
2014/02/03 职场文书
模具设计与制造专业推荐信
2014/02/16 职场文书
行政监察建议书
2014/05/19 职场文书
医院科室评语
2015/01/04 职场文书
国情备忘录观后感
2015/06/04 职场文书
2016年最美孝心少年事迹材料
2016/02/26 职场文书
Golang全局变量加锁的问题解决
2021/05/08 Golang
一篇文章告诉你如何实现Vue前端分页和后端分页
2022/02/18 Vue.js