php树形结构算法:&引用的妙用

广告位招租
扫码页面底部二维码联系

在最近的项目中,我需要从数据库中挑选出带【未经授权禁止转载】【作者:唐霜】有层级关系的数据,并根据记录的paren本文版权归作者所有,未经授权不得转载。【版权所有】唐霜 www.tangshuang.nett_id构建数据之间的父子关系,最终形成【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。一个树状结构的父子孙关系结构,并用数组的【关注微信公众号:wwwtangshuangnet】【版权所有,侵权必究】形式保存起来。经过搜索和研究,最终找到了本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。下方的算法代码,仅做参考。

【作者:唐霜】原创内容,盗版必究。【关注微信公众号:wwwtangshuangnet】【原创不易,请尊重版权】【版权所有】唐霜 www.tangshuang.net

首先,查询得到的数据以数组的形式返回:未经授权,禁止复制转载。

【本文受版权保护】【原创不易,请尊重版权】转载请注明出处:www.tangshuang.net原创内容,盗版必究。本文作者:唐霜,转载请注明出处。
array(
  array('id' => 5,'parent_id' => 0,...),
  array('id' => 9,'parent_id' => 0,...),
  array('id' => 12,'parent_id' => 5,...),
  array('id' => 19,'parent_id' => 5,...),
  array('id' => 21,'parent_id' => 0,...),
  array('id' => 23,'parent_id' => 12,...),
  array('id' => 24,'parent_id' => 12,...),
  array('id' => 26,'parent_id' => 23,...),
  array('id' => 29,'parent_id' => 9,...),
  array('id' => 31,'parent_id' => 12,...),
  ...
)

通过观察,我们可以发现顶级数组是由包含i【本文首发于唐霜的博客】原创内容,盗版必究。d,parent_id的数组组成的,这些【本文受版权保护】原创内容,盗版必究。数组之间存在着一定的父子关系,而且仅以p转载请注明出处:www.tangshuang.net【版权所有,侵权必究】arent_id的形式表现出来。现在,我【本文受版权保护】本文版权归作者所有,未经授权不得转载。们实际上想要得到下面的结果:

【转载请注明来源】【未经授权禁止转载】【原创不易,请尊重版权】
array(
  array(
    'id' => 5,
    'children' => array(
      array(
        'id' => 12,
        'children' => array(
          array(
            'id' => 23,
            'children' => array(
              array('id' => 26)
            )
          ),
          array('id' => 24),
          array('id' => 31)
        )
      ),
      array('id' => 19)
    )
  ),
  array(
    'id' => 9,
    'children' => array(
      array('id' => 29)
    )
  ),
  array('id' => 21),
  ...
)

也就是通过数组的形式,用一个childr本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。en元素来包含该元素的子元素,并且层级分【未经授权禁止转载】【原创不易,请尊重版权】明。

【未经授权禁止转载】转载请注明出处:www.tangshuang.net【关注微信公众号:wwwtangshuangnet】【原创不易,请尊重版权】未经授权,禁止复制转载。

那么如何来快速实现呢?我们通过下面一个函【本文受版权保护】原创内容,盗版必究。数来实现。

原创内容,盗版必究。【原创内容,转载请注明出处】本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。转载请注明出处:www.tangshuang.net
function array_tree($array) {
  $result = array();
  $tmp = array();
  foreach($array as $item) {
    if($item['parent_id'] == 0) {
      $i = count($result);
      $result[$i] = $item;
      $id = $item['id'];
      $tmp[$id] = &$result[$i];
    }
    else {
      $id = $item['id'];
      $parent_id = $item['parent_id'];
      $parent = $tmp[$parent_id];
      $i = count($parent['children']);
      $tmp[$parent_id]['children'][$i] = $item;
      $tmp[$id] = &$tmp[$parent_id]['children'][$i];
    }
  }
  return $result;
}

上面我用红色标注了&符号,具体是【访问 www.tangshuang.net 获取更多精彩内容】本文版权归作者所有,未经授权不得转载。什么意思,及其原理又是怎么实现的呢?

本文版权归作者所有,未经授权不得转载。【版权所有,侵权必究】转载请注明出处:www.tangshuang.net本文作者:唐霜,转载请注明出处。【未经授权禁止转载】

