使用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. 异步FIFO设计

    参考http://www.cnblogs.com/BitArt/archive/2013/04/10/3010073.html http://blog.sina.com.cn/s/blog_6d30f ...

  2. TensorFlow 学习(十五)—— tensorflow.python.platform

    tensorflow.python.platform 下的常用工具类和工具函数:tensorflow/tensorflow/python/platform at master · tensorflow ...

  3. jQuery笔记---选择器(三)

    1.1查找隐藏的tr元素的个数 $(“table tr:hidden”).size() 查找所有可见的tr元素的个数 $(“table tr:not(:hidden)”).size()   一般是不使 ...

  4. shiro简单配置(转)

    注:这里只介绍spring配置模式. 因为官方例子虽然中有更加简洁的ini配置形式,但是使用ini配置无法与spring整合.而且两种配置方法一样,只是格式不一样. 涉及的jar包 Jar包名称 版本 ...

  5. Spark Streaming教程

      废话不说,先来个示例,有个感性认识再介绍. 这个示例来自spark自带的example,基本步骤如下: (1)使用以下命令输入流消息: $ nc -lk 9999 (2)在一个新的终端中运行Net ...

  6. js进阶 12-6 如何获取正在发生事件的名字和是哪个元素在发生事件

    js进阶 12-6 如何获取正在发生事件的名字和是哪个元素在发生事件 一.总结 一句话总结:event.type 描述事件的类型.        event.target 触发该事件的 DOM 元素. ...

  7. 【例题3-4 UVA - 340】Master-Mind Hints

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 这里出现了没有在相同位置的只能唯一配对. 就是说 3322 2234 这种情况. 只有3个weak pair. 即key[1]=a[ ...

  8. embed-it_Integrator memory compile工具使用之一

    embed-it_Integrator memory compile工具使用之一 主要内容 使用Integrator compile memory 使用Integrator 对比筛选适合的memory ...

  9. Android使用BitmapFactory.Options解决加载大图片内存溢出问题

    由于Android对图片使用内存有限制,若是加载几兆的大图片便内存溢出.Bitmap会将图片的所有像素(即长x宽)加载到内存中,如果图片分辨率过大,会直接导致内存溢出(java.lang.OutOfM ...

  10. php 文件夹是否存在,不存在就创建

    $lujing = "./nihao/wohao"; if(!is_dir($liujing)){ mkdir(iconv("UTF-8", "GBK ...