使用DAX中的某些函数特别类似Calculate这种函数创建计算列时很容易出现一种错误,叫做检测到循环依赖关系,即:A circular dependency was detected。对于刚接触Dax语言的人来说,这个错误看着有点摸不到头脑,整个公式使用上似乎没有任何问题,怎么会出现这个错误呢?

要排查这个错误,先了解一下循环依赖关系指的是什么。如下图所示,循环依赖关系简单的可以理解为,A由B得出,而B又由A有关联关系。这就导致出现了一个循环的圈圈,你中有我,我中有你,然后,然后就是你和我都分不清楚了。

在Dax中,循环依赖错误在创建计算列时比较容易出现,有的时候排查起来也可能很困难。由浅入深,这个blog先介绍一下单张表中是如何出现循环依赖错误的。

先看下面这个最简单的产品销售报表,原始列有产品名(Product),单价(Unit Price),成本(Cost),以及销售量(Sales Volume)。:

可以创建一个新的计算列(column)Profit,用来计算每个产品的收益。



此时,我们还可以进一部的算一下每个产品的收益率%Profit,也就是收益占产品单价的比重。



目前一切都看起来平淡无奇。如果我们脑洞开一下,既然有了收益率,那么收益率*产品单价是不是可以反过来推出产品收益呢?我们把原来的产品收益率计算公式从Unit Price - Cost改成%Profit*Unit Price,看一下Power BI的计算结果:



此时果不其然,有循环依赖关系的错误返回。这个理解起来很简单,因为%Profit是由Profit推出来的,而现在我们又打算让DAX用%Profit去推Profit,这显然行不通。

这个例子看起来非常low,因为实际操作中是不会有人回头去改Profit计算公式而造成循环依赖关系的。但是这个例子能说明一个很重要的问题,因为DAX中有一种很重要的概念叫上下文,我们在使用某些对上下文非常敏感的函数时,很有可能就无意中掉进了这个循环依赖错误圈套里面,即虽然表面上看不到明显的A引用B,B又引用A的关系,但是通过上下文这一作用,隐式的出现了A和B之间相互引用的错误。这一情况最容易出现在Calculate函数身上。

还是用这个Product Table做例子,我们用CALCULATE+ SUMX函数对产品A来计算Total Sales。



计算结果没有什么问题,看起来很完美。同理的,我们套用这一公式对产品A来计算Total Cost。然而,当我们敲回车准备拿计算结果时,悲剧发生了,一个循环依赖错误蹦了出来。并且提示我们这个错误跟之前刚刚创建的Total Sales有关系。



这个提示错误很崩溃,因为跟最开始Profit那个例子不一样,我们Total Sales和Total Cost里面用的列都是原数据,根本就不存在推算关系。怎么用原数据计算的Total Cost就和Total Sales产生循环依赖关系了呢?原因吗,就是出现在Calculate函数身上。

Calculate函数是一个重度上下文依赖函数,也就是说,当一个计算列中出现了Calculate函数后,你要特别小心,因为当前行中所有的数据都可能跟Calculate函数的结果产生关系。

在当前例子中,当只有Total Sales这个计算列时,与其有关联关系的是:

元素 依赖值
SUMX Unit Price,Sales Volume
行: Product =A Product=A,Unit Price=75,Cost=49,Sales Volumn=10000

而当创建了Total Cost这个计算列时,整个Product是A的这一行上下文发生了变化,又增加了一列。这样对于用重度上下文依赖关系的Calculate函数计算得出的Total Sales来讲,与其有关联关系的多了一个Total Cost。即

元素 依赖值
SUMX Unit Price,Sales Volume
行: Product =A Product=A,Unit Price=75,Cost=49,Sales Volumn=10000,Total Cost=?

而同样的从Total Cost角度来看,从Product是A的这一行上下文关系中包含一个Total Sales=750,000值。这样就隐式的出现了一个循环依赖关系,即Total Sales依赖Total Cost,而Total Cost又依赖Total Sales。

元素 依赖值
SUMX Unit Price,Sales Volume
行: Product =A Product=A,Unit Price=75,Cost=49,Sales Volumn=10000,Total Sales=750,000

如何解决该问题呢?最简单直接的其实是尽量不要在计算列中使用Calculate函数。因为这个函数陷阱太多,容易把自己绕进去,Calculate与度量值其实更配哦。当然,如果要用也可以,尽量只用一个,多个的时候非常容易出现循环依赖错误。如果非要用多个呢?好吧,可以尝试CALCULATE+EXCEPTALL的组合来清理这种上下文关系。



当然,这只适合于比较简单的数据结构,如果是有多张表相互关联,要排查循环依赖错误就更加的麻烦。

