【心电信号ECG】基于支持向量机SVM心电图心搏检测与分类附Matlab复现含文献

简介: ✅作者简介:热爱科研的Matlab仿真开发者,擅长数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。🍎 往期回顾关注个人主页:Matlab科研工作室 👇 关注我领取海量matlab电子书和数学建模资料 🍊个人信条:格物致知,完整Matlab代码获取及仿真咨询内容私信。🔥 内容介绍 摘要——在第三项任务中,我们开发了一种基于支持向量机(SVM)的心电图(ECG)节律分类流程,利用MIT- BIH 心律失常数据库区分正常与异常心搏。通过潘-

✅作者简介:热爱科研的Matlab仿真开发者,擅长数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。

🍎 往期回顾关注个人主页:Matlab科研工作室

👇 关注我领取海量matlab电子书和数学建模资料

🍊个人信条:格物致知,完整Matlab代码获取及仿真咨询内容私信。

🔥 内容介绍
摘要——在第三项任务中,我们开发了一种基于支持向量机(SVM)的心电图(ECG)节律分类流程,利用MIT- BIH 心律失常数据库区分正常与异常心搏。通过潘-汤普金斯算法处理五份记录的ECG信号以检测QRS波群,随后提取形态学、时间及频谱特征。评估了两种特征配置:基线配置(8个特征)和扩展配置(12个特征)。采用 RBF 核训练的 SVM 模型在扩展特征集中实现了更高的分类准确率,同时提升了敏感性和特异性。通过混淆矩阵和基于主成分分析(PCA)的特征分离等可视化手段,验证了该方法在真实ECG监测场景中检测心律失常的有效性。

Image

Image

Image

⛳️ 运行结果
Image

Image

Image

📣 部分代码
= 1).';

