iOS 心电数据滤波处理(中值滤波)

前段时间接触了一个蓝牙接收心电数据的项目。由于收到的原始数据并没有在硬件上做滤波处理,心电数据绘制存在基线漂移的情况(如下图)。


原始心电数据

苦苦找寻网上滤波的代码,全是学术论文以及MATLAB的处理。看着论文里的各种公式,简直是要崩溃的节奏。在各种查阅资料后,发现中值滤波方法较为简单直接,所以选用了这种滤波方法处理心电数据。

中值滤波数学实现:对一个数字信号序列xj(-∞<j<∞)进行滤波处理时,首先要定义一个长度为奇数的n长窗口,n=2N+1,N为正整数。设在某一个时刻,窗口内的信号样本为x(i-N),…,x(i),…,x(i+N),其中x(i)为位于窗口中心的信号样本值。对这L个信号样本值按从小到大的顺序排列后,其中值,在i处的样值,便定义为中值滤波的输出值。

根据上面直接转换成代码,创建NSArray 分类 MedianFilter

//中值滤波 窗口值 n 此处n若为偶数 则中值取 n个数排序后 中间两个数的平均值
- (NSArray *)medianFilterWithN:(int)n {
    NSMutableArray *resultArray = [NSMutableArray array];
    for (int i = 0; i < self.count; i ++) {
        //开始截取的index
        int start = n % 2 == 0 ? i - n / 2 : i - (n - 1) / 2;
        //从目标数组中截取数组 个数 n
        //若start < 0 则用0 补齐
        //如 @[@1, @1, @5, @6, @2] n=3 i=0 时 start=-1 则 subArr = @[@0, @1, @1]
        //若start+n > array.count 则 后续数用0补齐
        NSArray *subArr = [self subZeroizeArrayLocation:start length:n];
        // 从小到大排序
        NSArray *sortArr = [subArr sortedArrayUsingComparator:^NSComparisonResult(NSString *  _Nonnull obj1, NSString *  _Nonnull obj2) {
            return [@(obj1.intValue) compare:@(obj2.intValue)];
        }];
        // 若n 为奇数 取中间值 若n 为偶数 去 中间两个数的平均值
        if (n % 2 == 0) {
            int index = n / 2;
            float result = ([sortArr[index - 1] floatValue] + [sortArr[index] floatValue]) / 2;
            [resultArray addObject:[NSString stringWithFormat:@"%.2f", result]];
        } else {
            int index = (n - 1) / 2;
            NSNumber *result = sortArr[index];
            [resultArray addObject:result];
        }
    }
    // 求出中值后 用原始数据 - 中值,得到滤波后的值
    NSArray *subtracArray = [self subtractWithMedianArray:resultArray];
    return subtracArray;
}



- (NSArray *)subZeroizeArrayLocation:(int)location length:(int)length {
    NSMutableArray *resultArr = [NSMutableArray array];
    //startIndex < 0 时
    if (location < 0) {
        //如果 从0开始截取的lenght + location 的 count < self.count
        //则可以直接用 subarrayWithRange 方法截取后半部分,前半部分用 0 填充
        if (length + location <= self.count) {
            NSArray *subArr = [self subarrayWithRange:NSMakeRange(0, length + location)];
            for (int i = 0; i < abs(location); i ++) {
                [resultArr addObject:@"0"];
            }
            [resultArr addObjectsFromArray:subArr];
        } else {//如果 self.count 不够长 则 前面补0 后面也要补0
            for (int i = 0; i < abs(location); i ++) {
                [resultArr addObject:@"0"];
            }
            [resultArr addObjectsFromArray:self];
            for (int i = 0; i < length - self.count; i ++) {
                [resultArr addObject:@"0"];
            }
        }
    } else if (location >= 0) {//startIndex >= 0 时
        //如果 self 够长 则直接用 subarrayWithRange 方法截取
        if (length + location <= self.count) {
            [resultArr addObjectsFromArray:[self subarrayWithRange:NSMakeRange(location, length)]];
        } else {// 否则 在后面补 0
            NSArray *subArr = [self subarrayWithRange:NSMakeRange(location, self.count - location)];
            [resultArr addObjectsFromArray:subArr];
            for (int i = 0; i < length - subArr.count; i ++) {
                [resultArr addObject:@"0"];
            }
        }
    }
    return resultArr;
}

- (NSArray *)subtractWithMedianArray:(NSArray *)medianArray {
    if (self.count == medianArray.count) {
        NSMutableArray *array = [NSMutableArray array];
        for (int i = 0; i < self.count; i ++) {
            float x = [self[i] floatValue];
            float y = [medianArray[i] floatValue];
            [array addObject:[NSString stringWithFormat:@"%.2f",x - y]];
        }
        return array;
    } else {
        return nil;
    }
}

MATLAB 模拟 中值滤波后的心电图 (窗口数55)


滤波后心电数据

处理效果显而易见,值得注意的是,这种基础的中值滤波算法,随着所选用窗口长度的增加,滤波的计算量将会迅速增加。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,392评论 5 470
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,258评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,417评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,992评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,930评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,199评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,652评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,327评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,463评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,382评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,432评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,118评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,704评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,787评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,999评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,476评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,057评论 2 341

推荐阅读更多精彩内容

  • 锤子科技在4 月 9 日下午 14:30 ,于北京工业大学奥林匹克体育馆举行春季新品发布会。 本次发布会的看点,除...
    汤帅同学阅读 540评论 0 2
  • 前段时间,《最好的我们》火热的时候,我并没有看。反而是最近,我一口气就把剧看完了。 仿佛每个人都在这部剧里看到了自...
    WAnnoQ_阅读 694评论 3 3
  • 同频:在小家(三口之家)的亲子关系中占有很大的作用,同频时关系更融洽、顺畅 与正能量的同频:更多人的关系变得和谐时...
    妞妞姐S阅读 99评论 0 0
  • 1. text-align: center的作用是什么,作用在什么元素上?能让什么元素水平居中? text-ali...
    madpluto阅读 242评论 0 0