DAX的圈圈大坑:循环依赖关系错误circular dependency (单表篇)的更多相关文章

  1. 在.NET Core中遭遇循环依赖问题"A circular dependency was detected"

    今天在将一个项目迁移至ASP.NET Core的过程中遭遇一个循环依赖问题,错误信息如下: A circular dependency was detected for the service of ...

  2. 巧用 Lazy 解决.NET Core中的循环依赖关系

    原文作者: Thomas Levesque 原文链接:https://thomaslevesque.com/2020/03/18/lazily-resolving-services-to-fix-ci ...

  3. tomcat - 解决 org.bouncycastle.asn1.ASN1Boolean 非法循环依赖的错误

    背景 记录遇到一次奇怪的错误,在发布war包到Tomcat的时候,出现了org.bouncycastle.asn1.ASN1Boolean非法循环依赖的错误. INFO: Deploying web ...

  4. Ubuntu安装deb软件包错误(依赖关系问题)解决

    执行命令 sudo dpkg -i XXX.deb 返回依赖关系错误提示 执行 sudo apt-get -f install 这条命令将自动安装需要的依赖包. 再次执行命令 sudo dpkg -i ...

  5. 面试必杀技,讲一讲Spring中的循环依赖

    本系列文章: 听说你还没学Spring就被源码编译劝退了?30+张图带你玩转Spring编译 读源码,我们可以从第一行读起 你知道Spring是怎么解析配置类的吗? 配置类为什么要添加@Configu ...

  6. 面试阿里,腾讯,字节跳动90%都会被问到的Spring中的循环依赖

    前言 Spring中的循环依赖一直是Spring中一个很重要的话题,一方面是因为源码中为了解决循环依赖做了很多处理,另外一方面是因为面试的时候,如果问到Spring中比较高阶的问题,那么循环依赖必定逃 ...

  7. Spring的循环依赖

    本文简要介绍了循环依赖以及Spring解决循环依赖的过程 一.定义 循环依赖是指对象之间的循环依赖,即2个或以上的对象互相持有对方,最终形成闭环.这里的对象特指单例对象. 二.表现形式 对象之间的循环 ...

  8. Spring 的循环依赖问题

    什么是循环依赖 什么是循环依赖呢?可以把它拆分成循环和依赖两个部分来看,循环是指计算机领域中的循环,执行流程形成闭合回路:依赖就是完成这个动作的前提准备条件,和我们平常说的依赖大体上含义一致.放到 S ...

  9. Springboot循环依赖实践纪实

    测试的Springboot版本: 2.6.4,禁止了循环依赖,但是可以通过application.yml开启(哈哈) @Lazy注解解决循环依赖 情况一:只有简单属性关系的循环依赖 涉及的Bean: ...

随机推荐

  1. Dynamics CRM 2016 Web API 消息列表

    Function Name Description CalculateTotalTimeIncident Function Calculates the total time, in minutes, ...

  2. [Angular] Update FormArray with patchValue

    Currently, patchValue doesn't support update FormArray. The workarround is you need to empty the for ...

  3. python 深浅拷贝 进阶

    主要理解新与旧究竟在哪里 这样也就理解了 深浅拷贝 先说说赋值,事实上python中的赋值事实上是赋值了一个引用.比如: foo1=1.0 foo2=foo1 用操作符is推断时.你能够发现结果是tr ...

  4. Want to archive tables? Use Percona Toolkit’s pt-archiver--转载

    原文地址:https://www.percona.com/blog/2013/08/12/want-to-archive-tables-use-pt-archiver/ Percona Toolkit ...

  5. 画pcb时丝印不能再焊盘上

    上图中U3就在焊盘上,这样印出来U3显示不全

  6. 【AtCoder Beginner Contest 074 A】Bichrome Cells

    [链接]h在这里写链接 [题意] 在这里写题意 [题解] 在这里写题解 [错的次数] 0 [反思] 在这了写反思 [代码] #include <bits/stdc++.h> using n ...

  7. Ansible 管理服务和软件

    [root@Ansible ~]# ansible RAC -m yum -a 'name=iscsi-initiator-utils state=installed' RAC_Node1 | suc ...

  8. java项目中VO和DTO以及Entity,各自是在什么情况下应用的

    j2ee中,经常提到几种对象(object),理解他们的含义有助于我们更好的理解面向对象的设计思维.     POJO(plain old java object):普通的java对象,有别于特殊的j ...

  9. iOS开发RunnLoop学习二:GCD中的定时器

    #import "ViewController.h" @interface ViewController () /** 注释 */ @property (nonatomic, st ...

  10. php实现 删除字符串中出现次数最少的字符

    php实现  删除字符串中出现次数最少的字符 一.总结 一句话总结:数组排序是改变数组的,而其它函数一般不改变原数据,比如str_replace(); 1.单案例测试通过而多案例测试不通过怎么办? 检 ...