PPG信号和ECG信号检测血管年龄

news2024/12/28 23:01:57

PAT 通常用作动脉硬度的间接测量值或心血管健康的指标。它与各种生理和病理状况有关,例如高血压、动脉硬化和内皮功能障碍。

通过脉搏到达时间进行测量,简单来说就是 先从脉冲传输时间 PPG 数据集中提取数据,提取此数据集中每个对象的脉冲到达时间,再提取参考年龄,进行一些比较。
用的pulse transit time PPG数据集
22*3个数据记录了走跑坐
Bp_sys_start/end systolic blood pressure 收缩压
Bp_dia_start/end diastolic blood pressure 舒张压
Hr_1_start/end 测量开始时使用欧姆龙 HEM-7322 血压计测量的心率 (bpm)
Hr_2_start/end 测量开始时使用 iHealth Air 无线脉搏血氧仪测量的心率 (bpm)
spo2_start/end SpO2血氧饱和度

对于ecg和ppg信号,用neurokit2和wfdb库导入数据,
有一个需要注意的预处理操作是:
让ppg = -1*ppg,使ppg信号看起来更像血压信号。
再用neurokit2进行信号过滤,消除噪音
然后检测峰值:
ppg:
请添加图片描述
ecg:
请添加图片描述
对于峰值时间的检测,ppg和ecg略有区别:
ecg可以直接提取R 波的时间,它表示心室去极化的时间(促使心室收缩的电活动)。
ppg则一般不使用收缩峰时间,因为脉搏开始和收缩峰之间可能存在可变延迟,所以一般会用脉冲开始时间
请添加图片描述

def get_ppg_onsets(ppg, pks, fs):
    ons = np.empty(0)
    for i in range(len(pks) - 1):
        start = pks[i]
        stop = pks[i + 1]
        ibi = ppg[start:stop]
        aux_ons = np.argmin(ibi)
        ind_ons = aux_ons.astype(int)
        ons = np.append(ons, ind_ons + start)

    ons = ons.astype(int)
    return ons

def get_data(record_list,database_name,required_signals,required_duration,required_activity):

    matching_recs = {'dir': [], 'name': [], 'length': [], 'start_sbp': [], 'end_sbp': [], 'delta_sbp': [], 'age': []}

    for record in record_list:
        print('Record: {}'.format(record), end="", flush=True)

        record_data = wfdb.rdheader(record, pn_dir=database_name, rd_segments=True)

        # Check whether the required signals are present in the record
        sigs_present = record_data.sig_name
        if not all(x in sigs_present for x in required_signals):
            print('   (missing signals)')
            continue

        if not required_activity in record:
            print('   (not required activity)')
            continue

        seg_length = record_data.sig_len / (record_data.fs)

        if seg_length < required_duration:
            print(f' (too short at {seg_length / 60:.1f} mins)')
            continue

        # This record does meet the requirements, so extract information and data from it
        # Information
        matching_recs['dir'].append(database_name)
        matching_recs['name'].append(record_data.record_name)
        matching_recs['length'].append(seg_length)

        # Blood pressure measurements
        start_el = record_data.comments[0].index('<bp_sys_start>: ') + len('<bp_sys_start>: ')
        end_el = record_data.comments[0].index('<bp_sys_end>') - 1
        matching_recs['start_sbp'].append(int(record_data.comments[0][start_el:end_el]))

        start_el = record_data.comments[0].index('<bp_sys_end>: ') + len('<bp_sys_end>: ')
        end_el = record_data.comments[0].index('<bp_dia_start>') - 1
        matching_recs['end_sbp'].append(int(record_data.comments[0][start_el:end_el]))

        matching_recs['delta_sbp'].append(matching_recs['end_sbp'][-1] - matching_recs['start_sbp'][-1])

        # ages
        start_el = record_data.comments[0].index('<age>: ') + len('<age>: ')
        end_el = record_data.comments[0].index('<bp_sys_start>') - 1
        matching_recs['age'].append(float(record_data.comments[0][start_el:end_el]))
        print('   (met requirements)')

    print(f"A total of {len(matching_recs['dir'])} out of {len(record_list)} records met the requirements.")
    return matching_recs