% right_ind = find(diff([poss_reg.' 0]) == -1).';

%

% % R, Q, S detection

% R_loc = zeros(numel(left_ind), 1);

% Q_loc = zeros(numel(left_ind), 1);

% S_loc = zeros(numel(left_ind), 1);

% for i = 1:numel(left_ind)

% [~, R_loc(i)] = max(ecg_m(left_ind(i):right_ind(i)));

% R_loc(i) = R_loc(i)-1+left_ind(i);

% [~, Q_loc(i)] = min(ecg_m(left_ind(i):R_loc(i)));

% Q_loc(i) = Q_loc(i)-1+left_ind(i);

% [~, S_loc(i)] = min(ecg_m(R_loc(i):right_ind(i)));

% S_loc(i) = S_loc(i)-1+R_loc(i);

% end

%

% % Clean incomplete edges

% if R_loc(1)*ts < 0.2

% Q_loc(1) = []; R_loc(1) = []; S_loc(1) = [];

% left_ind(1) = []; right_ind(1) = [];

% end

% if (numel(ecg_m)-R_loc(end))*ts < 0.2

% Q_loc(end) = []; R_loc(end) = []; S_loc(end) = [];

% left_ind(end) = []; right_ind(end) = [];

% end

% n_beats_det = numel(R_loc);

%

% % P and T detection

% P_loc = zeros(n_beats_det, 1);

% T_loc = zeros(n_beats_det, 1);

% for i = 1:n_beats_det

% pL = max(1, left_ind(i) - round(0.2 * fs));

% pR = left_ind(i);

% [~, p] = max(ecg_m(pL:pR));

% P_loc(i) = pL + p - 1;

%

% tL = right_ind(i);

% tR = min(numel(ecg_m), tL + round(0.4 * fs));

% [~, t] = max(ecg_m(tL:tR));

% T_loc(i) = tL + t - 1;

% end

%

% % Feature Extraction

% RS_width = ts * (S_loc - R_loc);

% QS_width = ts * (S_loc - Q_loc);

% QR_width = ts * (R_loc - Q_loc);

% pre_RR_int = [0; ts * diff(R_loc)];

% post_RR_int = [pre_RR_int(2:end); 0];

%

% MPSD = zeros(n_beats_det, 1);

% area_QR = zeros(n_beats_det, 1);

% area_RS = zeros(n_beats_det, 1);

% for i = 1:n_beats_det

% MPSD(i) = mean(abs(fft(ecg_m(P_loc(i):T_loc(i)))).^2);

% area_QR(i) = trapz(ecg_m(Q_loc(i):R_loc(i)));

% area_RS(i) = trapz(ecg_m(R_loc(i):S_loc(i)));

% end

%

% features = [QS_width, pre_RR_int, post_RR_int, QR_width, RS_width, MPSD, area_QR, area_RS];

% end

function [features, R_loc, Q_loc, S_loc, P_loc, T_loc, n_beats_det, left_ind, right_ind] = extract_ecg_features(ecg_m, ecg_MW, fs, template_normal)

ts = 1/fs;



% === Thresholding and region detection ===

poss_reg = (ecg_MW > mean(ecg_MW))';

left_ind = find(diff([0 poss_reg.']) == 1).';

right_ind = find(diff([poss_reg.' 0]) == -1).';



% === R, Q, S detection ===

R_loc = zeros(numel(left_ind), 1);

Q_loc = zeros(numel(left_ind), 1);

S_loc = zeros(numel(left_ind), 1);

for i = 1:numel(left_ind)

    [~, R_loc(i)] = max(ecg_m(left_ind(i):right_ind(i)));

    R_loc(i) = R_loc(i) - 1 + left_ind(i);

    [~, Q_loc(i)] = min(ecg_m(left_ind(i):R_loc(i)));

    Q_loc(i) = Q_loc(i) - 1 + left_ind(i);

    [~, S_loc(i)] = min(ecg_m(R_loc(i):right_ind(i)));

    S_loc(i) = S_loc(i) - 1 + R_loc(i);

end



% === Clean boundaries ===

if R_loc(1)*ts < 0.2

    Q_loc(1) = []; R_loc(1) = []; S_loc(1) = [];

    left_ind(1) = []; right_ind(1) = [];

end

if (numel(ecg_m)-R_loc(end))*ts < 0.2

    Q_loc(end) = []; R_loc(end) = []; S_loc(end) = [];

    left_ind(end) = []; right_ind(end) = [];

end

n_beats_det = numel(R_loc);



% === P and T detection ===

P_loc = zeros(n_beats_det, 1);

T_loc = zeros(n_beats_det, 1);

for i = 1:n_beats_det

    pL = max(1, left_ind(i) - round(0.2 * fs));

    pR = left_ind(i);

    [~, p] = max(ecg_m(pL:pR));

    P_loc(i) = pL + p - 1;



    tL = right_ind(i);

    tR = min(numel(ecg_m), tL + round(0.4 * fs));

    [~, t] = max(ecg_m(tL:tR));

    T_loc(i) = tL + t - 1;

end



% === Initialize feature arrays ===

QS_width = zeros(n_beats_det,1);

QR_width = zeros(n_beats_det,1);

RS_width = zeros(n_beats_det,1);

pre_RR_int = [0; ts * diff(R_loc)];

post_RR_int = [pre_RR_int(2:end); 0];

MPSD = zeros(n_beats_det,1);

area_QR = zeros(n_beats_det,1);

area_RS = zeros(n_beats_det,1);

auto_corr_val = zeros(n_beats_det,1);

ST_dev = zeros(n_beats_det,1);

ST_slope = zeros(n_beats_det,1);

template_corr = zeros(n_beats_det,1);



for i = 1:n_beats_det

    % === Duration Features ===

    QS_width(i) = ts * (S_loc(i) - Q_loc(i));

    QR_width(i) = ts * (R_loc(i) - Q_loc(i));

    RS_width(i) = ts * (S_loc(i) - R_loc(i));



    % === MPSD ===

    if P_loc(i) < T_loc(i)

        window = ecg_m(P_loc(i):T_loc(i));

        MPSD(i) = mean(abs(fft(window)).^2);

    else

        MPSD(i) = 0;

    end



    % === Area under QR, RS ===

    if Q_loc(i) < R_loc(i)

        area_QR(i) = trapz(ecg_m(Q_loc(i):R_loc(i)));

    else

        area_QR(i) = 0;

    end

    if R_loc(i) < S_loc(i)

        area_RS(i) = trapz(ecg_m(R_loc(i):S_loc(i)));

    else

        area_RS(i) = 0;

    end



    % === Autocorrelation ===

    if Q_loc(i) < S_loc(i)

        segment = ecg_m(Q_loc(i):S_loc(i));

        if length(segment) >= 2

            acf = xcorr(segment, 'coeff');

            acf(acf == 1 | isnan(acf)) = [];  % Remove trivial peak

            auto_corr_val(i) = max(acf(:), [], 'omitnan');

        else

            auto_corr_val(i) = 0;

        end

    else

        auto_corr_val(i) = 0;

    end



    % === ST deviation and slope ===

    st_start = S_loc(i) + round(0.02 * fs);

    st_end   = min(T_loc(i), numel(ecg_m));

    if st_end > st_start

        y = ecg_m(st_start:st_end);

        x = (st_start:st_end) * ts;

        ST_dev(i) = mean(y);

        p = polyfit(x, y, 1);

        ST_slope(i) = p(1);

    else

        ST_dev(i) = 0;

        ST_slope(i) = 0;

    end



    % === Correlation with normal template ===

    if Q_loc(i) < S_loc(i)

        beat_seg = ecg_m(Q_loc(i):S_loc(i));

        if length(beat_seg) >= 2 && length(template_normal) >= 2

            beat_seg = interp1(1:length(beat_seg), beat_seg, ...

                linspace(1,length(beat_seg), length(template_normal)), 'linear', 'extrap');

            c = corrcoef(beat_seg, template_normal);

            if size(c,1)==2 && size(c,2)==2

                template_corr(i) = c(1,2);

            else

                template_corr(i) = 0;

            end

        else

            template_corr(i) = 0;

        end

    else

        template_corr(i) = 0;

    end

end



% === Final Feature Matrix (12 features) ===

features = [QS_width, pre_RR_int, post_RR_int, ...

            QR_width, RS_width, MPSD, ...

            area_QR, area_RS, auto_corr_val, ...

            ST_dev, ST_slope, template_corr];

end

🔗 参考文献
图片
🏆团队擅长辅导定制多种科研领域MATLAB仿真,助力科研梦:

相关文章
|
Java 数据安全/隐私保护 Spring
SpringSecurity 权限管理的实现
SpringSecurity是一个权限管理框架,核心是认证和授权,前面介绍过了认证的实现和源码分析,本文重点来介绍下权限管理这块的原理。
311 0
|
移动开发 资源调度 JavaScript
夹吃灰,推荐:实现 SVG 动画的 5 个 JavaScript 库~
现如今,SVG 已经广泛应用于 Web 开发当中,可缩放而不失真(矢量图形),就是它的最大优势! 但是,你知道 SVG 除了可以被用作于静态图片,还可以应用于 SVG 动画吗? 本篇带来实现 SVG 动画的 5 个 JavaScript 库!
1487 0
|
5月前
|
安全 网络安全 开发工具
Mac电脑多平台Git账号设置
通过配置SSH密钥与config文件,可为GitHub、Gitee等平台分配独立密钥,实现自动识别与认证。生成密钥后,将公钥添加至对应平台,并在~/.ssh/config中设置主机别名与密钥路径。此后使用SSH地址克隆仓库,Git将自动选用正确密钥,免去手动切换与密码输入,提升效率与安全性。(238字)
624 2
|
C语言
【C语言程序设计——入门】基本数据类型与表达式(头歌实践教学平台习题)【合集】
这份文档详细介绍了编程任务的多个关卡,涵盖C语言的基础知识和应用。主要内容包括: 1. **目录**:列出所有关卡,如`print函数操作`、`转义字符使用`、`数的向上取整`等。 2. **各关卡的任务描述**:明确每关的具体编程任务,例如使用`printf`函数输出特定字符串、实现向上取整功能等。 3. **相关知识**:提供完成任务所需的背景知识,如格式化输出、算术运算符、关系运算符等。 4. **编程要求**:给出具体的代码编写提示。 5. **测试说明**:包含预期输入输出,帮助验证程序正确性。 6. 文档通过逐步引导学习者掌握C语言的基本语法和常用函数,适合初学者练习编程技能。
339 1
|
SQL 安全 PHP
PHP安全性深度剖析:防范常见漏洞与最佳实践####
本文深入探讨了PHP编程中不可忽视的安全隐患,重点介绍了SQL注入、XSS攻击、CSRF攻击及文件包含漏洞等四大常见安全威胁。通过详尽的案例分析与防御策略阐述,为开发者提供了一套实用的安全编码指南。文章强调,提升代码安全性是保障Web应用稳健运行的关键,鼓励开发者在日常开发中积极践行安全最佳实践。 ####
|
人工智能 小程序 前端开发
【一步步开发AI运动小程序】六、人体骨骼图绘制
随着AI技术的发展,阿里体育等公司推出的AI运动APP如“乐动力”、“天天跳绳”等,使云上运动会、线上健身等概念广受欢迎。本文将引导您从零开始,利用“云智AI运动识别小程序插件”,在小程序中实现类似功能,包括人体骨骼图的绘制原理及其实现代码,确保骨骼图与人体图像精准重合。下篇将继续介绍运动分析方法。
|
关系型数据库 分布式数据库 数据库
PolarDB闪电助攻,《香肠派对》百亿好友关系实现毫秒级查询
PolarDB分布式版助力《香肠派对》实现百亿好友关系20万QPS的毫秒级查询。
PolarDB闪电助攻,《香肠派对》百亿好友关系实现毫秒级查询
|
传感器 安全
透明铝:未来材料的强度与透明度
【10月更文挑战第20天】透明铝是一种由氮氧化铝或氧化铝陶瓷基板制成的透明陶瓷材料,具备高强度和高透明度。本文探讨其基本原理、技术特性、应用领域及未来趋势,揭示其在军事、航空航天、汽车制造和建筑设计等领域的广泛应用前景和巨大潜力。
|
存储 Python
【Python】已解决:Python读取字典查询键报错“KeyError: ‘d‘”
【Python】已解决:Python读取字典查询键报错“KeyError: ‘d‘”
694 1
|
安全 网络安全 数据安全/隐私保护
BUUCTF:Misc 解析(十)
BUUCTF:Misc 解析(十)