&引用是PHP中非常特殊的一种变原创内容,盗版必究。【原创内容,转载请注明出处】量引用方式。我们不过多深入的去讲解,简单【关注微信公众号:wwwtangshuangnet】【未经授权禁止转载】的讲,就是:

【版权所有】唐霜 www.tangshuang.net本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。【本文首发于唐霜的博客】

当$b = &$a时,$b和$a本文版权归作者所有,未经授权不得转载。【未经授权禁止转载】同时引用同一个内容(指针指向同一块内存)著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net,无论$a或$b谁发生变化,这个内容都会【版权所有,侵权必究】【转载请注明来源】发生变化,进而呈现为$a和$b保持同步的【作者:唐霜】【未经授权禁止转载】变化。

【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。【本文受版权保护】转载请注明出处:www.tangshuang.net
本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。

要解除这种引用,只有先将引用释放。可是并未经授权,禁止复制转载。未经授权,禁止复制转载。没有一种运算可以达到释放引用的目的,所以转载请注明出处:www.tangshuang.net【版权所有,侵权必究】我们可以通过unset($a)的方式,也转载请注明出处:www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】可以通过重新引用另外一个变量来释放原来的未经授权,禁止复制转载。【转载请注明来源】引用(但是又和其他变量引用,比如$b =【本文受版权保护】【关注微信公众号:wwwtangshuangnet】 &$c)。

原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net【作者:唐霜】

这时,我们再来分析上面的实现函数。【作者:唐霜】

【关注微信公众号:wwwtangshuangnet】【访问 www.tangshuang.net 获取更多精彩内容】【版权所有】唐霜 www.tangshuang.net【转载请注明来源】

$tmp[$id] = &$re【原创不易,请尊重版权】【版权所有】唐霜 www.tangshuang.netsult[$i];

本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】原创内容,盗版必究。

这一句保证了当无论$tmp[$id]还是原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net$result[$i]发生变化,都会让另【访问 www.tangshuang.net 获取更多精彩内容】本文版权归作者所有,未经授权不得转载。外一个值同时发生变化。而

未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。

$tmp[$parent_id][本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net216;children’]【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】[$i] = $item;

未经授权,禁止复制转载。【未经授权禁止转载】【访问 www.tangshuang.net 获取更多精彩内容】【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。

实际上导致$result[$parent【原创不易,请尊重版权】著作权归作者所有,禁止商业用途转载。_id]发生了变化。

本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。本文作者:唐霜,转载请注明出处。

用我们上面的数据进行演示验证:【本文受版权保护】

【原创内容,转载请注明出处】未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。【本文首发于唐霜的博客】著作权归作者所有,禁止商业用途转载。

[0] => array(R未经授权,禁止复制转载。未经授权,禁止复制转载。16;id’ => 5未经授权,禁止复制转载。【未经授权禁止转载】,’parent_id本文作者:唐霜,转载请注明出处。著作权归作者所有,禁止商业用途转载。217; => 0,…【原创不易,请尊重版权】【原创不易,请尊重版权】)

原创内容,盗版必究。原创内容,盗版必究。【本文首发于唐霜的博客】未经授权,禁止复制转载。【作者:唐霜】

