基于speex的音频处理系列之一-speex在pc上的移植与回声消除测试

原创 嵌入式Lee 2025-01-12 08:30

.前言


注:完整代码见后面总结部分git地址

最近在做音频相关的内容,接触到音频对讲中的一个需求:回声消除。所谓的回声消除即对应以下模型, 在对讲过程中远端的讲话通过一定方式传输到近端,在近端通过喇叭播放,

这个喇叭播放的声音以及其环境的各种反射,加上近端的语音(包括噪声等)被近端的麦克风采集传送到远端,这样远端就会听到对方的讲话声叠加了自己的讲话声。远端听到自己的讲话声反射回来了,需要消除这部分。这个工作是在近端完成的,近端已知的是A处即远端传过来的语音,以及B处麦克风采集的叠加数据,实际上简单的来说就是要从B中减去A的数据。但是这个减是没办法直接减的,A处的原始数据播放到B处采集,实际上有一个映射关系,我们一般要求其是线性的,这个减法需要一定的算法模型去实现,这就是回声消除算法。

由于A处的原始数据和B处的麦克风采集可能软件上没法比较好的同步,并且A处原始数据经过PA到喇叭播出来之后的效果存在一定映射关系,这个对应线性度不好的话影响也较大,所以有时候硬件上可以在PA之后做一路回声采集(理想是应该要能采集喇叭播放之后的效果但是做不到,只能采集PA之后的),这个采集和B处的麦克风采集使用同一个声卡采集这样可以保持同步,同时也可以减少PA过程的影响。

在找回声消除算法方案时,正好就找到了一个开源的实现speex,见其官网https://www.speex.org/。可以浏览下其官网了解一些背景知识,尤其是其文档可以下载下来先好好看看。

我们这一篇就来分享下speexpc上的移植,实现一个回声消除测试的pc端的小程序,一方面先体验一下speex,一方面也可以作为后面调试验证的工具,可以dump数据先在pc端进行仿真测试验证,再移植到具体的平台上去。

二. PC环境移植speex

我这里使用WSL+Ubuntu

2.1下载源码

git clone https://gitlab.xiph.org/xiph/speexdsp.git

cd speexdsp

./autogen.sh 生成各种工程,config.h.in等。

2.2添加文件

添加以下文件

speexdsp\include\speex下所有h文件

speexdsp_config_types.h.in改为

speexdsp_config_types.h

speexdsp\libspeexdsp下所有hc文件

test开头的不需要添加,可以单独放一个文件夹,在写自己的应用时参考。

config.h.in改为config.h

最终文件如下

