프로젝트 아이디어 분석: PHP 프로젝트는 분류를 사용해야 하는데 몇 단계로 나누어질지 모르기 때문에 무한 분류로 만들고 싶습니다.
처음에는 이전처럼 데이터베이스에 다음과 같이 4개의 값을 생성할 것이라고 생각했습니다.
id: 자동 증가 | pid: 상위 클래스 ID | xid: 정렬 ID | 클래스 이름
나중에는 이게 데이터를 읽어 수정을 할 때 불편할 것 같고, 특히 제품을 읽을 때 불편할 것 같아서 다음 해결 방법으로 변경했습니다.
Mysql 테이블에 새 필드가 추가되었습니다. 데이터베이스 는 이제 다음과 같습니다.
테이블 이름 w_faqclass: id: 자동 증가 | pid: 상위 클래스 ID | xid: 정렬 ID | 클래스 이름: 카테고리 이름: 수준
정의:
레벨 1 분류, pid는 0, 순위는 "/"
2차 분류, pid는 1차 분류의 id, Rank는 "/1차 분류의 id/"
3차 분류, pid는 2차 분류의 id, Rank는 "/1차 분류의 id/2차 분류의 id/"
등등...
1. 기본 기능
/* 利于递归返回已经进行了排序的无限级分类的数组 不想用递归的话也可以用 like 来获取后再进行排序,我比较懒,就不写那种获取方式了,其实用 like 更好,推荐用那种方式 $datatable : 数据表名 $startid : 开始父类ID $wheretColumns :父类列名 $xColumns : 排序列名 $xtype : 排序方式 $returnArr : 返回数组 */ function ReadClass($datatable,$startid,$xtype,$returnArr){ $db = $datatable; $sid = $startid; $xtype = $xtype; $lu = $returnArr; $sql = "select * from `".$db."` where `pid`='".$sid."' order by xid ".$xtype.";"; $cresult= mysql_query($sql); if(mysql_num_rows($cresult)>0){ while($rs = mysql_fetch_array($cresult)){ $lunum = count($lu); $lu[$lunum]['id'] = $rs['id']; $lu[$lunum]['pid'] = $rs['pid']; $lu[$lunum]['rank'] = $rs['rank']; $lu[$lunum]['classname']= $rs['classname']; $lu[$lunum]['xid'] = $rs['xid']; $lu = ReadClass($db,$rs['id'],$xtype,$lu); } } return $lu; } /* 查询某表中的某个值,只会返回一个值 $datatable : 数据表名 $wherevalue : 条件值 $selectColumns : 查询列名 $whereColumns : 条件列 */ function SelectValue($datatable,$wherevalue,$selectColumns,$whereColumns){ $sql = "select `".$selectColumns."` from `".$datatable."` where `".$whereColumns."`='".$wherevalue."';"; $result = mysql_query($sql); while($rs = mysql_fetch_array($result)){ return $rs[$selectColumns]; } }
2. 카테고리 추가(선택에서 바로 선택 가능)
<?php $classArr = ReadClass('w_faqclass','0','asc',array()); $canum = count($classArr); echo "<select name='pid'>"; echo "<option value='0'>主分类</option>"; for($i=0; $i<$canum; $i++){ $rankArr = split("/",$classArr[$i]['rank']); $ranknum = count($rankArr); $t = ""; for($j=1; $j<$ranknum; $j++){ //用于格式化显示子类 $t .= "├┄┄"; } echo "<option value='".$classArr[$i]['id']."'>".$t.$classArr[$i]['classname']."</option>"; } echo "</select>" ?> //保存时的操作,需要判断是否为主分类,当为主类时, rank 值设为 / //查询父类的 rank 值,用父类的 rank 加上 父类的 id 值 if($pid != 0){ $pidrank = SelectValue('w_faqclass',$pid,'rank','id'); $rank = $pidrank.$pid."/"; }else{ $rank = "/"; }
3. 분류 수정
<?php /* 注意,因为是修改,在此页面加载时已将当前分类的所有值读出来了,对应是:$pid,$rank */ $classArr = ReadClass('w_faqclass','0','asc',array()); $canum = count($classArr); echo "<select name='pid'>"; echo "<option value='0'>主分类</option>"; for($i=0; $i<$canum; $i++){ // 因为是修改,所以当前分类不能选择自身或自身以下的分类,多加个 rank 值的优势啊,哈哈,以前做单pid值的时候这里还得用次递归查询 while($ids == $classArr[$i]['id'] || strstr($classArr[$i]['rank'],$rank.$ids."/")){ $i++; } $rankArr = split("/",$classArr[$i]['rank']); $ranknum = count($rankArr); $t = ""; for($j=1; $j<$ranknum; $j++){ $t .= "├┄┄"; } if($pid == $classArr[$i]['id']){ $selected = "selected"; }else{ $selected = ""; } echo "<option value='".$classArr[$i]['id']."' ".$selected.">".$t.$classArr[$i]['classname']."</option>"; } echo "</select>" ?> // 保存时的操作 // 要做到改动时该分类的所有子分类rank值都需要变动,选取得原来子分类通用到的 rank 值,也就是该分类的 rank值加上它的ID值 // 利于 mysql 的REPLACE语句进行替换 if($pid != 0){ $pidrank = SelectValue('w_faqclass',$pid,'rank','id'); $rank = $pidrank.$pid."/"; }else{ $rank = "/"; } $orank = SelectValue('w_faqclass',$ids,'rank','id').$ids."/"; $nrank = $rank.$ids."/"; mysql_query("UPDATE `w_faqclass` SET rank = REPLACE(rank,'".$orank."','".$nrank."');"); mysql_query("UPDATE `w_faqclass` SET `classname`='".$classname."',`xid`='".$xid."',`pid`='".$pid."',`rank`='".$rank."' where `id`='".$ids."';");
4. 삭제 및 쿼리는 간단합니다. 이에 대해서는 자세히 설명하지 않겠습니다. 삭제하기 전에 클래스 아래에 하위 클래스가 있는지 확인하세요.
$zid = SelectValue('w_faqclass',$ids,'id','pid'); if($zid>0){ ... }
위 내용은 php mysql에서 Infinitus 분류를 구현하는 방법입니다. 모든 분들의 학습에 도움이 되었으면 좋겠습니다.