草庐IT

Mysql 查询指定节点的所有子节点

周伯通之草堂笔记 2023-03-28 原文

原文链接:https://www.zhoubotong.site/post/92.html

通常我们直接通过递归查询来达到实现子节点数据获取的需求,这里不谈存储过程的实现,存储过程普通账号有权限限制,通常也不易于开发者维护,
这里介绍下纯mysql递归实现的方式:
测试数据可以通过之前的一篇文章来模拟。在正式介绍实现之前,我们先了解下几个mysql实现涉及的相关知识点:

Mysql用户变量

用户变量无需声明,直接赋值就行。用户变量名不区分大小写。名称的最大长度为64个字符。常用的赋值方式有:

方式一:使用 SET 赋值。

可以使用形如 set @变量名=变量值 或者 set@变量名:=变量值 的方式赋值。

SET @var_name = expr [, @var_name = expr] ...
或
SET @var_name := expr [, @var_name := expr] ...

方式二:使用 select 赋值。

select @变量名:=变量值
select @变量名:=字段名 from table where ... limit 1;

继续举个例子,表记录如下:

注意: 通过查询表给变量赋值时,需保证查询结果只有一条记录,如上result2的结果集这种查询了2条。

另外再介绍本文实现中涉及的另外2个mysql函数,这里就简单介绍下:


if(express1,express2,express3)条件语句:

if语句类似三目运算符,当exprss1成立时,执行express2,否则执行express3;


FIND_IN_SET(str,strlist),str 要查询的字符串,strlist 字段名 参数以”,”分隔 如 (1,2,3,6),查询字段(strlist)中包含(str)的结果.


concat_ws()函数, 表示concat with separator,即有分隔符的字符串连接:

select concat_ws(',','11','22',NULL); 返回 11,22。

下面进入本文正题,查询当前节点下的所有子节点:

select id
from (
        select t1.id,
            if(
                find_in_set(pid, @pids) > 0,
                @pids := CONCAT_WS(',',@pids, id),
                0
            ) as ischild
        from (
                select id,
                    pid
                from city t
                order by id
            ) t1,
            (
                select @pids := 11
            ) t2
    ) t3
where ischild != 0;

上面我们查询节点id=11(武汉市)下的所有节点。上面语句看似复杂,其实不难理解,我们来分下该sql是怎么实现结果集的。
我们先从最里面的子查询分析:
我们看到第二个from后面是跟了两张表:t1和t2, t2是一个用户变量,其结果集作为t2,

,t1表很好理解就是city的所有记录作为表t1,我们再看t3表是什么?

上面高亮部分即为t3的结果集,其目的就是将当前要查询的子节点id用逗号连接,

如果pid值在@pids中,则设置@pids用其用户变量+id逗号连接组成新字段ischild。因为@pids查询到匹配记录就重新赋值了,
所以大家不难理解其满足条件下的子节点。

上面就是关于递归查询的实现。当然还有另外一种查法:

SELECT t1.id 
FROM (SELECT id,pid FROM city WHERE pid IS NOT NULL) t1,
     (SELECT @pid := 11) t2
WHERE FIND_IN_SET(pid, @pid) > 0 
  AND @pid := concat(@pid, ',', id)
-- union select id from city where id = 11 order by id;

如果想查询结果包含自身ID(如上面的id=11),加上后边的union即可。

