我想找一个矢量化方式来计算矢量的累积总和,但是有上限和下限。
在我的情况下,输入只包含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
你有什么想法吗?
答案 0 :(得分:5)
这是一个有点 hackish 的解决方案,但也许值得一提。
您可以使用有符号整数数据类型来执行求和,并使用利用该数据类型的固有限制。为此,输入需要转换为整数类型并乘以适当的因子,并且需要应用初始偏移。选择因子和偏移量作为lower
和upper
的函数。在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吗?