上部和上部的cumsum下限?

时间:2015-08-06 06:31:32

标签: matlab vectorization cumsum

我想找一个矢量化方式来计算矢量的累积总和,但是有上限和下限。

在我的情况下,输入只包含1&1和#1。您可以在答案中使用此假设。当然,也欢迎更通用的解决方案。

例如:

x     = [1 1 1 1 -1 -1 -1 -1 -1 -1];
upper = 3;
lower = 0;
s     = cumsum(x)                    %// Ordinary cumsum.
s =
     1     2     3     4     3     2     1     0    -1    -2

y     = cumsumlim(x, upper, lower)   %// Cumsum with limits.
y =
     1     2     3     3     2     1     0     0     0     0
                 ^                       ^
                 |                       |
            upper limit             lower limit

当累计金额达到上限(第3个元素)时,它不再增加。同样,当累积总和达到下限(在第7个元素)时,它不再减少。 for循环版本将是这样的:

function y = cumsumlim(x, upper, lower)

y = zeros(size(x));
y(1) = x(1);

for i = 2 : numel(x)
    y(i) = y(i-1) + x(i);
    y(i) = min(y(i), upper);
    y(i) = max(y(i), lower);
end

end

你有什么想法吗?

2 个答案:

答案 0 :(得分:5)

这是一个有点 hackish 的解决方案,但也许值得一提。

您可以使用有符号整数数据类型来执行求和,并使用利用该数据类型的固有限制。为此,输入需要转换为整数类型并乘以适当的因子,并且需要应用初始偏移。选择因子和偏移量作为lowerupper的函数。在cumsum之后,撤消乘法和偏移以获得所需的结果。

在您的示例中,数据类型int8就足够了;所需的因子和偏移量分别为85-128

x = [1 1 1 1 -1 -1 -1 -1 -1 -1];
result = cumsum([-128 int8(x)*85]); %// integer sum, with factor and initial offset
result = (double(result(2:end))+128)/85; %// undo factor and offset

给出了

result =
     1     2     3     3     2     1     0     0     0     0

答案 1 :(得分:4)

我不会为您提供神奇的矢量化方法,但我会向您提供一些可能会帮助您继续工作的数据。

您的cumsumlim功能非常快!

tic
for ii = 1:100
    y = cumsumlim(x,3,0);
end
t = toc;
disp(['Length of vector: ' num2str(numel(x))])
disp(['Total time for one execution: ' num2str(t*10), ' ms.'])
Length of vector: 65000
Total time for one execution: 1.7965 ms.

我真的怀疑这是你的瓶颈。你试过profiling the code吗?