有关Mysql 查询指定节点的所有子节点的更多相关文章

  1. ruby - ECONNRESET (Whois::ConnectionError) - 尝试在 Ruby 中查询 Whois 时出错 - 2

    我正在用Ruby编写一个简单的程序来检查域列表是否被占用。基本上它循环遍历列表,并使用以下函数进行检查。require'rubygems'require'whois'defcheck_domain(domain)c=Whois::Client.newc.query("google.com").available?end程序不断出错(即使我在google.com中进行硬编码),并打印以下消息。鉴于该程序非常简单,我已经没有什么想法了-有什么建议吗?/Library/Ruby/Gems/1.8/gems/whois-2.0.2/lib/whois/server/adapters/base.

  2. ruby - 如何以所有可能的方式将字符串拆分为长度最多为 3 的连续子字符串? - 2

    我试图获取一个长度在1到10之间的字符串,并输出将字符串分解为大小为1、2或3的连续子字符串的所有可能方式。例如:输入:123456将整数分割成单个字符,然后继续查找组合。该代码将返回以下所有数组。[1,2,3,4,5,6][12,3,4,5,6][1,23,4,5,6][1,2,34,5,6][1,2,3,45,6][1,2,3,4,56][12,34,5,6][12,3,45,6][12,3,4,56][1,23,45,6][1,2,34,56][1,23,4,56][12,34,56][123,4,5,6][1,234,5,6][1,2,345,6][1,2,3,456][123

  3. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  4. ruby-on-rails - 在 ruby​​ .gemspec 文件中,如何指定依赖项的多个版本? - 2

    我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这

  5. ruby-on-rails - 跳过状态机方法的所有验证 - 2

    当我的预订模型通过rake任务在状态机上转换时,我试图找出如何跳过对ActiveRecord对象的特定实例的验证。我想在reservation.close时跳过所有验证!叫做。希望调用reservation.close!(:validate=>false)之类的东西。仅供引用,我们正在使用https://github.com/pluginaweek/state_machine用于状态机。这是我的预订模型的示例。classReservation["requested","negotiating","approved"])}state_machine:initial=>'requested

  6. ruby - 如果指定键的值在数组中相同,如何合并哈希 - 2

    我有一个这样的哈希数组:[{:foo=>2,:date=>Sat,01Sep2014},{:foo2=>2,:date=>Sat,02Sep2014},{:foo3=>3,:date=>Sat,01Sep2014},{:foo4=>4,:date=>Sat,03Sep2014},{:foo5=>5,:date=>Sat,02Sep2014}]如果:date相同,我想合并哈希值。我对上面数组的期望是:[{:foo=>2,:foo3=>3,:date=>Sat,01Sep2014},{:foo2=>2,:foo5=>5:date=>Sat,02Sep2014},{:foo4=>4,:dat

  7. ruby - Nokogiri 剥离所有属性 - 2

    我有这个html标记:我想得到这个:我如何使用Nokogiri做到这一点? 最佳答案 require'nokogiri'doc=Nokogiri::HTML('')您可以通过xpath删除所有属性:doc.xpath('//@*').remove或者,如果您需要做一些更复杂的事情,有时使用以下方法遍历所有元素会更容易:doc.traversedo|node|node.keys.eachdo|attribute|node.deleteattributeendend 关于ruby-Nokog

  8. ruby-on-rails - 在 Rails 和 ActiveRecord 中查询时忽略某些字段 - 2

    我知道我可以指定某些字段来使用pluck查询数据库。ids=Item.where('due_at但是我想知道,是否有一种方法可以指定我想避免从数据库查询的某些字段。某种反拔?posts=Post.where(published:true).do_not_lookup(:enormous_field) 最佳答案 Model#attribute_names应该返回列/属性数组。您可以排除其中一些并传递给pluck或select方法。像这样:posts=Post.where(published:true).select(Post.attr

  9. ruby - 当使用::指定模块时,为什么 Ruby 不在更高范围内查找类? - 2

    我刚刚被困在这个问题上一段时间了。以这个基地为例:moduleTopclassTestendmoduleFooendend稍后,我可以通过这样做在Foo中定义扩展Test的类:moduleTopmoduleFooclassSomeTest但是,如果我尝试通过使用::指定模块来最小化缩进:moduleTop::FooclassFailure这失败了:NameError:uninitializedconstantTop::Foo::Test这是一个错误,还是仅仅是Ruby解析变量名的方式的逻辑结果? 最佳答案 Isthisabug,or

  10. ruby - 获取模块中定义的所有常量的值 - 2

    我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c

随机推荐