数据库设计:
create table `header`(  // 父表
 
parent int not null, //父级
  poster varchar(20) not null, //作者
  posted
datetime,            //时间
  title varchar(100),        //标题    
  children
tiny(1) default 0, //是否有回复(1)[YES] (0)[NO]
  postid int not null primary key
// 主键
);

create table `body`( //具体内容
  postid int not null primary
key,
  text text
);

代码示例(省略头尾,只显示大概内容区):
<?php
/*
 * expand_all()      全部展示    
 *
$expanded         标示那些已经展开的
 * $expand           那些要被展开
 *
$_GET['expand']   表示那些是要被展示的
 * $_GET['collapse']
表示那些是要折叠的
*/
session_start();

function
db_connect(){
  $conn = new mysqli("localhost", "root", "123456",
"dbname");
  $conn->query("set names utf8");
  return $conn
}
//
全部要展示出来
function expand_all(&$expanded){
  $sql = "select postid from
header where children = 1";
  $conn = db_connect();
  $result =
$conn->query($sql);
  while($row = $result->fetch_assoc()){
   
$expanded[$row['postid']] = true;
  }
}

// 调用函数 start开始的parent,
$row 标示行号,用于交替颜色
function display_tree($expanded, $row = 0, $start = 0){
 
echo "<table width='580'>\n";
  $tree = new treenode($start, 1, true,
-1, $expanded);
  $tree->display($row);
  echo
"</table>\n";
}
if(!isset($_GET['expand'])){
 
$_SESSION['expanded'] = array();
}

//
那些是要显示出来的
if(isset($_GET['expand'])){
  if($_GET['expand'] ==
'all'){
    expand_all($_SESSION['expanded']);
  }else{
   
$_SESSION[$_GET['expanded']] = true;    
  }
}

//
那些是要折叠
if(isset($_GET['collaspe'])){
  if($_GET['collaspe'] ==
'all'){
   $_SESSION['expanded'] = array();
  }else{
  
unset($_SESSION[$_GET['collapse']]);
  }
}
// class
class treenode
{
  public $m_postid;      // id
  public $m_children;    // 是否有回复
 
public $m_depth;      // 深度,就是递归深入的深度
  public $m_childlist; // 是一个数组,表示要展示的
id

function __construct($postid, $children, $expand, $depth,
$expanded){
   $this->m_postid = $postid;
   $this->m_children =
$chilren;
   $this->m_depth = $depth;
   $this->m_childlist =
array();

if($expand && $this->m_children){
    $conn =
db_connect();
    $query = "select * from header where parent = '" .
$this->postid .  "' order by posted";
    $result =
$conn->query($query);
    for($count=0; $row = $conn->fetch_assoc();
$count++){
      if($expanded[$row['postid']] == true){
       $expand =
true;
      }else{
       $expand = false;
      }
     
$this->m_childlist[$count] = new treenode($row['postid'],
$row['children'],$expand, $depth+1, $expanded);
    }
   }
  }

function display($row){
    if($this->depth>-1){
      echo
"<tr><td bgcolor=\"";
      if($row%2){
    echo
"#cccccc\">";
      }else{
        echo "#ffffff\">"
     
}
      // 缩进
      for($i=0; $i<$this->depth; $i++){
    echo
"&nbsp;&nbspp;";
      }
      if($this->children &&
sizeof($this->m_childlist)){
    echo "<a href=\"index.php?collapse=\""
. $this->postid . "\"><img
src=\"images/--.gif\"/></a>";
      }else
if($this->children){
    echo "<a href=\"index.php?expand=\"" .
$this->postid . "\"><img
src=\"images/++.gif\"/></a>";
      }else{
    echo
"&nbsp;&nbsp;";
      }
      echo "<a
href=\"view.php?parent={$this->postid}\">" . $this->postid
."</a></td></tr>\n";
      $row++;
    }
   
$num_children = sizeof($this->m_childlist);
    for($i=0;
$i<$num_children; $i++){
      $row =
$this->m_childlist[$i]->display($row);
    }
    return $row;
 
}

}
//
最后调用
display_tree($_SESSION['expanded']);
?>

图片效果如下:

首页初始化时如下:

全部展开时如下

以下是部分展开的两个效果图:

