如何快速求和文件中的所有数字?

2024-10-11 08:36:00
admin
原创
69
摘要:问题描述:我有一个包含几千个数字的文件,每个数字占一行:34 42 11 6 2 99 ... 我想编写一个脚本来打印文件中所有数字的总和。我有一个解决方案,但效率不高。(运行需要几分钟。)我正在寻找一个更有效的解决方案。有什么建议吗?解决方案 1:您可以使用 awk:awk '{ sum ...

问题描述:

我有一个包含几千个数字的文件,每个数字占一行:

34
42
11
6
2
99
...

我想编写一个脚本来打印文件中所有数字的总和。我有一个解决方案,但效率不高。(运行需要几分钟。)我正在寻找一个更有效的解决方案。有什么建议吗?


解决方案 1:

您可以使用 awk:

awk '{ sum += $1 } END { print sum }' file

解决方案 2:

到目前为止,还没有解决方案使用paste。这是一个:

paste -sd+ filename | bc

如果文件末尾有换行符,则末尾+将导致语法错误。通过删除末尾的 来修复错误+

paste -sd+ fiilename | sed 's/+$//g' | bc

例如,计算 Σn,其中 1<=n<=100000:

$ seq 100000 | paste -sd+ | bc -l
5000050000

(对于好奇的人来说,给定一个正数,将打印从到seq n的数字序列。)1`n`n

解决方案 3:

对于 Perl 单行代码来说,它基本上与Ayman Hourieh 的答案awk中的解决方案相同:

 % perl -nle &#039;$sum += $_ } END { print $sum&#039;

如果你对 Perl 单行命令的作用感到好奇,你可以解析它们:

 %  perl -MO=Deparse -nle &#039;$sum += $_ } END { print $sum&#039;

结果是该程序的版本更加冗长,其形式是没有人会自己编写的:

BEGIN { $/ = &quot;
&quot;; $ = &quot;
&quot;; }
LINE: while (defined($_ = &lt;ARGV>)) {
    chomp $_;
    $sum += $_;
}
sub END {
    print $sum;
}
-e syntax OK

只是为了好玩,我尝试用一​​个包含 1,000,000 个数字(范围在 0 - 9,999 之间)的文件执行此操作。在我的 Mac Pro 上,它几乎立即返回。这太糟糕了,因为我希望使用mmap会非常快,但时间却一样:

use 5.010;
use File::Map qw(map_file);

map_file my $map, $ARGV[0];

$sum += $1 while $map =~ m/(d+)/g;

say $sum;

解决方案 4:

只是为了好玩,我们来对它进行基准测试:

$ for ((i=0; i&lt;1000000; i++)) ; do echo $RANDOM; done > random_numbers

$ time perl -nle &#039;$sum += $_ } END { print $sum&#039; random_numbers
16379866392

real    0m0.226s
user    0m0.219s
sys     0m0.002s

$ time awk &#039;{ sum += $1 } END { print sum }&#039; random_numbers
16379866392

real    0m0.311s
user    0m0.304s
sys     0m0.005s

$ time { { tr &quot;
&quot; + &lt; random_numbers ; echo 0; } | bc; }
16379866392

real    0m0.445s
user    0m0.438s
sys     0m0.024s

$ time { s=0;while read l; do s=$((s+$l));done&lt;random_numbers;echo $s; }
16379866392

real    0m9.309s
user    0m8.404s
sys     0m0.887s

$ time { s=0;while read l; do ((s+=l));done&lt;random_numbers;echo $s; }
16379866392

real    0m7.191s
user    0m6.402s
sys     0m0.776s

