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常用函数小技巧
Sep 11 PHP
php 将bmp图片转为jpg等其他任意格式的图片
Jun 29 PHP
建站常用13种PHP开源CMS比较
Aug 23 PHP
检查用户名是否已在mysql中存在的php写法
Jan 20 PHP
php给一组指定关键词添加span标签的方法
Mar 31 PHP
PHP curl伪造IP地址和header信息代码实例
Apr 27 PHP
php中PDO方式实现数据库的增删改查
May 17 PHP
PHP新建类问题分析及解决思路
Nov 19 PHP
php封装的smartyBC类完整实例
Oct 19 PHP
thinkPHP5.0框架模块设计详解
Mar 18 PHP
利用laravel+ajax实现文件上传功能方法示例
Aug 13 PHP
YII2框架使用控制台命令的方法分析
Mar 18 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
奉献出一个封装的curl函数 便于调用(抓数据专用)
2013/07/22 PHP
PHP中__autoload和Smarty冲突的简单解决方法
2016/04/08 PHP
PHP实现Unicode编码相互转换的方法示例
2020/11/17 PHP
总结PHP内存释放以及垃圾回收
2018/03/29 PHP
一个javascript图片阅览组件
2010/11/09 Javascript
利用JQuery动画制作滑动菜单项效果实现步骤及代码
2013/02/07 Javascript
很好用的js日历算法详细代码
2013/03/07 Javascript
jQuery实现监控页面所有ajax请求的方法
2015/12/10 Javascript
快速掌握Node.js模块封装及使用
2016/03/21 Javascript
最佳的JavaScript错误处理实践
2016/07/16 Javascript
基于构造函数的五种继承方法小结
2017/07/27 Javascript
解决vue build打包之后首页白屏的问题
2018/03/06 Javascript
webpack4.x下babel的安装、配置及使用详解
2019/03/07 Javascript
vue基础之模板和过滤器用法实例分析
2019/03/12 Javascript
Vue结合路由配置递归实现菜单栏功能
2020/06/16 Javascript
JS实现密码框效果
2020/09/10 Javascript
javascript实现点击小图显示大图
2020/11/29 Javascript
Node快速切换版本、版本回退(降级)、版本更新(升级)
2021/01/07 Javascript
[01:17:12]职来职往完美电竞专场
2014/09/18 DOTA
Python自动重试HTTP连接装饰器
2015/04/28 Python
Python学生信息管理系统修改版
2018/03/13 Python
python实现类之间的方法互相调用
2018/04/29 Python
用python生成(动态彩色)二维码的方法(使用myqr库实现)
2019/06/24 Python
使用python代码进行身份证号校验的实现示例
2019/11/21 Python
基于python实现破解滑动验证码过程解析
2020/05/28 Python
HTML5 本地存储和内容按需加载的思路和方法
2011/04/07 HTML / CSS
Html5新特性用canvas标签画多条直线附效果截图
2014/06/30 HTML / CSS
利用HTML5实现使用按钮控制背景音乐开关
2015/09/21 HTML / CSS
英国50岁以上人群的交友网站:Ourtime
2018/03/28 全球购物
函数指针的定义是什么
2016/08/14 面试题
2014年商场超市庆元旦活动方案
2014/02/14 职场文书
采购助理岗位职责
2014/02/16 职场文书
地下停车场租赁协议范本
2014/10/07 职场文书
优秀班组事迹材料
2014/12/24 职场文书
村干部任职承诺书
2015/01/21 职场文书
《战锤40K:暗潮》跳票至9月 公布新宣传片
2022/04/03 其他游戏