web 中 bbs 例子(多次递归)的更多相关文章

  1. HT for Web中3D流动效果的实现与应用

    流动效果在3D领域有着广泛的应用场景,如上图中医学领域可通过3D的流动直观的观察人体血液的流动,燃气领域可用于监控管道内流动的液体或气体的流向.流速和温度等指标. 如今企业数据中心机房普遍面临着设备散 ...

  2. 基于gSOAP使用头文件的C语言版web service开发过程例子

    基于gSOAP使用头文件的C语言版web service开发过程例子 一服务端 1 打开VS2005,创建一个工程,命名为calcServer. 2 添加一个头文件calc.h,编辑内容如下: 1// ...

  3. 【spring实战第五版遇到的坑】3.1中的例子报错

    按照书中的例子,一直做到第3.1章使用JDBC读写数据时,在提交设计的taco表单时,报了如下的异常信息: Failed to convert property value of type java. ...

  4. Web中的积累:外观模式 Facade

    摘要: 原创出处: http://www.cnblogs.com/Alandre/ 泥沙砖瓦浆木匠 希望转载,保留摘要,谢谢! 壹 前言 目测好久没写文章了,距离上一篇文章也有二十多天.我是怎么了?哈 ...

  5. Html的本质及在web中的作用

    概要 本文以一个Socket程序为例由浅及深地揭示了Html的本质问题,同时介绍了作为web开发者我们在开发网站时需要做的事情 Html的本质以及开发需要的工作 1.服务器-客户端模型 其实,对于所有 ...

  6. [翻译]Spring MVC RESTFul Web Service CRUD 例子

    Spring MVC RESTFul Web Service CRUD 例子 本文主要翻译自:http://memorynotfound.com/spring-mvc-restful-web-serv ...

  7. C#中的函数式编程:递归与纯函数(二) 学习ASP.NET Core Razor 编程系列四——Asp.Net Core Razor列表模板页面

    C#中的函数式编程:递归与纯函数(二)   在序言中,我们提到函数式编程的两大特征:无副作用.函数是第一公民.现在,我们先来深入第一个特征:无副作用. 无副作用是通过引用透明(Referential ...

  8. java web中各种context的关系

    我举得这篇文章解决了我的很多疑惑,理清了我以前不太清楚的Context关系,读懂这篇文章很有助于理解源码, 原文链接在这里:https://www.jianshu.com/p/2537e2fec546 ...

  9. 优化Web中的性能

    优化Web中的性能 简介 web的优化就是一场阻止http请求最终访问到数据库的战争. 优化的方式就是加缓存,在各个节点加缓存. web请求的流程及节点 熟悉流程及节点,才能定位性能的问题.而且优化的 ...

随机推荐

  1. Android开发中如何强制横屏和强制竖屏设置

    Android开发中如何强制横屏和强制竖屏设置 强制横屏设置: 按照下面代码示例修改Activity的onResume方法 @Override protected void onResume() { ...

  2. 从bug中学习怎么写代码

    博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:从bug中学习怎么写代码.

  3. Android之发送短信和接收验证码

      最近项目需求需要发送短信和接收验证码并将验证码显示在输入框中 以下是我的记录    前提---权限     <uses-permission android:name="andro ...

  4. 在JAVA中线程到底起到什么作用

    这是javaeye上非常经典的关于线程的帖子,写的非常通俗易懂的,适合任何读计算机的同学. 线程同步 我们可以在计算机上运行各种计算机软件程序.每一个运行的程序可能包括多个独立运行的线程(Thread ...

  5. AndroidのUI设计研究(一)——自定义ProgressBar

    最近迷上进度条,使用进度条可以增强用户体验,让用户心里有个底,再无奈的等待中体会loading的乐趣. 记得以前优乐美的官网,进入主页加载资源,显示给用户看的就是,炫彩背景下,一个杯子里的奶茶随着加载 ...

  6. sizeof strlen strncpy用法总结 结构体实际所占内存大小 以及memset用法

    sizeof测类型(数组名除外) strlen测实际长度 strncpy返回指针类型 #include <stdio.h> #include <stdlib.h> #inclu ...

  7. FPGA异步时钟设计中的同步策略

    1 引言    基于FPGA的数字系统设计中大都推荐采用同步时序的设计,也就是单时钟系统.但是实际的工程中,纯粹单时钟系统设计的情况很少,特别是设计模块与外围芯片的通信中,跨时钟域的情况经常不可避免. ...

  8. KindEditor 修改多图片上传显示限制大小和张数

    在使用KindEditor的时候用到多图片上传时,提示有最多上传20张图片,单张图片容量不超过1MB: 修改的文件的地方在:kindeditor\plugins\multiimage\multiima ...

  9. react 编写组件 五

    看以下示例了解如何定义一个组件 // 定义一个组件LikeButton var LikeButton = React.createClass({ // 给state定义初始值 getInitialSt ...

  10. 破解SQLYog30天试用方法

    开始-运行-regedit ,进入注册表,在 \HEYK_CURRENT_USER\Software\{FCE28CE8-D8CE-4637-9BC7-93E4C0D407FA}下的InD保存着SQL ...