在 MATLAB 中分割向量 [英] Split vector in MATLAB

查看:278
本文介绍了在 MATLAB 中分割向量的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我正在尝试优雅地分割向量.例如,

I'm trying to elegantly split a vector. For example,

vec = [1 2 3 4 5 6 7 8 9 10]

根据另一个长度相同的 0 和 1 向量,其中 1 表示应该分割向量的位置 - 或者更确切地说是剪切:

According to another vector of 0's and 1's of the same length where the 1's indicate where the vector should be split - or rather cut:

cut = [0 0 0 1 0 0 0 0 1 0]

给我们一个类似于以下的单元格输出:

Giving us a cell output similar to the following:

[1 2 3] [5 6 7 8] [10]

推荐答案

解决方案代码

您可以使用 cumsum &accumarray 一个有效的解决方案 - >

%// Create ID/labels for use with accumarray later on
id = cumsum(cut)+1   

%// Mask to get valid values from cut and vec corresponding to ones in cut
mask = cut==0        

%// Finally get the output with accumarray using masked IDs and vec values 
out = accumarray(id(mask).',vec(mask).',[],@(x) {x})


基准测试

以下是在列出的解决此问题的三种最流行方法上使用大量输入时的一些性能数据 -


Benchmarking

Here are some performance numbers when using a large input on the three most popular approaches listed to solve this problem -

N = 100000;  %// Input Datasize

vec = randi(100,1,N); %// Random inputs
cut = randi(2,1,N)-1;

disp('-------------------- With CUMSUM + ACCUMARRAY')
tic
id = cumsum(cut)+1;
mask = cut==0;
out = accumarray(id(mask).',vec(mask).',[],@(x) {x});
toc

disp('-------------------- With FIND + ARRAYFUN')
tic
N = numel(vec);
ind = find(cut);
ind_before = [ind-1 N]; ind_before(ind_before < 1) = 1;
ind_after = [1 ind+1]; ind_after(ind_after > N) = N;
out = arrayfun(@(x,y) vec(x:y), ind_after, ind_before, 'uni', 0);
toc

disp('-------------------- With CUMSUM + ARRAYFUN')
tic
cutsum = cumsum(cut);
cutsum(cut == 1) = NaN;  %Don't include the cut indices themselves
sumvals = unique(cutsum);      % Find the values to use in indexing vec for the output
sumvals(isnan(sumvals)) = [];  %Remove NaN values from sumvals
output = arrayfun(@(val) vec(cutsum == val), sumvals, 'UniformOutput', 0);
toc

运行时

-------------------- With CUMSUM + ACCUMARRAY
Elapsed time is 0.068102 seconds.
-------------------- With FIND + ARRAYFUN
Elapsed time is 0.117953 seconds.
-------------------- With CUMSUM + ARRAYFUN
Elapsed time is 12.560973 seconds.


特殊情况:如果您可能有 1 的运行,您需要修改下面列出的一些内容 -


Special case scenario: In cases where you might have runs of 1's, you need to modify few things as listed next -

%// Mask to get valid values from cut and vec corresponding to ones in cut
mask = cut==0  

%// Setup IDs differently this time. The idea is to have successive IDs.
id = cumsum(cut)+1
[~,~,id] = unique(id(mask))
      
%// Finally get the output with accumarray using masked IDs and vec values 
out = accumarray(id(:),vec(mask).',[],@(x) {x})

在这种情况下运行的示例 -

Sample run with such a case -

>> vec
vec =
     1     2     3     4     5     6     7     8     9    10
>> cut
cut =
     1     0     0     1     1     0     0     0     1     0
>> celldisp(out)
out{1} =
     2
     3
out{2} =
     6
     7
     8
out{3} =
    10

这篇关于在 MATLAB 中分割向量的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