”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > Google Sheets:如何花费数小时构建 SUMIFS

Google Sheets:如何花费数小时构建 SUMIFS

发布于2024-11-08
浏览:903

Google Sheets: How to Build a SUMIFS for Hours

大家好!今天我想分享一个我创建的超级有用的脚本,用于解决日常生活中的常见问题。

如果您曾经尝试在 Google 表格中对“持续时间”求和,您可能已经注意到,SUMIF 和 SUMIFS 公式无法根据特定条件对事件或产品的持续时间求和。根据您需要执行的计算类型,这可能会成为一个障碍。但别担心! Google 表格允许您在 JavaScript 中创建脚本并将其用作自定义公式。

在我的脚本中,我创建了两种变体:第一个接受一个条件,第二个接受最多两个条件。我计划将来改进这一点,使功能更加灵活。

值得一提的是,自定义公式不会返回程序可直接计算的值。要解决此问题,您可以使用 =VALUE() 函数包装结果。然后,只需将相应的格式应用于数据类型 - 在我们的例子中为“持续时间”。准备好查看脚本了吗?

创建用于测试的通用数据

首先,让我们生成数据来测试公式。我为此使用了我们的朋友 GPT。

标题 期间 类别 地位
火星人 01:00:00 电影 已观看
星际穿越 02:49:00 电影 已观看
约翰威克 01:30:00 电影 已观看
复仇者联盟:终局之战 03:00:00 电影 想看
怪奇物语 00:45:00 系列 观看
巫师 01:00:01 系列 观看
曼达洛人 00:40:00 系列 观看
绝命毒师 00:50:00 系列 已观看
金钱抢劫 00:55:00 系列 想看
权力的游戏 01:10:00 系列 想看

脚本

我尝试尽可能最好地记录一切。我决定将其分解为更小的函数,并使用更具声明性的内容来提高代码清晰度。

function allAreArrays(...arrays) {
  return arrays.every(Array.isArray);
}

function allArraysHaveSameLength(...arrays) {
  const lengths = arrays.map((arr) => arr.length);
  return lengths.every((val) => val === lengths[0]);
}

function convertHMSToSeconds(hms) {
  // Breaks the string in HH:MM:SS format into parts
  const parts = String(hms).split(":");

  // Converts the parts into integers
  const [hours, minutes, seconds] = parts;

  // Converts hours and minutes into seconds and adds the seconds
  const totalSeconds =
    Number(hours) * 3600   Number(minutes) * 60   Number(seconds);

  return Number(totalSeconds);
}

function convertSecondsToHMS(seconds) {
  // Calculates the number of hours, minutes, and seconds
  const hours = Math.floor(seconds / 3600);
  const minutes = Math.floor((seconds % 3600) / 60);
  const remainingSeconds = seconds % 60;

  // Adds a leading zero to ensure it always has two digits
  const hourFormat = String(hours).padStart(2, "0");
  const minuteFormat = String(minutes).padStart(2, "0");
  const secondFormat = String(remainingSeconds).padStart(2, "0");

  // Returns the HH:MM:SS format
  return `${hourFormat}:${minuteFormat}:${secondFormat}`;
}

/**
 * Sums hours based on a criterion.
 *
 * @param {string[]} sum_range - Set of time intervals in HH:MM:SS format.
 * @param {number[]} criteria_range - Set of criteria corresponding to the time intervals.
 * @param {number} criterion - The criterion for which hours should be summed.
 * @returns {string} Sum of the passed durations, or an error message.
 */
function sumHoursIf(sum_range, criteria_range, criterion) {
  if (!allAreArrays(sum_range, criteria_range))
    return "Pass the intervals for the calculation!";

  if (!allArraysHaveSameLength(sum_range, criteria_range))
    return "Intervals must be the same size";

  // Filters the time intervals for the specific criterion
  const hoursToSum = sum_range.filter(
    (row, index) =>
      String(criteria_range[index]).trim() == String(criterion).trim()
  );

  // Converts the filtered hours to seconds
  const hoursInSeconds = hoursToSum.map((n) => convertHMSToSeconds(n));

  // Sums all the seconds
  const sumOfSeconds = hoursInSeconds.reduce((accumulator, currentValue) => {
    return accumulator   currentValue;
  }, 0);

  // Converts the total seconds back to HH:MM:SS format
  return convertSecondsToHMS(sumOfSeconds);
}

/**
 * Sums hours based on criteria.
 *
 * @param {string[]} sum_range - Set of time intervals in HH:MM:SS format.
 * @param {number[]} criteria_range1 - First set of criteria corresponding to the time intervals.
 * @param {number} criterion1 - The first criterion for which hours should be summed.
 * @param {string[]} criteria_range2 - Second set of criteria corresponding to the time intervals.
 * @param {string} criterion2 - The second criterion for which hours should be summed.
 * @returns {string} Sum of the passed durations, or an error message.
 */
