![113ca595d03ea2193ef30ae903f90e86.png](https://i-blog.csdnimg.cn/blog_migrate/e038fcf382379e5ccbe1ac5ebc20f2e5.jpeg)
本文介绍了操作集合的DAX函数的功能,它们对于创建查询很有用,有时还有助于制定度量。在本文中,我们将“集合函数”称为对集合进行操作的函数。DAX中可用的三个设置函数是:UNION,INTERSECT和EXCEPT。它们的功能非常直观:
- UNION执行两个或更多表的联合。
- INTERSECT执行两个表之间的相交设置。
- EXCEPT从第一个参数中删除第二个参数的行。
这些函数将两个或多个表作为参数并返回一个表。它们不仅对编写DAX查询非常有用,而且开发人员在实施措施时也可以使用这些功能来准备复杂的过滤器。
set函数最常使用的功能是保持数据沿袭,这在准备过滤器时至关重要。如果沿袭丢失,则可以使用TREATAS恢复沿袭或强制重新传承。
我们从设置函数的基础开始,然后是关于数据沿袭的见解。
UNION
UNION接受两个或多个表,并返回一个表,其中所有表的所有行均作为参数接收。结果的结构与源表的结构相同,并且保留重复(如果存在)。如果需要删除重复项,则可以在UNION上使用DISTINCT。
为了使用UNION来实现示例,我们使用两个变量表。每个表都包含一个带有“星期几”列的表格,每一行代表一个工作日。从周日开始,我们对工作日进行编号。因此,1代表星期日,2代表星期一,7代表星期六。
用Dax表达式创建新表,操作如下图;
![a65917d0e81f414fdca41d50beb022b9.png](https://i-blog.csdnimg.cn/blog_migrate/e09cd168fce745c7da6d02fd020edef5.jpeg)
![d03804e940e24b7048c0385537a11428.png](https://i-blog.csdnimg.cn/blog_migrate/cc21ee1a2cd84c954f3f4ffd1521e5ba.jpeg)
![c8a0cad8d72d374fd64e0b7aeffa811d.png](https://i-blog.csdnimg.cn/blog_migrate/13983e5e4cb5ea49a3f16dbe71684a9e.jpeg)
用Dax表达式创建MonTue表,用以下Dax表达式如下:
![a79b0fcdc4845c345ad8d8fc4f75b2c0.png](https://i-blog.csdnimg.cn/blog_migrate/d68fd9f11043fc7d83c1ae29dd1c89f5.jpeg)
![57f367fc3f73e9f17164df31ffca64a2.png](https://i-blog.csdnimg.cn/blog_migrate/4263549de83d57e4c2b9480ce58f71b1.jpeg)
这些变量每个都包含两行。在下面的示例中,我们在两个表上使用UNION。结果是一个包含源表的所有行的表,包括重复项:
![2f31b5037517715e9c6a7022ce0af955.png](https://i-blog.csdnimg.cn/blog_migrate/d67a6212127c78dbf8693aee57062918.jpeg)
![f3db25c5ed1f7a0d2ba0ae13224e4ef7.png](https://i-blog.csdnimg.cn/blog_migrate/34b1ccab6c35b726decf4d011dca80ed.jpeg)
以上结果如我们所见,包含两个表中的所有行,星期一的重复行不会被删除。
DISTINCT
DISTINCT被证明对删除重复项很有用,如下:
![20c1d1b1ad7a508db4c3991298dd99d7.png](https://i-blog.csdnimg.cn/blog_migrate/8d1c24dc58dd110df0f3a6c69ec85565.jpeg)
![735dad4337a2573dffbcd9c7e3c926a5.png](https://i-blog.csdnimg.cn/blog_migrate/6b30f25f83d11f68008fe082969d532c.jpeg)
INTERSECT
INTERSECT接受两个表作为参数。它返回第一个参数中也存在于第二个参数中的所有行,并且保留第一个参数中存在的所有重复项。参数的顺序很重要:只有在第一个参数中存在重复项时,才保留重复项。
在以下示例中,我们在前面示例中使用的一周的同一天的临时表上使用INTERSECT。结果仅包含星期一,因为这是两个表之间唯一的共同工作日:
![13bb7d58999724d6c9387145af92c035.png](https://i-blog.csdnimg.cn/blog_migrate/eecbb26f3a6b8caaf9917d6f903857c7.jpeg)
![00fbf332ceaeb5c9c255f18ec0428333.png](https://i-blog.csdnimg.cn/blog_migrate/95bdfb88977d75e66a09bc89ddeae2d9.jpeg)
为了测试重复项,我们添加了SunMonMonWed变量,该变量包含SunMon和MonWed的并集。该表包含星期一两次;因此,它拥有一个副本:
![30b4281738dd8d531598537b81912668.png](https://i-blog.csdnimg.cn/blog_migrate/88dd2afaf7db8e8a6988cdecefb15327.jpeg)
![eee9e2391e290790cdb4a97ad7632567.png](https://i-blog.csdnimg.cn/blog_migrate/b83e57f7e7ad5a0c1fb38daa4be2eb87.jpeg)
在下面的示例中,我们在SunMonMonWed上使用INTERSECT作为第一个参数,在MonTue上使用第二个参数。结果仅包含星期一,但是重复;的确MonTue包含星期一,没有在其他日期SunMonMonWed。
![264f4769e9e4559fe5e599e3312a5fe7.png](https://i-blog.csdnimg.cn/blog_migrate/ba87f53671fcee581967323efda81e80.jpeg)
![68677257de5135c0c01254595589907a.png](https://i-blog.csdnimg.cn/blog_migrate/02c01f40cf98982d734756c6071824f2.jpeg)
更改参数的顺序会更改结果。因为星期一仅在MonTue变量中出现一次(我们现在将其用作第一个参数),所以结果仅包含星期一一次。
![5fc5d6fa303e5bc9d02fcf5d3c3634fb.png](https://i-blog.csdnimg.cn/blog_migrate/781162b59447ee93797af0a7fecca57c.jpeg)
![aa3f951b39570bca59959e49fdde101f.png](https://i-blog.csdnimg.cn/blog_migrate/69608b1190c0aa314be431b8fec9aee3.jpeg)
EXCEPT
设置功能的第三个也是最后一个是EXCEPT。除接受两个表作为参数外,它返回表1中不存在于表2中的所有行。使用EXCEPT时,参数的顺序至关重要。确实,EXCEPT仅在第一个参数中出现时才保留重复项。
作为第一个示例,我们将EXCEPT与SunMon作为第一个参数,将MonTue作为第二个参数。结果是一个只有星期日的表,因为第二个参数中存在星期一,并将其从结果中删除:
![d9ec3abc88aed358520785c3fe03f97d.png](https://i-blog.csdnimg.cn/blog_migrate/a95cb596953de6bf8d1f650af7e67b7c.jpeg)
![6e4918cbc1e1487b90b30aaf55219405.png](https://i-blog.csdnimg.cn/blog_migrate/bf650c60b84b31567c570ac0e5b1e57c.jpeg)
在下一个示例中,我们更改参数的顺序。我们将EXCEPT与MonTue作为第一个参数,将SunMon作为第二个参数。结果仅包含星期二,因为它是第二个参数中不存在的唯一工作日。
![c4ec24294f9ae5abc6a81ec01d19b9ed.png](https://i-blog.csdnimg.cn/blog_migrate/82ca7132014ac28db0f87c5de4004221.jpeg)
![95c1ef4085dc21d2f3c8a3f7e0cf3f01.png](https://i-blog.csdnimg.cn/blog_migrate/72d72555975f3253ee4e5ddcf6d8cba2.jpeg)
具有不同列名和数据沿袭的表
set函数的参数可能具有不同的列名和不同的数据沿袭。这些函数按其位置将表格中的列匹配。当同一位置的列名称不同时,结果将使用第一个表中的名称。
关于数据沿袭,其行为取决于set函数:当自变量的数据沿袭不同时,UNION丢失数据沿袭,而INTERSECT和EXCEPT都保留其第一个自变量的沿袭。
我们将通过一些示例看到UNION在实践中的一些行为。
第一个示例显示UNION与不同的列名一起使用。下面的代码在三个不同的表上使用UNION,其中一列在每个表中具有不同的名称。结果使用第一个表中的列名:
![533474e1d0a386ec6c0bb24ca4948c68.png](https://i-blog.csdnimg.cn/blog_migrate/ff86adbd7b31c60779c19a32888eb356.jpeg)
![3d10e43b0bcfe579e88955dedf50f556.png](https://i-blog.csdnimg.cn/blog_migrate/de33452a5d99a45979441507f807d983.jpeg)
第二个示例显示了在具有相同数据沿袭的两个表上使用UNION时,数据沿袭得以保留。为此,我们在第一个代码示例中添加了度量评估。根据“星期几”列,结果每一行包含一个不同的值:
![17adeb86a2fc43832d5cb074aa86dab8.png](https://i-blog.csdnimg.cn/blog_migrate/4339650275217e056a6771868471d4c4.jpeg)
![d98ec68e4cb6b1a10d1edd8c976ace08.png](https://i-blog.csdnimg.cn/blog_migrate/1d59792f836d3efa2dd33cc12ccf1676.jpeg)
以上结果,由于两个表的数据沿袭都在同一“日期” [星期几]列中,因此UNION保留数据沿袭。星期一出现两次,因为UNION不会删除重复项。
在下一个示例中,当对具有不同数据沿袭的表使用UNION时,数据沿袭将丢失。我们更换MonTue与MyMonTue包含相同的日子,但没有数据沿袭。由于UNION的两个自变量具有不同的数据沿袭,因此结果将丢失数据沿袭。此外,度量的评估为所有行产生相同的数字;
![eaad1336152be62a20e6a66abc2ccc7c.png](https://i-blog.csdnimg.cn/blog_migrate/89af756ac39f38bab011c7409660de6a.jpeg)
![0d32d7a3f413ea3d37f6d393375cf9de.png](https://i-blog.csdnimg.cn/blog_migrate/f62f992044eea240b31189d2b7c9a66e.jpeg)
如果需要,我们可以使用TREATAS恢复数据沿袭。为了证明这一点,我们创建了MyMonTueDataLineage变量,该变量使用TREATAS恢复数据沿袭。现在,由于恢复了数据沿袭,因此结果是一周中每一天的销售额:
![84cd5330e30114e86064f09ad7e52db3.png](https://i-blog.csdnimg.cn/blog_migrate/7326f4e5da78de46adabdc5126598396.jpeg)
![fe0e2d016f3a06f0ec59065aa66ab6cb.png](https://i-blog.csdnimg.cn/blog_migrate/c890ad36d66bea4ac34985df5aa3cdd8.jpeg)
列数不同的表
set函数均不接受具有不同列数的参数。
在以下示例中,我们在SunMon(两列)和MonTue(一列)上使用EXCEPT。结果是一个错误:
![3a7bf7b1cdb9adcde39a8dea173fa38a.png](https://i-blog.csdnimg.cn/blog_migrate/f5405cc78a2570bce41e62a499b54be3.jpeg)
具有不同列类型的表
如果set函数的参数具有相同的列数,但对应的列具有不同的数据类型,会发生什么情况?在这种情况下,UNION的行为不同于INTERSECT和EXCEPT。实际上,UNION将列类型从数字转换为字符串,而INTERSECT和EXCEPT则不进行任何转换,而是返回错误。另一方面,数字类型之间的转换适用于所有设置的函数。
在下面的示例中,我们创建两个表,T1,其中STRING和INTEGER类型的两列,T2,具有相反的配置。然后,我们应用UNION,该表返回包含两列的表,这两列均为STRING类型;
![a7b44bc4815157bc64ea81bbcd03a020.png](https://i-blog.csdnimg.cn/blog_migrate/f0a224230b9e1db40177139cfdf65a1a.jpeg)
![ed35bdc1c69d398e5a09625628a67974.png](https://i-blog.csdnimg.cn/blog_migrate/89b554c7f73c008016f1d7be4d567d66.jpeg)
带有INTERSECT或EXCEPT的相同示例返回错误:
![f49dc419c0b9f97b6e54aacb261b98b9.png](https://i-blog.csdnimg.cn/blog_migrate/3aee5a0282b500dbd82810fa5b74fa6a.jpeg)
最后一个示例显示了使用INTERSECT在两个不同数值类型之间的有效转换。返回的表包含1,这是T1和T2之间的交集:
![34c8900ff0eba0d9bf03a2e9d41f1b32.png](https://i-blog.csdnimg.cn/blog_migrate/ee349730f96b22b5de4a1de59472e47e.jpeg)
![508f0b3e439f4ebfc788009ac5ac1b56.png](https://i-blog.csdnimg.cn/blog_migrate/ab34d02602afc395c0932308f236c242.jpeg)
结论
集合函数的优点在于它们易于使用,并且通常可以完全按我们期望的那样工作。使用设置函数时最相关的主题是数据沿袭。通过遵循本文概述的规则,我们就可以轻松预测数据沿袭是保留还是丢失。万一丢失,可以使用TREATAS进行恢复。
推荐阅读
利用好这3个隐藏技巧,Power BI 开发体验更丝滑
Power BI数据流——新图表视图
如何自动播放 Power BI 移动应用的幻灯片