def get_pat(matching_recs,start_seconds,n_seconds_to_load,required_signals):
    subj_nos = [i for i in range(len(matching_recs['name']))]
    median_pats = []
    for subj_no in subj_nos:

        # specify this subject's record name and directory:
        record_name = matching_recs['name'][subj_no]
        record_dir = matching_recs['dir'][subj_no]

        # extract this subject's signals

        record_data = wfdb.rdheader(record_name, pn_dir=record_dir, rd_segments=True)
        fs = record_data.fs

        # Specify timings of segment to be extracted
        sample_start = fs * start_seconds
        sample_end = fs * (start_seconds + n_seconds_to_load)

        # Load segment data
        segment_data = wfdb.rdrecord(record_name=record_name,
                                     channel_names=required_signals,
                                     sampfrom=sample_start,
                                     sampto=sample_end,
                                     pn_dir=record_dir)

        ppg_col = segment_data.sig_name.index("pleth_1")
        ppg_final = segment_data.p_signal[:, ppg_col]

        ecg_col = segment_data.sig_name.index("ecg")
        ecg_final = segment_data.p_signal[:, ecg_col]

        fs = segment_data.fs
        ppg = -1 * ppg_final

        # filter signals
        ppg = nk2.ppg_clean(ppg, sampling_rate=fs)
        ecg = nk2.ecg_clean(ecg_final, sampling_rate=fs)

        # detect beats in signals
        ppg_clean = nk2.ppg_clean(ppg, sampling_rate=fs)
        ppg_peaks = nk2.ppg_findpeaks(ppg_clean, method="elgendi", show=False)['PPG_Peaks']

        ecg_clean = nk2.ecg_clean(ecg, sampling_rate=fs)
        ecg_signals, ecg_info = nk2.ecg_peaks(ecg_clean, method="neurokit", show=False)
        ecg_peaks = ecg_info["ECG_R_Peaks"]

        # obtain timings
        ecg_timings = ecg_peaks
        ppg_timings = get_ppg_onsets(ppg, ppg_peaks, fs)

        # extract pulse arrival times
        rel_ppg_timings = []
        for ecg_timing in ecg_timings:
            ppg_timings = np.asarray(ppg_timings)
            different = ppg_timings - ecg_timing
            different = np.where(different > 0, different, 100000)
            idx = different.argmin()
            rel_ppg_timings.append(ppg_timings[idx])

        pats = (rel_ppg_timings - ecg_timings) / fs

        # find median pulse arrival time for this subject
        curr_median_pat = median(pats)
        median_pats.append(curr_median_pat)

    return median_pats

def get_age(matching_recs):
    ages = []
    subj_nos = [i for i in range(len(matching_recs['name']))]
    for subj_no in subj_nos:
        # specify this subject's record name and directory:
        record_name = matching_recs['name'][subj_no]
        record_dir = matching_recs['dir'][subj_no]

        # extract this subject's age
        curr_age = matching_recs['age'][subj_no]
        ages.append(curr_age)
    return ages

分别比较走,跑,坐
在这里插入图片描述
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/658000.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Python从入门到进阶】24、urllib获取网站电影排行

接上篇《23、urllib使用post请求百度翻译》 上一篇我们讲解了如何使用urllib实现百度翻译的效果。本篇我们来讲解如何使用urllib抓取某某电影排行榜信息。 一、某某电影介绍 1、某某电影网站 某某电影成立于2005年&#xff0c;最初只是一个小型的电影社区&#xff0c;但随着…

【备战秋招】每日一题:2023.05-B卷-华为OD机试 - 报文重排序

为了更好的阅读体检&#xff0c;可以查看我的算法学习博客报文重排序 题目描述 对报文进行重传和重排序是常用的可靠性机制&#xff0c;重传缓中区内有一定数量的子报文&#xff0c;每个子报文在原始报文中的顺序已知&#xff0c;现在需要恢复出原始报文。 输入描述 输入第…

改进YOLOv8 | 优化器篇 | YOLOv8 引入谷歌 Lion 优化器

论文地址:https://arxiv.org/pdf/2302.06675.pdf 代码地址:https://github.com/google/automl/tree/master/lion 我们提出了一种将算法发现作为程序搜索的方法,并将其应用于发现用于深度神经网络训练的优化算法。我们利用高效的搜索技术来探索一个无限且稀疏的程序空间。为了…

【SCADA】测试用KingIOServer采集杰控OPC DA服务器数据

Hello&#xff0c;大家好&#xff0c;我是雷工&#xff01; 现场做数据采集时经常会遇到需要通过OPC采集数据的情况&#xff0c;本篇测试KingIOServer采集北京杰控组态软件的OPCDA服务器数据。 以下为测试记录过程。 一、KingIOServer的OPC DA数据采集介绍 KingIOServer可以作…

Vue中如何进行分布式日志管理与日志分析

Vue中如何进行分布式日志管理与日志分析 在现代应用程序中&#xff0c;日志是一项重要的功能&#xff0c;用于帮助开发人员和运维人员了解应用程序的行为并进行故障排除。随着应用程序的规模和复杂性的增加&#xff0c;日志管理和分析变得越来越困难。本文将介绍如何在Vue应用…

JSON对象花样进阶

JSON 对象使用在大括号{ }中书写。 对象可以包含多个 key/value&#xff08;键/值&#xff09;对。 key 必须是字符串&#xff0c;value 可以是合法的 JSON 数据类型&#xff08;字符串, 数字, 对象, 数组, 布尔值或 null&#xff09;。 key 和 value 中使用冒号(:)分割。 每个…

Postman接口自动化之postman脚本编写

这是之前搞的接口自动化方案&#xff0c;已经在业务测试中实现了使用postman编写接口脚本&#xff0c;通过GitHubJenkinsemail html report实现了接口自动化&#xff0c;现在分块整理一下。 postman脚本编写 1、创建集合 和 目录&#xff1a; 一条业务线下的接口可以放到一个…

ACL2023 | 大模型如何快速构建指令遵循数据集?self-instruct:用175条种子数据追上InstructGPT001效果