$result[0] = array(&【原创不易,请尊重版权】原创内容,盗版必究。#8216;id’ =>【原创不易,请尊重版权】【版权所有,侵权必究】; 5,’parent_id本文作者:唐霜,转载请注明出处。【原创不易,请尊重版权】’ => 0,R【版权所有】唐霜 www.tangshuang.net【未经授权禁止转载】30;)

【关注微信公众号:wwwtangshuangnet】【原创内容,转载请注明出处】【关注微信公众号:wwwtangshuangnet】未经授权,禁止复制转载。【本文首发于唐霜的博客】

$tmp[5] = &$resu【转载请注明来源】本文作者:唐霜,转载请注明出处。lt[0]  // 第一个&出现本文作者:唐霜,转载请注明出处。【未经授权禁止转载】

原创内容,盗版必究。【原创不易,请尊重版权】著作权归作者所有,禁止商业用途转载。

$tmp[5] = array(R著作权归作者所有,禁止商业用途转载。【本文首发于唐霜的博客】16;id’ => 5未经授权,禁止复制转载。【原创不易,请尊重版权】,’parent_id著作权归作者所有,禁止商业用途转载。【作者:唐霜】217; => 0,…未经授权,禁止复制转载。【未经授权禁止转载】)  // 内部结果

本文版权归作者所有,未经授权不得转载。【作者:唐霜】本文版权归作者所有,未经授权不得转载。
本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net未经授权,禁止复制转载。【原创不易,请尊重版权】

而当执行往下执行,碰到array(【作者:唐霜】本文作者:唐霜,转载请注明出处。216;id’ => 【未经授权禁止转载】【版权所有】唐霜 www.tangshuang.net12,’parent_id&本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】#8217; => 5,̷【版权所有】唐霜 www.tangshuang.net【版权所有】唐霜 www.tangshuang.net0;)时,发生了:

【访问 www.tangshuang.net 获取更多精彩内容】本文版权归作者所有,未经授权不得转载。【未经授权禁止转载】

[2]=> array(̵【版权所有,侵权必究】转载请注明出处:www.tangshuang.net6;id’ => 12【转载请注明来源】【原创内容,转载请注明出处】,’parent_id本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】217; => 5,…【关注微信公众号:wwwtangshuangnet】原创内容,盗版必究。)

【关注微信公众号:wwwtangshuangnet】【本文受版权保护】本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.net

$tmp[5][‘child本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。ren’][0] = arr【作者:唐霜】本文版权归作者所有,未经授权不得转载。ay(‘id’ 未经授权,禁止复制转载。著作权归作者所有,禁止商业用途转载。=> 12,’pare转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.netnt_id’ => 5【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】,…) // ①

转载请注明出处:www.tangshuang.net【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。

由于&引用的关系,所以实际上【版权所有,侵权必究】

【关注微信公众号:wwwtangshuangnet】【本文受版权保护】原创内容,盗版必究。

$result[0]原创内容,盗版必究。[‘childrenR【作者:唐霜】【作者:唐霜】17;][0] = array(R【关注微信公众号:wwwtangshuangnet】【本文受版权保护】16;id’ => 1原创内容,盗版必究。本文作者:唐霜,转载请注明出处。2,’parent_id&#著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net8217; => 5,…原创内容,盗版必究。【未经授权禁止转载】;) // 内部结果

原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。【作者:唐霜】本文版权归作者所有,未经授权不得转载。
【访问 www.tangshuang.net 获取更多精彩内容】【未经授权禁止转载】【转载请注明来源】转载请注明出处:www.tangshuang.net

这个时候$temp[5]是一个已经包含c著作权归作者所有,禁止商业用途转载。【未经授权禁止转载】hildren子元素的二维数组,如果往下转载请注明出处:www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】执行遇到array(‘id&著作权归作者所有,禁止商业用途转载。【本文受版权保护】#8217; => 23,R本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】17;parent_id’ 著作权归作者所有,禁止商业用途转载。【本文首发于唐霜的博客】=> 12,…)时,你本文作者:唐霜,转载请注明出处。【版权所有,侵权必究】就会发现

【作者:唐霜】【本文首发于唐霜的博客】【原创内容,转载请注明出处】【转载请注明来源】

[5]=> array(̵著作权归作者所有,禁止商业用途转载。【作者:唐霜】6;id’ => 23本文版权归作者所有,未经授权不得转载。【版权所有,侵权必究】,’parent_id【未经授权禁止转载】【作者:唐霜】217; => 12,…【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】;)

【版权所有,侵权必究】本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。本文作者:唐霜,转载请注明出处。

$tmp[12][‘chil转载请注明出处:www.tangshuang.net【原创不易,请尊重版权】dren’][0] = ar【本文受版权保护】【未经授权禁止转载】ray(‘id’【关注微信公众号:wwwtangshuangnet】【未经授权禁止转载】 => 23,’par【作者:唐霜】转载请注明出处:www.tangshuang.netent_id’ => 【原创内容,转载请注明出处】转载请注明出处:www.tangshuang.net12,…)

本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】原创内容,盗版必究。转载请注明出处:www.tangshuang.net【版权所有,侵权必究】

$result[【未经授权禁止转载】?【版权所有,侵权必究】][‘children原创内容,盗版必究。本文作者:唐霜,转载请注明出处。217;][0] = array(著作权归作者所有,禁止商业用途转载。【版权所有】唐霜 www.tangshuang.net216;id’ => 【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。23,’parent_id&【本文受版权保护】【本文受版权保护】#8217; => 12,R【本文受版权保护】原创内容,盗版必究。30;) // 我们以为会发生的结果,其【版权所有】唐霜 www.tangshuang.net【原创不易,请尊重版权】实并没有发生

著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。【版权所有,侵权必究】著作权归作者所有,禁止商业用途转载。
【本文受版权保护】【未经授权禁止转载】转载请注明出处:www.tangshuang.net转载请注明出处:www.tangshuang.net

因为没有找到一个$tmp[12] = &【原创不易,请尊重版权】本文版权归作者所有,未经授权不得转载。amp; $result[?],也就是说本文版权归作者所有,未经授权不得转载。本文作者:唐霜,转载请注明出处。没有一个与$tmp[12]对应的引用,$本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】tmp[12]的变化不会引起其他任何值的【原创不易,请尊重版权】【作者:唐霜】变化,$tmp[12]这个家伙会在返回值本文作者:唐霜,转载请注明出处。【作者:唐霜】中丢失(因为返回值是$result)。

【作者:唐霜】【未经授权禁止转载】转载请注明出处:www.tangshuang.net

【关注微信公众号:wwwtangshuangnet】$tmp[$id] = &$tm【本文受版权保护】本文版权归作者所有,未经授权不得转载。p[$parent_id][‘未经授权,禁止复制转载。未经授权,禁止复制转载。;children’][$i原创内容,盗版必究。未经授权,禁止复制转载。];这一句起到了关键性作用。加上这一句之后,【原创内容,转载请注明出处】未经授权,禁止复制转载。我们再来跑一遍array(‘【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。id’ => 23,&【关注微信公众号:wwwtangshuangnet】【本文受版权保护】#8217;parent_id̵【本文首发于唐霜的博客】本文作者:唐霜,转载请注明出处。7; => 12,…)【原创不易,请尊重版权】【关注微信公众号:wwwtangshuangnet】这个元素。

【转载请注明来源】【未经授权禁止转载】【原创内容,转载请注明出处】

[2]=> array(̵【本文首发于唐霜的博客】【本文首发于唐霜的博客】6;id’ => 12本文版权归作者所有,未经授权不得转载。著作权归作者所有,禁止商业用途转载。,’parent_id【未经授权禁止转载】【作者:唐霜】217; => 5,…【作者:唐霜】【本文首发于唐霜的博客】)

未经授权,禁止复制转载。【作者:唐霜】【版权所有】唐霜 www.tangshuang.net

$tmp[5][‘child本文版权归作者所有,未经授权不得转载。【本文受版权保护】ren’][0] = arr【本文受版权保护】【转载请注明来源】ay(‘id’ 【作者:唐霜】【原创不易,请尊重版权】=> 12,’pare转载请注明出处:www.tangshuang.net【作者:唐霜】nt_id’ => 5【版权所有】唐霜 www.tangshuang.net原创内容,盗版必究。,…)  // 重新从①开始【原创内容,转载请注明出处】【本文受版权保护】演示

未经授权,禁止复制转载。本文作者:唐霜,转载请注明出处。【关注微信公众号:wwwtangshuangnet】

$result[0]【版权所有】唐霜 www.tangshuang.net[‘childrenR本文版权归作者所有,未经授权不得转载。转载请注明出处:www.tangshuang.net17;][0] = 本文作者:唐霜,转载请注明出处。array(‘id̵【转载请注明来源】【访问 www.tangshuang.net 获取更多精彩内容】7; => 12,’p【本文受版权保护】【转载请注明来源】arent_id’ =>【原创不易,请尊重版权】【版权所有,侵权必究】; 5,…) // 内部结果

原创内容,盗版必究。【本文受版权保护】【作者:唐霜】【本文首发于唐霜的博客】本文版权归作者所有,未经授权不得转载。

$tmp[12] = &$tmp本文版权归作者所有,未经授权不得转载。【作者:唐霜】[5][‘children&【访问 www.tangshuang.net 获取更多精彩内容】【关注微信公众号:wwwtangshuangnet】#8217;][0]  // 第二个&a【原创内容,转载请注明出处】【本文首发于唐霜的博客】mp;出现了

【关注微信公众号:wwwtangshuangnet】【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。【原创不易,请尊重版权】

[5]=> array(̵【版权所有,侵权必究】【原创内容,转载请注明出处】6;id’ => 23著作权归作者所有,禁止商业用途转载。【原创内容,转载请注明出处】,’parent_id【本文受版权保护】原创内容,盗版必究。217; => 12,…【访问 www.tangshuang.net 获取更多精彩内容】【未经授权禁止转载】;)

【原创内容,转载请注明出处】转载请注明出处:www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】

$tmp[12][‘chil本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。dren’][0] = ar转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。ray(‘id’本文版权归作者所有,未经授权不得转载。【未经授权禁止转载】 => 23,’par转载请注明出处:www.tangshuang.net【版权所有,侵权必究】ent_id’ => 本文作者:唐霜,转载请注明出处。【本文首发于唐霜的博客】12,…)

【版权所有】唐霜 www.tangshuang.net【版权所有】唐霜 www.tangshuang.net【本文首发于唐霜的博客】转载请注明出处:www.tangshuang.net【转载请注明来源】

由于第二个&引用的原因,实际上发本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。生了:

【版权所有,侵权必究】【访问 www.tangshuang.net 获取更多精彩内容】原创内容,盗版必究。【关注微信公众号:wwwtangshuangnet】

$tmp[5][‘child【未经授权禁止转载】【原创不易,请尊重版权】ren’][0][‘childrenR【版权所有】唐霜 www.tangshuang.net【原创不易,请尊重版权】17;][0] = array(R【本文首发于唐霜的博客】【本文受版权保护】16;id’ => 2【本文受版权保护】【原创内容,转载请注明出处】3,’parent_id&#【作者:唐霜】【未经授权禁止转载】8217; => 12,̷本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。0;) // 内部结果

【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】【版权所有】唐霜 www.tangshuang.net

$result[0]【转载请注明来源】[‘childrenR【原创不易,请尊重版权】【版权所有,侵权必究】17;][0][‘child未经授权,禁止复制转载。【原创不易,请尊重版权】ren’][0] = arr本文版权归作者所有,未经授权不得转载。【转载请注明来源】ay(‘id’ 原创内容,盗版必究。未经授权,禁止复制转载。=> 23,’pare【本文首发于唐霜的博客】【原创内容,转载请注明出处】nt_id’ => 1【版权所有】唐霜 www.tangshuang.net未经授权,禁止复制转载。2,…) // 内部结果

未经授权,禁止复制转载。【本文首发于唐霜的博客】原创内容,盗版必究。【访问 www.tangshuang.net 获取更多精彩内容】【原创内容,转载请注明出处】
原创内容,盗版必究。【原创内容,转载请注明出处】【版权所有,侵权必究】【作者:唐霜】著作权归作者所有,禁止商业用途转载。

这个时候你可能已经看出了端倪。父子关系变本文版权归作者所有,未经授权不得转载。【本文受版权保护】成了 5 > 12 > 23未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。,而这组关系,全部存储在了id=5的这个【关注微信公众号:wwwtangshuangnet】【访问 www.tangshuang.net 获取更多精彩内容】顶级元素中,以多重的children元素【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。实现了父子孙结构。

未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】

如此循环往复,你就会发现,无论你的父子关转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。系多深,这个函数都能准确的构建对应的父子【转载请注明来源】【本文首发于唐霜的博客】关系,最终形成树形结构存储在$resul未经授权,禁止复制转载。【转载请注明来源】t中,将其返回。

本文作者:唐霜,转载请注明出处。原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】原创内容,盗版必究。

该算法有一个问题,就是要求原始数组中,p著作权归作者所有,禁止商业用途转载。【原创内容,转载请注明出处】arent_id必须出现在该元素之前,否【本文首发于唐霜的博客】【原创内容,转载请注明出处】则将导致该元素无法加入到某个元素的chi著作权归作者所有,禁止商业用途转载。【作者:唐霜】ldren中去。

【转载请注明来源】著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。

2015-11-16 6790

为价值买单,打赏一杯咖啡

本文价值67.9RMB