function sumHoursIf2(
  sum_range,
  criteria_range1,
  criterion1,
  criteria_range2,
  criterion2
) {
  if (
    !allAreArrays(
      sum_range,
      criteria_range1,
      criteria_range2
    )
  )
    return "Pass the intervals for the calculation!";

  if (
    !allArraysHaveSameLength(
      sum_range,
      criteria_range1,
      criteria_range2
    )
  )
    return "Intervals must be the same size";

  // Filters the time intervals for the passed criteria
  const hoursToSum = sum_range.filter(
    (row, index) =>
      String(criteria_range1[index]) == String(criterion1).trim() &&
      String(criteria_range2[index]).trim() === String(criterion2).trim()
  );

  // Converts the filtered hours to seconds
  const hoursInSeconds = hoursToSum.map((n) => convertHMSToSeconds(n));

  // Sums all the seconds
  const sumOfSeconds = hoursInSeconds.reduce((accumulator, currentValue) => {
    return accumulator   currentValue;
  }, 0);

  // Converts the total seconds back to HH:MM:SS format
  return convertSecondsToHMS(sumOfSeconds);
}

如何使用?

条件可以是文本或数字,但小时数必须采用“纯文本”格式。转到应用程序脚本并粘贴脚本,然后按“CTRL S”。完毕。使用它的过程与本机公式相同。

应用公式后,我们可以使用 VALUE 再次将其视为程序可以理解的类型;你的代码应该是这样的:

=VALUE(sumHoursIf2($C$2:$C$11;$D$2:$D$11;C$14;$E$2:$E$11;$B15))

如果一切顺利,这应该是您的结果:

类别 电影 系列
协助 5:19:00 0:50:00
协助 0:00:00 2:25:01
Quero 助手 3:00:00 2:05:00

只是一个提示,希望您喜欢它,如果您有建议,请在评论中留下。干杯。

原始帖子:https://dev.to/casewinter/como-somar-horas-no-google-sheets-usando-criterios-para-filtrar-linhas-364p