一、概述 title&#xff1a;SELF-INSTRUCT: Aligning Language Models with Self-Generated Instructions 论文地址&#xff1a;https://arxiv.org/abs/2212.10560 代码&#xff1a;GitHub - yizhongw/self-instruct: Aligning pretrained language models with instruction…

chatgpt赋能python:Python怎么输出Unicode值

Python怎么输出Unicode值 Python 是一种高级编程语言&#xff0c;因其简单易学和快速开发已成为许多开发者的首选。Python 可以输出多种数据类型&#xff0c;包括字符串和数字。 在许多情况下&#xff0c;输出 Unicode 值是必需的&#xff0c;本文将介绍在 Python 中如何输出 …

shardingsphere第二课-shardingsphere-jdbc的基本使用及各种分片策略

第一章介绍 一、ShardingJDBC客户端分库分表 ShardingSphere-JDBC 定位为轻量级 Java 框架&#xff0c;在 Java 的 JDBC 层提供的额外服务。 它使用客户端直连数据库&#xff0c;以 jar 包形式提供服务&#xff0c;无需额外部署和依赖&#xff0c;可理解为增强版的 JDBC 驱动…

chatgpt赋能python:Python中如何输入一个列表

Python中如何输入一个列表 输入一个列表是Python编程的基本任务之一。列表可以看做是一种序列&#xff0c;其中包含多个元素&#xff0c;用逗号隔开&#xff0c;并用方括号括起来。在Python中&#xff0c;列表是一种非常常见的数据类型&#xff0c;常用于存储和处理一系列相关…

断言操作符介绍

目录 1.延时操作符&#xff08;##&#xff09; 1.1 ##m 1.2 ##[m:n] 2.蕴含操作符&#xff08;|>,|->&#xff09; 2.1 |>操作符 2.2 |->操作符 3 重复操作符 ([*m][->m][m]) 3.1 连续重复操作符&#xff08;[*m][*m:n]&#xff09; 3.2 跟随重复操作…

Java选择题刷题记录5

Java堆栈 图片来自https://www.cnblogs.com/cici-new/p/14963762.html 数组、String都在堆里 枚举类 1.枚举类可以实现一个或多个接口&#xff0c;使用enum定义的枚举类默认继承java.lang.Enum类&#xff0c;而不是默认继承Object类&#xff0c;其中 java.lang.Enum类实现了…

机器视觉硬件的选择-标定板

康耐视智能相机Insight-缺陷检测 一>棋盘格的作用 a>畸变校正 径向畸变,径向畸变就是沿着透镜半径方向分布的畸变,产生原因是光线在原理透镜中心的地方比靠近中心的地方更加弯曲,这种畸变在短焦镜头中表现更加明显,径 向畸变主要包括桶形畸变和枕形畸变两种。以下分别…

chatgpt赋能python:Python输入π的方法及其应用

Python输入π的方法及其应用 Python是一门强大的编程语言&#xff0c;其支持的数学函数功能能够帮助用户完成各种复杂的计算操作。当我们需要在Python代码中使用π值时&#xff0c;不同的场景需要不同的处理方法。本文将详细介绍如何在Python中输入π值&#xff0c;并且探讨它…

Ansible 的脚本 --- playbook 剧本

playbook的相关知识 playbook 的简介 playbook是 一个不同于使用Ansible命令行执行方式的模式&#xff0c;其功能更强大灵活。 简单来说&#xff0c;playbook是一个非常简单的配置管理和多主机部署系统&#xff0c; 不同于任何已经存在的模式&#xff0c;可作为一个适合部署…

【C++】STL之string类(2)

个人主页&#xff1a;平行线也会相交&#x1f4aa; 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 平行线也会相交 原创 收录于专栏【C之路】&#x1f48c; 本专栏旨在记录C的学习路线&#xff0c;望对大家有所帮助&#x1f647;‍ 希望我们一起努力、成长&…

Vue中如何进行分布式任务调度与任务监控

Vue中如何进行分布式任务调度与任务监控 在复杂的系统中&#xff0c;如何有效地进行任务调度和监控是一个非常重要的问题。分布式系统中&#xff0c;任务调度和监控则更加复杂。Vue是一款流行的前端框架&#xff0c;本文将介绍如何在Vue中进行分布式任务调度和监控。 什么是分…

QML基础

从 Qt 4.7 开始&#xff0c;Qt 引入了一种声明式脚本语言&#xff0c;称为 QML&#xff08;Qt Meta Language 或者 Qt Modeling Language&#xff09;&#xff0c;作为 C 语言的一种替代。而 Qt Quick 就是使用 QML 构建的一套类库。 QML 是一种基于 JavaScript 的声明式语言。…

Qt下存储读写应用程序设置的三种方法

一、简介 List item 用户对应用程序经常有这样的要求&#xff1a;要求它能记住它的settings&#xff0c;比如窗口大小、位置和密码等等。有三种方法可以实现&#xff1a; 使用注册表&#xff1b;使用配置文件(.ini)&#xff1b;使用自定义文件(例如.txt)。 二、使用注册表 …