oracle 递归应用(挺复杂的)
最近做数据过滤觉得很有必要记录下整个过程,说不定下次就不知道了。
废话不多说开始:
表结构:
企业表(自关联,采用树的形式记录分子公司)
区域表(自关联,采用树的形式记录省/市/县/乡,数据量大)
公司管辖区域表(公司-区域的映射表,一对多,记录了公司所具有的管辖区域)
场景:根据自己的管辖区域构建区域查询条件
分子公司都有可能维护了管辖区域,上级具有下级的管辖权限(原因:上级公司有可能没有维护下级公司具有的管辖区域,而上级公司要管理下级就不需具有下级的管辖区域,所用需同时把下级的管辖区域给上级公司一次类推),在地址维护时,维护到任何级别都具有其下级区域的管辖(比如:某个公司维护了成都,它就具有成都和成都下的全部区域的管辖权限),维护到任何级别都具有该区域上级权限(比如:某个公司维护了成都/武侯区,那它就具有武侯区,成都,四川三个权限)
说到这是不是已经晕了,(拿到公司的管辖区域进行一次向上递归和一次向下递归然后合并结果集)
第一步:递归出子公司 (4319 是总部的id)
sql:SELECT *
FROM C_COMPANY
START WITH corp_id =4319
CONNECT BY prior corp_id= corp_parentid
结果:
第二步:拿出子公司的管辖区域 c_dealers_zone 管辖区域表1,c_store_zone 管辖区域表2
sql:SELECT zid
FROM
( SELECT d.c_z_id AS zid ,d.c_com_id AS comid FROM c_dealers_zone d
UNION
SELECT s.csz_zid AS zid ,s.csz_sid AS comid FROM c_store_zone s
)
WHERE comid IN
(SELECT corp_id
FROM C_COMPANY
START WITH corp_id =4319
CONNECT BY prior corp_id= corp_parentid
)
结果: 表示4319和其下级公司已维护了的管辖区域。
第三步 将管辖区域id关联到区域信息表已拿出详细信息,并同时向上递归拿出上级(以便检索出下级)
sql:SELECT DISTINCT *
FROM c_zone
START WITH zone_id IN
(SELECT zid
FROM
( SELECT d.c_z_id AS zid ,d.c_com_id AS comid FROM c_dealers_zone d
UNION
SELECT s.csz_zid AS zid ,s.csz_sid AS comid FROM c_store_zone s
)
WHERE comid IN
(SELECT corp_id
FROM C_COMPANY
START WITH corp_id =4319
CONNECT BY prior corp_id= corp_parentid
)
)
CONNECT BY zone_id=prior zone_parent
结果:去重前:
第四步 将管辖区域向下递归(维护到某个区域表示具有该区域下级管辖)
sql:SELECT *
FROM c_zone
START WITH zone_id IN
(SELECT zid
FROM
( SELECT d.c_z_id AS zid ,d.c_com_id AS comid FROM c_dealers_zone d
UNION
SELECT s.csz_zid AS zid ,s.csz_sid AS comid FROM c_store_zone s
)
WHERE comid IN
(SELECT corp_id
FROM C_COMPANY
START WITH corp_id =4319
CONNECT BY prior corp_id= corp_parentid
)
)
CONNECT BY prior zone_id=zone_parent
结果:(维护到永州市就具有永州市的子集全部管辖)
第五步,将向上递归和向下递归合并结果集作为总部的管辖区域 ,加上where条件查询指定区域下的直接子集,0:中国
sql:SELECT *
FROM
(SELECT * from
(SELECT *
FROM c_zone
START WITH zone_id IN
(SELECT zid
FROM
( SELECT d.c_z_id AS zid ,d.c_com_id AS comid FROM c_dealers_zone d
UNION
SELECT s.csz_zid AS zid ,s.csz_sid AS comid FROM c_store_zone s
)
WHERE comid IN
(SELECT corp_id
FROM C_COMPANY
START WITH corp_id =4441
CONNECT BY prior corp_id= corp_parentid
)
)
CONNECT BY prior zone_id=zone_parent
)
UNION
( SELECT *
FROM c_zone
START WITH zone_id IN
(SELECT zid
FROM
( SELECT d.c_z_id AS zid ,d.c_com_id AS comid FROM c_dealers_zone d
UNION
SELECT s.csz_zid AS zid ,s.csz_sid AS comid FROM c_store_zone s
)
WHERE comid IN
(SELECT corp_id
FROM C_COMPANY
START WITH corp_id =4319
CONNECT BY prior corp_id= corp_parentid
)
)
CONNECT BY zone_id=prior zone_parent
)
)
WHERE zone_parent=0
结果:不加WHERE zone_parent=0:
加WHERE zone_parent=0(只拿出省份):
整个运行也只需要100多毫米挺快的
至此已经完成区域条件构建只拿出自己所具有的区域
总结:做完感觉数据库设计的重要性。一个好的数据库设计能够使整个开发变得简单清晰,一个糟糕的数据设计完全有可能使得某个需求无法满足。
oracle 递归查询(start with …… connect by prior ……)的灵活使用 以及 union (去重) 合并结果集
oracle 递归应用(挺复杂的)的更多相关文章
- Oracle 递归的写法(start with) 以及where条件作用域
先转一个讲Oracle递归讲得非常透彻的文章: http://blog.csdn.net/weiwenhp/article/details/8218091 前言:嗯,这也是一个前人挖坑,后人来填的故事 ...
- oracle 递归和connect by【转】
oracle递归查询(单表包含多级上下级关系) http://www.cnblogs.com/walk-the-Line/p/4882866.html -- 查找所有第一层(包括)下的所有子节点 -- ...
- oracle递归层级查询 start with connect by prior
递归层级查询:start with connect by prior 以部门表作为解析 表结构:dept{id:'主键',name:'部门名称',parent_id:'父亲id'} select * ...
- Oracle 递归
当对象存在父节点.子节点时,通过特定的方式获取父节点.子节点数据构建树状结构或其它形式结构时,通常都会使用递归,如:一个公司有多个部门.每个部门下可能有多个小部门,小部门下面又有组-.为了数据容易 ...
- Oracle递归sql笔记
查询一个机构下所辖机构: select * from t00_organ t start with t.organkey=#uporgankey# connect by prior t.organke ...
- Oracle递归 start with...connect by...prior
prior一侧是父节点 另一侧是子节点 --查询region_id等于4519的节点下面的所有子节点 查找出给定节点的所有子节点 SELECT sr.* FROM spc_region sr wher ...
- Oracle递归操作
需求:找出代理商中没有挂商家的代理商 简单SQL如下: select * from t_proxy tp where tp.id not in (SELECT tp.id as p_id FROM t ...
- [oracle] 递归追溯完整部门名称 函数
create or replace function fn_DeptWholeName2(objectid in number) return nvarchar2 is wholename nvarc ...
- oracle 分页(rownum的理解) 以及 树节点的查询
1:什么是rownum, rownum的生成, rownum相关的符号操作 Rownum是oracle生成结果集时得到的一个伪列, 按照读出行的顺序, 第一条rownum=1, 第二条=2. 对于 O ...
随机推荐
- 【leetcode❤python】 168. Excel Sheet Column Title
class Solution(object): def convertToTitle(self, n): """ :type n: in ...
- 浪潮 NF5240M3 ,NP5540M3 服务器安装2008 R2
1,服务器系统的安装会出现错误的地方一般都是在Raid 卡驱动 略过Raid 卡配置, 具体 http://jingyan.baidu.com/article/ca41422fddfd201eae99 ...
- Python -- 数据加载、存储与文件格式
标签(空格分隔): Python 读入读出通常可以划分为几个大类:读取文本文件和其他更高效的磁盘存储格式,加载数据库中的数据,利用Web API操作网络资源. 读写文本格式的数据 pandas提供了一 ...
- 设置hr标签的粗细
hr {border:0px;border-bottom:1px solid #5c5c3d;}
- ShaderForge
什么是ShaderForge ShaderForge的目标是推动统一的视觉质量提升到了新的高度, 给你自由的材质创建在一个视觉和直观的方式——不需要代码! ShaderForge的特性 •实时着色器预 ...
- 浅谈一下关于使用css3来制作圆环进度条
最近PC端项目要做一个这样的页面出来,其他的都很简单,关键在于百分比的圆环效果.我最初打算是直接使用canvas来实现的,因为canvas实现一个圆是很简便的. 下面贴出canvas实现圆环的代码,有 ...
- JAVA 1.3 (原生数据类型 Primitive Data Type)续
1. 原生数据类型一共有4类8种 >> 整数类型 int表示一个int代表32位 2^32(-2147483648 - 2147483647) >> 字符类型 byte 表示一 ...
- TinyOS 中的 task
task 的目的 做过界面的编程的同学可能会有这种经历,界面不响应,那,其实程序总是在后台运行,但是后台可能是个for循环,那么界面的点击等事件都不能执行. 在windows界面编程中利用了事件机制来 ...
- error: command 'cc' failed with exit status 1
报错: Complete output from command /usr/bin/python -u -c "import setuptools, tokenize;__file__='/ ...
- 调试多线程 & 查死锁的bug & gcore命令 & gdb对多线程的调试 & gcore & pstack & 调试常用命令
gdb thread apply all bt 如果你发现有那么几个栈停在 pthread_wait 或者类似调用上,大致就可以得出结论:就是它们几个儿女情长,耽误了整个进程. 注意gdb的版本要高于 ...