版本声明 本文转载于:https://dev.to/casewinter/google-sheets-how-to-build-a-sumifs-for-hours-16ee?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 使用 React 构建 Sunnyside Agency 网站
    使用 React 构建 Sunnyside Agency 网站
    Introduction Welcome to the detailed breakdown of the Sunnyside Agency website, a modern and stylish site built using React. This project sho...
    编程 发布于2024-11-08
  • 在 Next.js 项目中通过裁剪和压缩优化图像上传
    在 Next.js 项目中通过裁剪和压缩优化图像上传
    作为前端开发人员,您很有可能曾经或正在从事涉及发布和显示图像的项目。如果你还没有,那么你很快就会的。因此,最近,在我们完成项目后,我们发现在显示用户提供的图像时遇到了麻烦。 这一切是如何解开的 最大的问题是如何处理尺寸,尤其是图像的高度与宽度。将图像设置为 object-fit: cover 似乎是...
    编程 发布于2024-11-08
  • 为什么 JavaScript 中的“this”运算符不一致以及如何解决?
    为什么 JavaScript 中的“this”运算符不一致以及如何解决?
    在 Javascript 中,为什么“this”运算符不一致?在 JavaScript 中,“this”运算符表现出不同的行为,具体取决于调用上下文。这可能会导致混乱和意外结果,特别是在使用回调和对象时。调用模式和“this”绑定“this”运算符绑定到函数调用时的对象或类,this 绑定由调用模式...
    编程 发布于2024-11-08
  • 如何生成等概率地求和到预定义值的随机数?
    如何生成等概率地求和到预定义值的随机数?
    生成随机数求和到预定义值在这种情况下,我们的目标是生成一个伪随机数列表,这些伪随机数共同添加直至特定的预定值。一种方法是随机生成指定范围内的数字,将其从总数中减去,然后重复此过程,直到总和等于所需值。然而,这种方法在对总和的贡献方面有利于第一个生成的数字。为了确保一致性,开发了更复杂的解决方案:im...
    编程 发布于2024-11-08
  • 如何在 Matplotlib 中正确对齐旋转的 XTickLabels?
    如何在 Matplotlib 中正确对齐旋转的 XTickLabels?
    对齐旋转的 XTickLabels 以实现精确对齐在给定的图中,旋转的 x 轴刻度标签看起来向右移动而不是对齐与他们各自的刻度线。出现这种不对齐的原因是默认围绕文本标签中间的旋转居中。要解决此问题,您可以使用 ha 参数指定刻度标签的水平对齐方式。此参数定义旋转标签周围的假想矩形框的哪一侧应与刻度点...
    编程 发布于2024-11-08
  • Java 内存模型:深入理解并发性
    Java 内存模型:深入理解并发性
    介绍 Java 内存模型 (JMM) 是 Java 并发编程的一个基本但经常被误解的方面。 JMM 是随 Java 5 引入的,它定义了线程如何与内存交互,确保多线程程序的一致性和可预测性。在本文中,我们将深入探讨 JMM,探讨其关键概念,并研究它如何影响并发 Java 应用程序开...
    编程 发布于2024-11-08
  • 如何在 JavaScript 中访问 iFrame 元素并与之交互?
    如何在 JavaScript 中访问 iFrame 元素并与之交互?
    在 JavaScript 中访问 iFrame 元素浏览错综复杂的 JavaScript 通常会带来挑战,尤其是在处理 iFrame 时。当尝试从 iFrame 的子页面检索驻留在 iFrame 中的文本区域的值时,会出现常见的困境。传统方法在这方面存在不足。要深入研究解决方案,承认浏览器施加的安全...
    编程 发布于2024-11-08
  • 如何通过修改 PYTHONPATH 或 ~/.pylintrc 解决 PyLint 中的“无法导入”错误?
    如何通过修改 PYTHONPATH 或 ~/.pylintrc 解决 PyLint 中的“无法导入”错误?
    如何通过设置 PYTHONPATH 解决 PyLint 中的“无法导入”错误使用时遇到“无法导入 X”错误时PyLint,特别是从子目录导入的模块,根本原因可能在于Python路径配置。解决方案1:修改PYTHONPATH环境变量一有效的解决方案是调整 PYTHONPATH 环境变量以合并包含导入的...
    编程 发布于2024-11-08
  • 网站所有者如何防止第三方 Iframe 嵌入?
    网站所有者如何防止第三方 Iframe 嵌入?
    防止第三方 iframe 嵌入网站所有者经常面临其页面嵌入第三方框架 (iframe) 的问题网站。虽然引用请求标头在页面加载期间检测这种情况时被证明无效,但有多种方法可以解决此问题。JavaScript 检测页面加载后,JavaScript可以用来检测它是否正在框架内显示。通过比较 top 和 s...
    编程 发布于2024-11-08
  • 激动人心的体育周:亮点和亮点
    激动人心的体育周:亮点和亮点
    过去的一周对于全球最新体育评论爱好者来说是一段令人兴奋的旅程。从破纪录的表演到扣人心弦的结局,这里汇总了体育界最激动人心的赛事。 足球:英超联赛令人震惊 英超联赛震撼人心 英超联赛总是充满戏剧性,本周也不例外。曼城队在一场万众瞩目的比赛中迎战利物浦队,最终令人惊讶地3-3战平。两支球...
    编程 发布于2024-11-08
  • 如何使用 NumPy 的“np.newaxis”在数组操作中启用广播?
    如何使用 NumPy 的“np.newaxis”在数组操作中启用广播?
    NumPy 的 'np.newaxis' 是什么以及如何使用它理解 'np.newaxis'NumPy 的“np.newaxis”,也称为“None”,是一个伪索引,用于临时向数组添加轴。使用一次时,它将数组的维度增加一。例如,1D 数组变成 2D 数组,2D 数组变...
    编程 发布于2024-11-08
  • MySQL 如何处理较短列中的长整数:溢出或截断?
    MySQL 如何处理较短列中的长整数:溢出或截断?
    较短列中的长整数转换:机制和公式将长整数插入较短整数列时,MySQL 通常会截断该值以适合指定的长度。但是,在某些情况下,行为可能会有所不同,从而导致意外的转换。考虑一个 10 位长整数列 some_number。如果将超过最大整数范围 (2147483647) 的值插入到此列中,MySQL 会将该...
    编程 发布于2024-11-08
  • 如何在教义 2 中创建带有额外字段的多对多链接表?
    如何在教义 2 中创建带有额外字段的多对多链接表?
    Doctrine 2 和带有额外字段的多对多链接表本文解决了在 Dotrine 2 中创建多对多关系的问题,其中链接表包含一个附加值,特别是在库存系统的上下文中。原则 2 中的多对多关系可以使用不包含任何附加字段的链接表来建立。但是,当每个链接都需要额外的值时,必须将链接表重新定义为新实体。提供的代...
    编程 发布于2024-11-08
  • JavaScript 中的单管道运算符如何处理浮点数和整数?
    JavaScript 中的单管道运算符如何处理浮点数和整数?
    探索 JavaScript 中单管道运算符的按位性质在 JavaScript 中,单管道运算符(“|”)执行按位运算称为按位或的运算。理解此操作对于理解其对不同输入值的影响至关重要,如以下示例所示:console.log(0.5 | 0); // 0 console.log(-1 | 0); //...
    编程 发布于2024-11-08
  • 列表理解和Regae
    列表理解和Regae
    啊。我一直害怕的那一刻。 第一篇文章,包含我自己的想法、观点和可能的知识细分。 请注意,亲爱的读者,这并不是对 Python 单行 for 循环、追加到列表和返回一些数据的能力的深入探讨或令人难以置信的分解。不,不。这只是展示了如何有趣——以及如何愚蠢——小东西可以组合在一起,让一...
    编程 发布于2024-11-08

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3