首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >C编程:移动平均滤波器

C编程:移动平均滤波器
EN

Stack Overflow用户
提问于 2020-06-22 13:57:19
回答 3查看 967关注 0票数 0

我已经尝试用C语言实现了Matlab函数smooth(y,span)的等价物。函数的Matlab代码为:

代码语言:javascript
复制
n = length(y);
span = min(span,n);
width = span-1+mod(span,2); % force it to be odd
c = filter(ones(width,1)/width,1,y);
cbegin = cumsum(y(1:width-2));
cbegin = cbegin(1:2:end)./(1:2:(width-2))';
cend = cumsum(y(n:-1:n-width+3));
cend = cend(end:-2:1)./(width-2:-2:1)';
c = [cbegin;c(width:end);cend];

下面是我的C代码:

代码语言:javascript
复制
#include <stdio.h>
#include <stdlib.h>
#include <math.h>

// Compute cumulative sum of a signal of a given size.
void cumulative_sum(float *cum_vector, float *signal, int size)
{
    int i;
    cum_vector[0]=0;
    cum_vector[1]=signal[0];
    for(i=2;i<=size;i++)
    {
        cum_vector[i] = cum_vector[i-1] + signal[i-1];
    }
 }

 // Moving average filter of a signal of a given size with a pre-defined span.
 void moving_average(float *vector, int size, int span)
{
if (span > size)
{
    span = size; //span = min(span,size);
}
int i;
int j=0;
int width = span - 1 + (span % 2); //force it to be odd
float *cum_x_tmp;
cum_x_tmp = (float *)malloc(size*sizeof(float));
cumulative_sum(cum_x_tmp,vector,size);

for (i=0;i<size;i++)
{
    if (i< (width - 2)/2 +((width-2)%2 !=0))
    {
        vector[i] = cum_x_tmp[2*i+1]/(2*i+1);
    }
    else if (i>=(width - 2)/2 +((width-2)%2 !=0) && i<= (width - 2)/2 +((width-2)%2 !=0) + size - width)
    {
        int ind1 =i + floor(width/2)+1;
        int ind2 =i - floor(width/2)-1+1;
        vector[i] = (cum_x_tmp[ind1] - cum_x_tmp[ind2])/width;
    }
    else if (i>(width - 2)/2 +((width-2)%2 !=0) + size - width)
    {
        int ind4 = size-1-width+3-1+2*j+1;
        vector[i] = (cum_x_tmp[size] - cum_x_tmp[ind4])/(width -2*(j+1));
        j++;
    }
}
}int main(int argc, const char * argv[]) {

int N=900;
float data;
FILE *fp;
fp = fopen("file.txt","r");

float *signal;
signal = (float*)malloc(N*sizeof(float));

int i;
for (i=0;i<N;i++)
{
    fscanf(fp, "%f", &data);
    signal[i]=data;
}
fclose(fp);

moving_average(signal,N,50);

free(signal);

return 0;}

我获得了与Matlab代码相同的值,直到代码中止时信号的某个索引。有人知道为什么和如何修复它吗?

提前感谢!

EN

回答 3

Stack Overflow用户

发布于 2020-06-22 15:13:42

如果需要在cumulative_sum中索引cum_vector[size],则需要确保cum_vector具有size+1元素。在您的代码中,它只有size。因此,您编写的代码超出了界限,这可能会导致稍后的崩溃。

moving_average中,您也可以访问cum_x_tmp[size]

按如下方式分配数组:

代码语言:javascript
复制
cum_x_tmp = (float *)malloc((size+1)*sizeof(float));

在Valgrind或类似的内存检查器下运行你的程序会指出这个问题,以及另一个答案中指出的内存泄漏。

票数 1
EN

Stack Overflow用户

发布于 2020-06-22 14:40:21

你有一个内存泄漏:

每次调用moving_average时,都会泄漏为cum_x_tmp分配的内存

代码语言:javascript
复制
    float *cum_x_tmp;    
    cum_x_tmp = (float *)malloc(size*sizeof(float));

当你不再需要它的时候,你需要释放它。此外,您应该向从malloc返回的值添加一个空检查,以检测何时分配失败。否则,您将在释放空指针时调用未定义的行为。

票数 0
EN

Stack Overflow用户

发布于 2021-05-05 06:44:58

对原始代码进行了清理,以便实际编译和正常工作。也就是Matlab的smooth()函数。如果你发现了一种更快和/或更安全的方法,请回帖。

代码语言:javascript
复制
#include <stdio.h>
#include <stdlib.h>
#include <math.h>

void get_cum_sum(double *cum_vector, double *signal, int size)
{
   int i;
   cum_vector[0] = signal[0];
   for(i = 1; i < size; i++)
   {
      cum_vector[i] = signal[i] + cum_vector[i-1];
   }
}

void smooth(double *vector, int size, int span)
{
   int i, window, half_window, tmp_index;
   double *cum_x_tmp;

   if (span > size)
   {
      span = size;
   }
   window = span - 1 + (span % 2);
   half_window = window / 2;

   cum_x_tmp = (double *)malloc((size) * sizeof(double));

   get_cum_sum(cum_x_tmp, vector, size);

   for (i = 0; i < size; i++)
   {
      if (i <= half_window)
      {
         vector[i] = (cum_x_tmp[2 * i]) / (2 * i + 1);
      }
      else if (i >= (size - half_window))
      {
         tmp_index = (size - (i + 1)) * 2 + 1;
         vector[i] = (cum_x_tmp[size - 1] - cum_x_tmp[i - (tmp_index / 2) - 1]) / tmp_index;
      }
      else
      {
         vector[i] = (cum_x_tmp[i + half_window] - cum_x_tmp[i - half_window - 1]) / window;
      }
   }
   free(cum_x_tmp);
}

int main(int argc, const char * argv[])
{
   int i;
   int N = 7;
   double data;
   FILE *fp;
   double *signal;

   signal = (double*)malloc(N*sizeof(double));
   fp = fopen("data.txt","r");

   for (i = 0; i < N; i++)
   {
      fscanf(fp, "%lf", &data);
      signal[i] = data;
   }

   fclose(fp);

   smooth(signal, N, 7);

   for (i = 0; i < N; i++)
   {
      printf("signal[%d] = %f\n", i, signal[i]);
   }

   free(signal);

   return 0;
}
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/62508232

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档