lhj@lhj:~/speex/speexdsptree ..|-- config.h|-- include|   `-- speex|       |-- speex_buffer.h|       |-- speex_echo.h|       |-- speex_jitter.h|       |-- speex_preprocess.h|       |-- speex_resampler.h|       |-- speexdsp_config_types.h|       `-- speexdsp_types.h|-- libspeexdsp|   |-- _kiss_fft_guts.h|   |-- arch.h|   |-- bfin.h|   |-- buffer.c|   |-- fftwrap.c|   |-- fftwrap.h|   |-- filterbank.c|   |-- filterbank.h|   |-- fixed_arm4.h|   |-- fixed_arm5e.h|   |-- fixed_bfin.h|   |-- fixed_debug.h|   |-- fixed_generic.h|   |-- jitter.c|   |-- kiss_fft.c|   |-- kiss_fft.h|   |-- kiss_fftr.c|   |-- kiss_fftr.h|   |-- math_approx.h|   |-- mdf.c|   |-- misc_bfin.h|   |-- os_support.h|   |-- preprocess.c|   |-- pseudofloat.h|   |-- resample.c|   |-- resample_neon.h|   |-- resample_sse.h|   |-- scal.c|   |-- smallft.c|   |-- smallft.h|   `-- vorbis_psy.h`-- test    |-- testdenoise.c    |-- testecho.c    |-- testjitter.c    |-- testresample.c    `-- testresample2.c 4 directories, 44 files

对应的还要配置或者指定相应的头文件包含路径。

2.3配置speexdsp_config_types.h

该文件定义一些基本的数据类型

编译器支持则直接包含

定义对应类型

否则按照具体平台定义

#ifndef __SPEEX_TYPES_H__#define __SPEEX_TYPES_H__ #include  typedef int16_t spx_int16_t;typedef uint16_t spx_uint16_t;typedef int32_t spx_int32_t;typedef uint32_t spx_uint32_t; #endif

2.4 配置config.h

工程配置宏

HAVE_CONFIG_H

比如gcc中使用-DHAVE_CONFIG_H选项。

定义该宏后,每个c文件都会包含config.h文件,通过config.h进行相应的配置。

2.4.1EXPORT

config.h

#undef EXPORT

改为

#define EXPORT

2.4.2配置使用浮点还是定点

config.h

如果支持浮点则

#undef FLOATING_POINT

改为

#define FLOATING_POINT

如果使用定点则

#undef FIXED_POINT

改为

#define FIXED_POINT

我这里使用浮点

2.4.3指定FFT实现

内部有fft实现

kiss_fft.c

kiss_fftr.c

我们先使用内部实现,硬件支持fft的后面再改为硬件实现。

#undef USE_KISS_FFT

改为

#define USE_KISS_FFT

还可以使用

USE_SMALLFT

以下依赖具体平台

USE_GPL_FFTW3

USE_INTEL_IPP

USE_INTEL_MKL

2.4.4 math依赖

speexdsp/libspeexdsp/math_approx.h中依赖

sqrt

acos

exp

atan

fabs

log

floor

等接口

这里默认直接使用标准库

链接时-lm指定链接数学库即可。

嵌入式平台则根据自己平台实现对应的接口即可。

2.4.5Os依赖

speexdsp/libspeexdsp/os_support.h中相关的动态内存管理接口,内存拷贝等接口,打印等接口。

Pc直接使用标准库即可。嵌入式平台则需要根据自己平台去实现对应接口。

2.5编译

我这里是基于WSLLinux环境。

新建build.sh文件

添加如下内容

#! /bin/sh gcc libspeexdsp/*.c speexecho.c -Iinclude -I. -DHAVE_CONFIG_H -lm -o speexecho

其中speexecho.c是用户自己的应用代码。

chmod +x build.sh 

编译./build.sh

.实现回声消除测试程序

我们基于testecho.c,该demo读原始spk(echo)mic的原始数据流,输出回声消除后的数据。

我们这里在原来的基础上修改下,可以读wav文件并且输出也是wav文件方便后面直接播放。假设我们这里自己的平台也可以dumpmic和当前spkecho数据为wav格式。

我这里有个抓取的mic3.wavspk3.wav的测试数据。

我们这里假设wav文件都是单通道,16位。

实现代码如下

#ifdef HAVE_CONFIG_H#include "config.h"#endif#include "speex/speex_echo.h"#include "speex/speex_preprocess.h"#include <stdio.h>#include <stdlib.h>#include <string.h>/* WAV解析 */#define CHUNK_RIFF "RIFF"#define CHUNK_WAVE "WAVE"#define CHUNK_FMT "fmt "#define CHUNK_DATA "data"typedef struct{    uint32_t off;    uint32_t chunksize;    uint16_t audioformat;    uint16_t numchannels;    uint32_t samplerate;    uint32_t byterate;    uint16_t blockalign;    uint16_t bitspersample;    uint32_t datasize;}wav_t;static int wav_decode_head(uint8_t* buffer, wav_t* wav){    uint8_t* p = buffer;    uint32_t chunksize;    uint32_t subchunksize;    if(0 != memcmp(p,CHUNK_RIFF,4))    {        return -1;    }    p += 4;    chunksize = (uint32_t)p[0| ((uint32_t)p[1]<<8| ((uint32_t)p[2]<<16| ((uint32_t)p[3]<<24);    wav->chunksize = chunksize;    p += 4;    if(0 != memcmp(p,CHUNK_WAVE,4))    {        return -2;    }    p += 4;    do    {        if(0 == memcmp(p,CHUNK_FMT,4))        {            p += 4;            subchunksize = (uint32_t)p[0| ((uint32_t)p[1]<<8| ((uint32_t)p[2]<<16| ((uint32_t)p[3]<<24);            p += 4;            /* 解析参数 */            wav->audioformat = (uint16_t)p[0| ((uint16_t)p[1]<<8);            if((wav->audioformat == 0x0001|| (wav->audioformat == 0xFFFE))            {                p += 2;                wav->numchannels = (uint16_t)p[0| ((uint16_t)p[1]<<8);                p += 2;                wav->samplerate = (uint32_t)p[0| ((uint32_t)p[1]<<8| ((uint32_t)p[2]<<16| ((uint32_t)p[3]<<24);                p += 4;                wav->byterate = (uint32_t)p[0| ((uint32_t)p[1]<<8| ((uint32_t)p[2]<<16| ((uint32_t)p[3]<<24);                p += 4;                wav->blockalign = (uint16_t)p[0| ((uint16_t)p[1]<<8);                p += 2;                wav ->bitspersample = (uint16_t)p[0| ((uint16_t)p[1]<<8);                p += 2;                if(subchunksize >16)                {                    /* 有ext区域 */                    uint16_t cbsize = (uint16_t)p[0| ((uint16_t)p[1]<<8);                    p += 2;                    if(cbsize > 0)                    {                        /* ext数据 2字节有效bits wValidBitsPerSample ,4字节dwChannelMask 16字节SubFormat */                        p += 2;                        p += 4;                        /* 比对subformat */                        p += 16;                           }                }            }            else            {                p += subchunksize;            }        }        else if(0 == memcmp(p,CHUNK_DATA,4))        {            p += 4;            subchunksize = (uint32_t)p[0| ((uint32_t)p[1]<<8| ((uint32_t)p[2]<<16| ((uint32_t)p[3]<<24);            wav->datasize = subchunksize;            p += 4;            wav->off = (uint32_t)(p- buffer);            return 0;        }        else        {            p += 4;            subchunksize = (uint32_t)p[0| ((uint32_t)p[1]<<8| ((uint32_t)p[2]<<16| ((uint32_t)p[3]<<24);            p += 4;            p += subchunksize;        }    }while((uint32_t)(p - buffer) < (chunksize + 8));    return -3;}/* 填充44字节的wav头 */static void wav_fill_head(uint8_t* buffer, int samples, int chnum, int freq){    /*     * 添加wav头信息     */    uint32_t chunksize = 44-8+samples*chnum*16/8;    uint8_t* p = (uint8_t*)buffer;    uint32_t bps = freq*chnum*16/8;    uint32_t datalen = samples*chnum*16/8;    p[0] = 'R';    p[1] = 'I';    p[2] = 'F';    p[3] = 'F';    p[4] = chunksize & 0xFF;    p[5] = (chunksize>>8) & 0xFF;    p[6] = (chunksize>>16) & 0xFF;    p[7] = (chunksize>>24) & 0xFF;    p[8] = 'W';    p[9] = 'A';    p[10] = 'V';    p[11] = 'E';    p[12] = 'f';    p[13] = 'm';    p[14] = 't';    p[15] = ' ';    p[16] = 16;  /* Subchunk1Size */    p[17= 0;    p[18= 0;    p[19= 0;    p[20= 1;  /* PCM */    p[21= 0;    p[22= chnum; /* 通道数 */    p[23= 0;    p[24= freq & 0xFF;    p[25= (freq>>8& 0xFF;    p[26= (freq>>16& 0xFF;    p[27= (freq>>24& 0xFF    p[28= bps & 0xFF;      /* ByteRate */    p[29= (bps>>8& 0xFF;    p[30= (bps>>16& 0xFF;    p[31= (bps>>24& 0xFF    p[32= chnum*16/8; /* BlockAlign */    p[33= 0;    p[34= 16;  /* BitsPerSample */    p[35= 0;    p[36= 'd';    p[37= 'a';    p[38= 't';    p[39= 'a';    p[40= datalen & 0xFF;    p[41= (datalen>>8& 0xFF;    p[42= (datalen>>16& 0xFF;    p[43= (datalen>>24& 0xFF}void wav_print(wav_t* wav){   printf("off:%d\r\n",wav->off);    printf("chunksize:%d\r\n",wav->chunksize);    printf("audioformat:%d\r\n",wav->audioformat);    printf("numchannels:%d\r\n",wav->numchannels);    printf("samplerate:%d\r\n",wav->samplerate);    printf("byterate:%d\r\n",wav->byterate);    printf("blockalign:%d\r\n",wav->blockalign);    printf("bitspersample:%d\r\n",wav->bitspersample);    printf("datasize:%d\r\n",wav->datasize); }#define NN 128#define TAIL 1024int main(int argc, char **argv){   FILE *spk_fd, *mic_fd, *out_fd;      short spk_buf[NN], mic_buf[NN], out_buf[NN];   uint8_t spk_wav_buf[44]; /* 输入spk wav文件头缓存 */   uint8_t mic_wav_buf[44]; /* 输入mic wav文件头缓存 */   uint8_t out_wav_buf[44]; /* 输出文件wav头缓存 */   wav_t spk_wav;   wav_t mic_wav;   int samps;  /* 采样点数 */   int times;    /* 读取次数 */   SpeexEchoState *st;   SpeexPreprocessState *den;   int sampleRate;   char* mic_fname = argv[1];   char* spk_fname = argv[2];   char* out_fname = argv[3];   if (argc != 4)   {      fprintf(stderr, "testecho mic.wav spk.wav out.wav\n");      exit(1);   }   spk_fd = fopen(spk_fname, "rb");   if(spk_fd < 0){      fprintf(stderr, "open file %s err\n",spk_fname);      exit(1);   }   mic_fd  = fopen(mic_fname,  "rb");   if(mic_fd < 0){      fprintf(stderr, "open file %s err\n",mic_fname);      fclose(spk_fd);      exit(1);   }   out_fd    = fopen(out_fname, "wb");   if(out_fd < 0){      fprintf(stderr, "open file %s err\n",out_fname);      fclose(spk_fd);      fclose(mic_fd);      exit(1);   }   if(44 != fread(mic_wav_buf, 144, mic_fd)){      fprintf(stderr, "read file %s err\n",mic_fname);      fclose(spk_fd);      fclose(mic_fd);      fclose(out_fd);      exit(1);   }   if(44 != fread(spk_wav_buf, 144, spk_fd)){      fprintf(stderr, "read file %s err\n",spk_fname);      fclose(spk_fd);      fclose(mic_fd);      fclose(out_fd);      exit(1);    }   if(0 != wav_decode_head(spk_wav_buf, &spk_wav)){      fprintf(stderr, "decode file %s err\n",spk_fname);      fclose(spk_fd);      fclose(mic_fd);      fclose(out_fd);      exit(1);    }   printf("[spk_wav]\r\n");   wav_print(&spk_wav);   if(0 != wav_decode_head(mic_wav_buf, &mic_wav)){      fprintf(stderr, "decode file %s err\n",mic_fname);      fclose(spk_fd);      fclose(mic_fd);      fclose(out_fd);      exit(1);     }   printf("[mic_wav]\r\n");   wav_print(&mic_wav);   samps = spk_wav.datasize > mic_wav.datasize ? mic_wav.datasize : spk_wav.datasize; /* 获取较小的数据大小 */   samps /= spk_wav.blockalign;  /* 采样点数 =  数据大小 除以 blockalign */   printf("\r\nsamps:%d\r\n",samps);   sampleRate = spk_wav.samplerate;   wav_fill_head(out_wav_buf, samps, 1, sampleRate);  /* 输出文件头 */   if(44 != fwrite(out_wav_buf, 144, out_fd)){      fprintf(stderr, "write file %s err\n",out_fname);      fclose(spk_fd);      fclose(mic_fd);      fclose(out_fd);      exit(1);   }   st = speex_echo_state_init(NNTAIL);   den = speex_preprocess_state_init(NN, sampleRate);   speex_echo_ctl(st, SPEEX_ECHO_SET_SAMPLING_RATE&sampleRate);   speex_preprocess_ctl(den, SPEEX_PREPROCESS_SET_ECHO_STATE, st);   times = samps / NN;   /* 一次读取NN个点,读取times次 */   for(int i=0; i<times; i++)   {      if(NN != fread(mic_buf, sizeof(short), NN, mic_fd)){        fprintf(stderr, "read file %s err\n",mic_fname);        fclose(spk_fd);        fclose(mic_fd);        fclose(out_fd);        exit(1);      }      if(NN != fread(spk_buf, sizeof(short), NN, spk_fd)){        fprintf(stderr, "read file %s err\n",spk_fname);        fclose(spk_fd);        fclose(mic_fd);        fclose(out_fd);        exit(1);      }      speex_echo_cancellation(st, mic_buf, spk_buf, out_buf);      speex_preprocess_run(den, out_buf);      if(NN != fwrite(out_buf, sizeof(short), NN, out_fd)){        fprintf(stderr, "write file %s err\n",out_fname);        fclose(spk_fd);        fclose(mic_fd);        fclose(out_fd);        exit(1);      }   }   speex_echo_state_destroy(st);   speex_preprocess_state_destroy(den);   fclose(out_fd);   fclose(spk_fd);   fclose(mic_fd);   return 0;}

测试

./speexecho mic3.wav spk3.wav out3.wav

可以看到out3.wav相对与mic3.wav消除掉了spk3.wav的部分,但是还是有残留没有消除干净,后面再来优化。

.总结

以上分享了speexpc上的移植使用,实现了一个简单的回声消除的测试程序,先暂时体验一下,后面再来学习了解speex的细节,以及调试,和在平台上的移植等。

完整代码见
https://github.com/qinyunti/speex_test

评论
  • 职场是人生的重要战场,既是谋生之地,也是实现个人价值的平台。然而,有些思维方式却会悄无声息地拖住你的后腿,让你原地踏步甚至退步。今天,我们就来聊聊职场中最忌讳的五种思维方式,看看自己有没有中招。1. 固步自封的思维在职场中,最可怕的事情莫过于自满于现状,拒绝学习和改变。世界在不断变化,行业的趋势、技术的革新都在要求我们与时俱进。如果你总觉得自己的方法最优,或者害怕尝试新事物,那就很容易被淘汰。与其等待机会找上门,不如主动出击,保持学习和探索的心态。加入优思学院,可以帮助你快速提升自己,与行业前沿
    优思学院 2025-01-09 15:48 138浏览
  • 在当前人工智能(AI)与物联网(IoT)的快速发展趋势下,各行各业的数字转型与自动化进程正以惊人的速度持续进行。如今企业在设计与营运技术系统时所面临的挑战不仅是技术本身,更包含硬件设施、第三方软件及配件等复杂的外部因素。然而这些系统往往讲究更精密的设计与高稳定性,哪怕是任何一个小小的问题,都可能对整体业务运作造成严重影响。 POS应用环境与客户需求以本次分享的客户个案为例,该客户是一家全球领先的信息技术服务与数字解决方案提供商,遭遇到一个由他们所开发的POS机(Point of Sal
    百佳泰测试实验室 2025-01-09 17:35 155浏览
  • 一个真正的质量工程师(QE)必须将一件产品设计的“意图”与系统的可制造性、可服务性以及资源在现实中实现设计和产品的能力结合起来。所以,可以说,这确实是一种工程学科。我们常开玩笑说,质量工程师是工程领域里的「侦探」、「警察」或「律师」,守护神是"墨菲”,信奉的哲学就是「墨菲定律」。(注:墨菲定律是一种启发性原则,常被表述为:任何可能出错的事情最终都会出错。)做质量工程师的,有时会不受欢迎,也会被忽视,甚至可能遭遇主动或被动的阻碍,而一旦出了问题,责任往往就落在质量工程师的头上。虽然质量工程师并不负
    优思学院 2025-01-09 11:48 156浏览
  • Snyk 是一家为开发人员提供安全平台的公司,致力于协助他们构建安全的应用程序,并为安全团队提供应对数字世界挑战的工具。以下为 Snyk 如何通过 CircleCI 实现其“交付”使命的案例分析。一、Snyk 的挑战随着客户对安全工具需求的不断增长,Snyk 的开发团队面临多重挑战:加速交付的需求:Snyk 的核心目标是为开发者提供更快、更可靠的安全解决方案,但他们的现有 CI/CD 工具(TravisCI)运行缓慢,无法满足快速开发和部署的要求。扩展能力不足:随着团队规模和代码库的不断扩大,S
    艾体宝IT 2025-01-10 15:52 114浏览
  • 1月9日,在2025国际消费电子展览会(CES)期间,广和通发布集智能语音交互及翻译、4G/5G全球漫游、随身热点、智能娱乐、充电续航等功能于一体的AI Buddy(AI陪伴)产品及解决方案,创新AI智能终端新品类。AI Buddy是一款信用卡尺寸的掌中轻薄智能设备,为用户带来实时翻译、个性化AI语音交互助手、AI影像识别、多模型账户服务、漫游资费服务、快速入网注册等高品质体验。为丰富用户视觉、听觉的智能化体验,AI Buddy通过蓝牙、Wi-Fi可配套OWS耳机、智能眼镜、智能音箱、智能手环遥
    物吾悟小通 2025-01-09 18:21 133浏览
  • HDMI 2.2 规格将至,开启视听新境界2025年1月6日,HDMI Forum, Inc. 宣布即将发布HDMI规范2.2版本。新HDMI规范为规模庞大的 HDMI 生态系统带来更多选择,为创建、分发和体验理想的终端用户效果提供更先进的解决方案。新技术为电视、电影和游戏工作室等内容制作商在当前和未来提供更高质量的选择,同时实现多种分发平台。96Gbps的更高带宽和新一代 HDMI 固定比率速率传输(Fixed Rate Link)技术为各种设备应用提供更优质的音频和视频。终端用户显示器能以最
    百佳泰测试实验室 2025-01-09 17:33 172浏览
  • 在过去十年中,自动驾驶和高级驾驶辅助系统(AD/ADAS)软件与硬件的快速发展对多传感器数据采集的设计需求提出了更高的要求。然而,目前仍缺乏能够高质量集成多传感器数据采集的解决方案。康谋ADTF正是应运而生,它提供了一个广受认可和广泛引用的软件框架,包含模块化的标准化应用程序和工具,旨在为ADAS功能的开发提供一站式体验。一、ADTF的关键之处!无论是奥迪、大众、宝马还是梅赛德斯-奔驰:他们都依赖我们不断发展的ADTF来开发智能驾驶辅助解决方案,直至实现自动驾驶的目标。从新功能的最初构思到批量生
    康谋 2025-01-09 10:04 132浏览
  • 车机导航有看没有懂?智能汽车语系在地化不可轻忽!随着智能汽车市场全球化的蓬勃发展,近年来不同国家地区的「Automotive Localization」(汽车在地化)布局成为兵家必争之地,同时也是车厂在各国当地市场非常关键的营销利器。汽车在地化过程中举足轻重的「汽车语系在地化」,则是透过智能汽车产品文字与服务内容的设计订制,以对应不同国家地区用户的使用习惯偏好,除了让当地车主更能清楚理解车辆功能,也能进一步提高品牌满意度。客户问题与难处某车厂客户预计在台湾市场推出新一代车款,却由于车机导航开发人
    百佳泰测试实验室 2025-01-09 17:47 95浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球中空长航时无人机产值达到9009百万美元,2024-2030年期间年复合增长率CAGR为8.0%。 环洋市场咨询机构出版了的【全球中空长航时无人机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球中空长航时无人机总体规模,包括产量、产值、消费量、主要生产地区、主要生产商及市场份额,同时分析中空长航时无人机市场主要驱动因素、阻碍因素、市场机遇、挑战、新产品发布等。报告从中空长航时
    GIRtina 2025-01-09 10:35 143浏览
  • 光伏逆变器是一种高效的能量转换设备,它能够将光伏太阳能板(PV)产生的不稳定的直流电压转换成与市电频率同步的交流电。这种转换后的电能不仅可以回馈至商用输电网络,还能供独立电网系统使用。光伏逆变器在商业光伏储能电站和家庭独立储能系统等应用领域中得到了广泛的应用。光耦合器,以其高速信号传输、出色的共模抑制比以及单向信号传输和光电隔离的特性,在光伏逆变器中扮演着至关重要的角色。它确保了系统的安全隔离、干扰的有效隔离以及通信信号的精准传输。光耦合器的使用不仅提高了系统的稳定性和安全性,而且由于其低功耗的
    晶台光耦 2025-01-09 09:58 109浏览
  • 在智能网联汽车中,各种通信技术如2G/3G/4G/5G、GNSS(全球导航卫星系统)、V2X(车联网通信)等在行业内被广泛使用。这些技术让汽车能够实现紧急呼叫、在线娱乐、导航等多种功能。EMC测试就是为了确保在复杂电磁环境下,汽车的通信系统仍然可以正常工作,保护驾乘者的安全。参考《QCT-基于LTE-V2X直连通信的车载信息交互系统技术要求及试验方法-1》标准10.5电磁兼容试验方法,下面将会从整车功能层面为大家解读V2X整车电磁兼容试验的过程。测试过程揭秘1. 设备准备为了进行电磁兼容试验,技
    北汇信息 2025-01-09 11:24 135浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