使用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. Javascript和jquery事件--鼠标事件的小结

    1.鼠标事件的主要事件应该是mouseup, mousedown, mousewheel, mousemove, mouseover, moveout. <1>其中mouseup和mous ...

  2. swift项目第九天:正则表达式的学习

    import UIKit /* 练习1:匹配abc 练习2:包含一个a~z,后面必须是0~9 -->[a-z][0-9]或者[a-z]\d * [a-z] : a~z * [0-9]/\d : ...

  3. 你说你会C++? —— 智能指针

    智能指针的设计初衷是:      C++中没有提供自己主动回收内存的机制,每次new对象之后都须要手动delete.稍不注意就memory leak. 智能指针能够解决上面遇到的问题. C++中常见的 ...

  4. 在mac中导入hadoop2.6.0源代码至eclipse 分类: A1_HADOOP 2015-04-12 09:27 342人阅读 评论(0) 收藏

    一.环境准备 1.安装jdk.maven等 2.下载hadoop源代码,并解压 3.将tools.jar复制到Classes中,具体原因见http://wiki.apache.org/hadoop/H ...

  5. 【跟我一起学Unity3D】代码中分割图片而且载入帧序列动画

    在Cocos2dx中.对大图的处理已经封装好了一套自己的API,可是在Unity3D中貌似没有类似的API(好吧,实际上是有的,并且功能更强大),或者说我没找到. 只是这也在情理之中,毕竟Unity3 ...

  6. 利用a标签导出csv文件

    原文 简书原文:https://www.jianshu.com/p/a8687610cda3 大纲 1.需求分析 2.通过a标签实现文件导出 3.实现方式 1.需求分析 导出文件,使用最多的方式还是服 ...

  7. CAP理论/AP架构/CP架构

      原文地址:https://blog.csdn.net/u013058742/article/details/83541905  简书里的文章:Spring Cloud Eureka简介及与Zook ...

  8. [JS Compse] 4. A collection of Either examples compared to imperative code

    For if..else: const showPage() { if(current_user) { return renderPage(current_user); } else { return ...

  9. 1. java.util.concurrent - Java 并发工具包

    1. java.util.concurrent - Java 并发工具包 Java 5 添加了一个新的包到 Java 平台,java.util.concurrent 包.这个包包含有一系列能够让 Ja ...

  10. 【63.73%】【codeforces 560A】Currency System in Geraldion

    time limit per test2 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...