使用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. js进阶 13-5 jquery队列动画如何实现

    js进阶 13-5 jquery队列动画如何实现 一.总结 一句话总结:同一个jquery对象,直接写多个animate()就好. 1.什么是队列动画? 比如说先左再下,而不是左下一起走 2.怎么实现 ...

  2. C#委托与事件(生动故事)

    [委托] 1,工人Peter按工作步骤向老板报告的程序. 程序: using System; using System.Collections.Generic; using System.Linq; ...

  3. 软件——机器学习与Python,if __name__ == '__main__':函数

    if __name__ == '__main__': 想必很多初次接触python都会见到这样一个语句,if __name__ == "__main__": 那么这个语句到底是做什 ...

  4. POJ 3264 Balanced Lineup 线段树RMQ

    http://poj.org/problem?id=3264 题目大意: 给定N个数,还有Q个询问,求每个询问中给定的区间[a,b]中最大值和最小值之差. 思路: 依旧是线段树水题~ #include ...

  5. [React] Style the body element with styled-components and "injectGlobal"

    In this lesson, we see how we can apply styles globally with the "injectGlobal" helper met ...

  6. <九度 OJ>题目1012:畅通project

    题目描写叙述: 某省调查城镇交通状况,得到现有城镇道路统计表.表中列出了每条道路直接连通的城镇.省政府"畅通project"的目标是使全省不论什么两个城镇间都能够实现交通(但不一定 ...

  7. matlab 音频处理

    1. 读取与播放 load gong.mat; % y 42028x1 double soundsc(y); % 可调节其频率 soundsc(y, 2*Fs); 读取 .wav 等音频:audior ...

  8. python3中sum

    摘自https://blog.csdn.net/ikerpeng/article/details/17026011 其实python中sum有两种 一种是python自己的sum 另一种是python ...

  9. HDU 1405 The Last Practice 数学水题

    http://acm.hdu.edu.cn/showproblem.php?pid=1405 题目大意: 给你一个数,让你分解素因子,输出它的各次幂. 如60 输出:2 2 3 1 5 1 (60=2 ...

  10. swift学习第十二天:类的属性定义

    类的属性介绍 Swift中类的属性有多种 存储属性:存储实例的常量和变量 计算属性:通过某种方式计算出来的属性 类属性:与整个类自身相关的属性 存储属性 存储属性是最简单的属性,它作为类实例的一部分, ...