Replace NaN with mean of previous values

4 vues (au cours des 30 derniers jours)
Tomás Nunes
Tomás Nunes le 10 Mai 2018
Modifié(e) : Jan le 12 Mai 2018
Hello, I have a vector that contains some Nan's. How can I replace them with a mean of the previous 5 values? This is what I have tried until now:
for i = 254:1:size(vector)
vector(isnan(vector(i))) = mean(vector(i-5:i-1));
end
and
for i = 254:1:size(vector)
if vector(i) == NaN
vector(i) = mean(vector(i-5:i-1));
end
end
Thank you!
  4 commentaires
Tomás Nunes
Tomás Nunes le 11 Mai 2018
do you know what is wrong with the formula? Thank you
Jan
Jan le 12 Mai 2018
If the 2nd one is wanted, Ameer Hamza's answer does not, what you want.

Connectez-vous pour commenter.

Réponse acceptée

Ameer Hamza
Ameer Hamza le 10 Mai 2018
Modifié(e) : Ameer Hamza le 10 Mai 2018
You can avoid for loop altogether
ind = find(isnan(vector));
vector(ind) = arrayfun(@(x) nanmean(vector(x-5:x-1)), ind);
a vectorized code is usually faster and at worst equivalent to loops. Also using arrayfun() makes your code more readable.
  5 commentaires
Tomás Nunes
Tomás Nunes le 11 Mai 2018
i mean, this is kinda dumb, right? it would be easier to set the nans to a stupid value like 100000 (which has nothing to do with the remaining values of the vector) and then replace them with the mean of the previous days. it would be quite easier, correct?
Ameer Hamza
Ameer Hamza le 11 Mai 2018
Modifié(e) : Ameer Hamza le 11 Mai 2018
Why do you need to set nan to 100000. You can directly replace them as I showed in my answer. Does your vector contain more the 5 consecutive nan, in that case, this method will not work. Also, an @Jan pointed out in the comment, if more then one nan occur closely (less than 5 elements apart) my answer do average using first example in Jan's comment.

Connectez-vous pour commenter.

Plus de réponses (3)

Steven Lord
Steven Lord le 10 Mai 2018
Use the fillmissing function, specifically the F = fillmissing(A,movmethod,window) syntax. You're going to want to specify a two-element vector [b f] as the window input as described in the documentation for that input argument.

Fangjun Jiang
Fangjun Jiang le 10 Mai 2018
Use the second code, replace size() with numel()

Jan
Jan le 11 Mai 2018
Modifié(e) : Jan le 12 Mai 2018
You forgot to mention, which problem you have with the shown code.
1. size(vector) replies a vector. If vector is a row vector, the output is e.g. [1, 1000]. When this is used in
for i = 254:1:size(vector)
only the first element is used. So this is equivalent to:
for i = 254:1:1
As Fangjun Jiang has suggested already, use numel instead of size.
2. A comparison with NaN is FALSE in every case by definition. See:
NaN == NaN
Therefore
if vector(i) == NaN
must fail. Use this instead:
if isnan(vector(i))
A working version:
for k = 254:numel(vector)
if isnan(vector(k))
vector(k) = mean(vector(k-5:k-1));
end
end
Or
for k = find(isnan(vector))
vector(k) = mean(vector(k-5:k-1));
end

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!

Translated by