$ time { sed &#039;:a;N;s/
/+/;ta&#039; random_numbers|bc; }
^C

real    4m53.413s
user    4m52.584s
sys 0m0.052s

5 分钟后我中止了 sed 运行


我一直在潜水鲁阿,而且速度很快:

$ time lua -e &#039;sum=0; for line in io.lines() do sum=sum+line end; print(sum)&#039; &lt; random_numbers
16388542582.0

real    0m0.362s
user    0m0.313s
sys     0m0.063s

当我更新这个时,ruby:

$ time ruby -e &#039;sum = 0; File.foreach(ARGV.shift) {|line| sum+=line.to_i}; puts sum&#039; random_numbers
16388542582

real    0m0.378s
user    0m0.297s
sys     0m0.078s

听从 Ed Morton 的建议:使用$1

$ time awk &#039;{ sum += $1 } END { print sum }&#039; random_numbers
16388542582

real    0m0.421s
user    0m0.359s
sys     0m0.063s

与使用$0

$ time awk &#039;{ sum += $0 } END { print sum }&#039; random_numbers
16388542582

real    0m0.302s
user    0m0.234s
sys     0m0.063s

解决方案 5:

另一个选择是使用jq

$ seq 10|jq -s add
55

-s( --slurp) 将输入行读入数组。

解决方案 6:

这是直接的 Bash:

sum=0
while read -r line
do
    (( sum += line ))
done &lt; file
echo $sum

解决方案 7:

我更喜欢使用GNU datamash来完成此类任务,因为它比 perl 或 awk 更简洁易读。例如

datamash sum 1 &lt; myfile

其中 1 表示第一列数据。

解决方案 8:

拉库

say sum lines
~$ raku -e &#039;.say for 0..1000000&#039; > test.in

~$ raku -e &#039;say sum lines&#039; &lt; test.in
500000500000

它的工作原理是lines生成一个字符串序列,这些字符串是输入行。

sum获取该序列,将每行转换为数字并将它们相加。

剩下的就是say打印出该值,然后跟上换行符。(它本来可以是printput,但say更具头韵性。)

解决方案 9:

我更喜欢使用 R 来实现这一点:

$ R -e &#039;sum(scan(&quot;filename&quot;))&#039;

解决方案 10:

这是另一句台词

( echo 0 ; sed &#039;s/$/ +/&#039; foo ; echo p ) | dc

假设数字是整数。如果需要小数,请尝试

( echo 0 2k ; sed &#039;s/$/ +/&#039; foo ; echo p ) | dc

将 2 调整为所需的小数位数。

解决方案 11:

$ perl -MList::Util=sum -le &#039;print sum &lt;>&#039; nums.txt

解决方案 12:

更简洁:

# Ruby
ruby -e &#039;puts open(&quot;random_numbers&quot;).map(&amp;:to_i).reduce(:+)&#039;

# Python
python -c &#039;print(sum(int(l) for l in open(&quot;random_numbers&quot;)))&#039;

解决方案 13:

我不能就这样过去……这是我的 Haskell 单行代码。它实际上非常易读:

sum &lt;$> (read &lt;$>) &lt;$> lines &lt;$> getContents

不幸的是,没有ghci -e办法直接运行它,所以它需要主要功能、打印和编译。

main = (sum &lt;$> (read &lt;$>) &lt;$> lines &lt;$> getContents) >>= print

为了澄清起见,我们读取整个输入(getContents),将其分成lines数字readsum&lt;$>fmap运算符——我们使用它而不是通常的函数应用程序,因为确保这一切都发生在 IO 中。read需要额外的fmap,因为它也在列表中。

$ ghc sum.hs
[1 of 1] Compiling Main             ( sum.hs, sum.o )
Linking sum ...
$ ./sum 
1
2
4
^D
7

这是一个奇怪的升级,使其能够与浮点数一起工作:

main = ((0.0 + ) &lt;$> sum &lt;$> (read &lt;$>) &lt;$> lines &lt;$> getContents) >>= print
$ ./sum 
1.3
2.1
4.2
^D
7.6000000000000005

解决方案 14:

C 总是以速度取胜:

#include &lt;stdio.h>
#include &lt;stdlib.h>

int main(int argc, char **argv) {
    ssize_t read;
    char *line = NULL;
    size_t len = 0;
    double sum = 0.0;

    while (read = getline(&amp;line, &amp;len, stdin) != -1) {
        sum += atof(line);
    }

    printf(&quot;%f
&quot;, sum);
    return 0;
}

100 万个数字的计时(与我的 Python 答案相同的机器/输入):

$ gcc sum.c -o sum &amp;&amp; time ./sum &lt; numbers 
5003371677.000000
real    0m0.188s
user    0m0.180s
sys     0m0.000s

解决方案 15:

cat nums | perl -ne &#039;$sum += $_ } { print $sum&#039;

(与 brian d foy 的答案相同,但没有“END”)

解决方案 16:

只是为了好玩,让我们用Perl 的数组数学引擎PDL来做吧!

perl -MPDL -E &#039;say rcols(shift)->sum&#039; datafile

rcols将列读入矩阵(在本例中为 1D)并sum(惊讶地)对矩阵的所有元素求和。

解决方案 17:

这是一个使用 Python 和生成器表达式的解决方案。在我破旧的笔记本电脑上用一百万个数字进行了测试。

time python -c &quot;import sys; print sum((float(l) for l in sys.stdin))&quot; &lt; file

real    0m0.619s
user    0m0.512s
sys     0m0.028s

解决方案 18:

C++“一行”代码:

#include &lt;iostream>
#include &lt;iterator>
#include &lt;numeric>
using namespace std;

int main() {
    cout &lt;&lt; accumulate(istream_iterator&lt;int>(cin), istream_iterator&lt;int>(), 0) &lt;&lt; endl;
}

解决方案 19:

sed &#039;:a;N;s/
/+/;ta&#039; file|bc

解决方案 20:

运行 R 脚本

我编写了一个 R 脚本,以文件名作为参数并计算各行总数。

#! /usr/local/bin/R
file=commandArgs(trailingOnly=TRUE)[1]
sum(as.numeric(readLines(file)))

可以使用“data.table”或“vroom”包来加速,如下所示:

#! /usr/local/bin/R
file=commandArgs(trailingOnly=TRUE)[1]
sum(data.table::fread(file))
#! /usr/local/bin/R
file=commandArgs(trailingOnly=TRUE)[1]
sum(vroom::vroom(file))

基准测试

与@glenn jackman相同的基准数据。

for ((i=0; i&lt;1000000; i++)) ; do echo $RANDOM; done > random_numbers

与上面的 R 调用相比,运行 R 3.5.0 作为脚本与其他方法相当(在同一个 Linux Debian 服务器上)。

$ time R -e &#039;sum(scan(&quot;random_numbers&quot;))&#039;  
 0.37s user
 0.04s system
 86% cpu
 0.478 total

带有 readLines 的 R 脚本

$ time Rscript sum.R random_numbers
  0.53s user
  0.04s system
  84% cpu
  0.679 total

带有 data.table 的 R 脚本

$ time Rscript sum.R random_numbers     
 0.30s user
 0.05s system
 77% cpu
 0.453 total

带有 vroom 的 R 脚本

$ time Rscript sum.R random_numbers     
  0.54s user 
  0.11s system
  93% cpu
  0.696 total

与其他语言的比较

供参考,在同一硬件上建议一些其他方法

Python 2(2.7.13)

$ time python2 -c &quot;import sys; print sum((float(l) for l in sys.stdin))&quot; &lt; random_numbers 
 0.27s user 0.00s system 89% cpu 0.298 total

Python 3(3.6.8)

$ time python3 -c &quot;import sys; print(sum((float(l) for l in sys.stdin)))&quot; &lt; random_number
0.37s user 0.02s system 98% cpu 0.393 total

Ruby(2.3.3)

$  time ruby -e &#039;sum = 0; File.foreach(ARGV.shift) {|line| sum+=line.to_i}; puts sum&#039; random_numbers
 0.42s user
 0.03s system
 72% cpu
 0.625 total

Perl(5.24.1)

$ time perl -nle &#039;$sum += $_ } END { print $sum&#039; random_numbers
 0.24s user
 0.01s system
 99% cpu
 0.249 total

awk(4.1.4)

$ time awk &#039;{ sum += $0 } END { print sum }&#039; random_numbers
 0.26s user
 0.01s system
 99% cpu
 0.265 total
$ time awk &#039;{ sum += $1 } END { print sum }&#039; random_numbers
 0.34s user
 0.01s system
 99% cpu
 0.354 total

C(clang 版本 3.3;gcc(Debian 6.3.0-18)6.3.0)

 $ gcc sum.c -o sum &amp;&amp; time ./sum &lt; random_numbers   
 0.10s user
 0.00s system
 96% cpu
 0.108 total

更新更多语言

Lua(5.3.5)

$ time lua -e &#039;sum=0; for line in io.lines() do sum=sum+line end; print(sum)&#039; &lt; random_numbers 
 0.30s user 
 0.01s system
 98% cpu
 0.312 total

tr (8.26)必须在 bash 中计时,与 zsh 不兼容

$time { { tr &quot;
&quot; + &lt; random_numbers ; echo 0; } | bc; }
real    0m0.494s
user    0m0.488s
sys 0m0.044s

sed (4.4)必须在 bash 中计时,与 zsh 不兼容

$  time { head -n 10000 random_numbers | sed &#039;:a;N;s/
/+/;ta&#039; |bc; }
real    0m0.631s
user    0m0.628s
sys     0m0.008s
$  time { head -n 100000 random_numbers | sed &#039;:a;N;s/
/+/;ta&#039; |bc; }
real    1m2.593s
user    1m2.588s
sys     0m0.012s

注意:sed 调用似乎在具有更多可用内存的系统上运行得更快(请注意用于对 sed 进行基准测试的数据集较小)

朱莉娅(0.5.0)

$ time julia -e &#039;print(sum(readdlm(&quot;random_numbers&quot;)))&#039;
 3.00s user 
 1.39s system 
 136% cpu 
 3.204 total
$  time julia -e &#039;print(sum(readtable(&quot;random_numbers&quot;)))&#039;
 0.63s user 
 0.96s system 
 248% cpu 
 0.638 total

请注意,与 R 一样,文件 I/O 方法具有不同的性能。

解决方案 21:

Bash 变体

raw=$(cat file)
echo $(( ${raw//$&#039;
&#039;/+} ))

$ wc -l file
10000 file

$ time ./test
323390

real    0m3,096s
user    0m3,095s
sys     0m0,000s

这里发生了什么?将文件内容读入 $raw var。然后通过将所有新行更改为“+”从此 var 创建数学语句

解决方案 22:

只要只有整数,我基本上就会将文件转换为 bash 数学表达式并执行它。它类似于上面使用“bc”的解决方案,但速度更快。观察内部表达式末尾的零是最后一行参数所必需的。我已经用 475.000 行对其进行了测试,用时不到一秒。

echo $(($(cat filename | tr &#039;
&#039; &#039;+&#039;)0))

解决方案 23:

又一个好玩的

sum=0;for i in $(cat file);do sum=$((sum+$i));done;echo $sum

或者另一个仅限 bash

s=0;while read l; do s=$((s+$l));done&lt;file;echo $s

但 awk 解决方案可能是最好的,因为它最紧凑。

解决方案 24:

使用 Ruby:

ruby -e &quot;File.read(&#039;file.txt&#039;).split.inject(0){|mem, obj| mem += obj.to_f}&quot;

解决方案 25:

在 Go 中:

package main

import (
    &quot;bufio&quot;
    &quot;fmt&quot;
    &quot;os&quot;
    &quot;strconv&quot;
)

func main() {
    scanner := bufio.NewScanner(os.Stdin)
    sum := int64(0)
    for scanner.Scan() {
        v, err := strconv.ParseInt(scanner.Text(), 10, 64)
        if err != nil {
            fmt.Fprintf(os.Stderr, &quot;Not an integer: &#039;%s&#039;
&quot;, scanner.Text())
            os.Exit(1)
        }
        sum += v
    }
    fmt.Println(sum)
}

解决方案 26:

考虑到您需要通读整个文件,我不知道您是否可以得到比这更好的结果。

$sum = 0;
while(&lt;>){
   $sum += $_;
}
print $sum;

解决方案 27:

我还没有测试过但是它应该可以工作:

cat f | tr &quot;
&quot; &quot;+&quot; | sed &#039;s/+$/
/&#039; | bc

如果 bc 不处理 EOF 和 EOL,您可能必须在 bc 之前的字符串中添加“n”(例如通过 echo)...

解决方案 28:

这是另一个:

open(FIL, &quot;a.txt&quot;);

my $sum = 0;
foreach( &lt;FIL> ) {chomp; $sum += $_;}

close(FIL);

print &quot;Sum = $sum
&quot;;

解决方案 29:

您可以使用 Alacon( Alasql数据库的命令行实用程序)来完成此操作。

它与 Node.js 一起工作,因此您需要安装Node.js然后安装Alasql包:

要计算 TXT 文件的总和,可以使用以下命令:

> node alacon &quot;SELECT VALUE SUM([0]) FROM TXT(&#039;mydata.txt&#039;)&quot;

解决方案 30:

+用 替换所有新行,添加0并将其发送给解释器不是更容易吗Ruby

(sed -e &quot;s/$/+/&quot; file; echo 0)|irb

如果你没有irb,你可以将其发送到bc,但你必须删除除最后一个换行符(的echo)之外的所有换行符。最好使用tr,除非你有 博士学位sed

(sed -e &quot;s/$/+/&quot; file|tr -d &quot;
&quot;; echo 0)|bc
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用