基于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

评论 (0)
  • 体积大小:14*11*2.6CM,电气参数:输入100V-240V/10A,输出16V24A。PCB 正面如下图。PCB 背面如下图。根据实际功能可以将PCB分成几部分:EMI滤波,PFC电路,LLC电路。EMI滤波区域,两级共模电感,LN各用了保险丝加压敏电阻,继电器(HF32FV-G)用来切除NTC的,为了提高效率点,如下图。PFC电路区域,如下图。LLC电路区域,如下图。详细分析一下该电源用的主要IC还有功率器件。AC侧采用了两颗整流桥进行并联,器件增加电流应力,如下图。共模电感都有放电针
    liweicheng 2025-05-10 20:03 42浏览
  • 在印度与巴基斯坦的军事对峙情境下,歼10C的出色表现如同一颗投入平静湖面的巨石,激起层层涟漪,深刻印证了“质量大于数量”这一铁律。军事领域,技术优势就是决定胜负的关键钥匙。歼10C凭借先进的航电系统、强大的武器挂载能力以及卓越的机动性能,在战场上大放异彩。它能够精准捕捉目标,迅速发动攻击,以一敌多却毫不逊色。与之形成鲜明对比的是,单纯依靠数量堆砌的军事力量,在面对先进技术装备时,往往显得力不从心。这一现象绝非局限于军事范畴,在当今社会的各个领域,“质量大于数量”都已成为不可逆转的趋势。在科技行业
    curton 2025-05-11 19:09 180浏览
  • 行车记录仪是长这个样子的,如下图。从前面拆去玻璃挡板,可以清晰的看见里面的部件,5个按键电路板,液晶显示屏,摄像头,喇叭,电池包,还有一块主电路板。液晶显示屏正面,如下图。液晶显示屏背面,如下图。喇叭,如下图。5个按键的电路板,MENU,DOWN,POWER,UP,OK总共5个按键功能,导线连接到主电路板上,如下图。电池包,303040聚合物锂电池,3.7V,300mAH,如下图。如下图。摄像头,如下图。拿去摄像头外壳,如下图。分离广角聚集镜头和PCB板,如下图。广角聚焦镜头,具体结构如下图。P
    liweicheng 2025-05-09 22:50 33浏览
  • 【拆解】+自动喷香机拆解 家里之前买了从PDD买了一个小型自动喷香机放在厕所里。来增加家里的温馨感,这东西看着确实小巧,精致。可是这东西吧,耗电就是快,没过几天就没电了。今个就让我拆开看看什么在捣鬼。如下是产品的实物和宣传图: 由于螺丝孔太小和限位很深。对于我的螺丝刀套装没用。只能使用那种螺丝刀细头,同时又长的小螺丝刀进行拆解 拧下三颗螺丝钉,用一字螺丝刀撬开外壳,内部结构就呈现在眼前。 内部构造相当简单,部件没多少。就是锂电池供电,通过MCU实现按键控制,段码屏控制,LE
    zhusx123 2025-05-10 19:55 56浏览
  • ‌磁光克尔效应(Magneto-Optic Kerr Effect, MOKE)‌ 是指当线偏振光入射到磁性材料表面并反射后,其偏振状态(偏振面旋转角度和椭偏率)因材料的磁化强度或方向发生改变的现象。具体表现为:1、‌偏振面旋转‌:反射光的偏振方向相对于入射光发生偏转(克尔旋转角 θK)。2、‌椭偏率变化‌:反射光由线偏振变为椭圆偏振(克尔椭偏率 εK)。这一效应直接关联材料的磁化状态,是表征磁性材料(如铁磁体、反铁磁体)磁学性质的重要非接触式光学探测手段,广泛用于
    锦正茂科技 2025-05-12 11:02 100浏览
  • 【拆解】+CamFi卡菲单反无线传输器拆解 对于单反爱好者,想要通过远程控制自拍怎么办呢。一个远程连接,远程控制相机拍摄的工具再合适不过了。今天给大伙介绍的是CamFi卡菲单反无线传输器。 CamFi 是专为数码单反相机打造的无线传输控制器,自带的 WiFi 功能(无需手机流量),不但可通过手机、平板、电脑等设备远程连接操作单反相机进行拍摄,而且还可实时传输相机拍摄的照片到 iPad 和电视等大屏设备进行查看和分享。 CamFi 支持大部分佳能和尼康单反相机,内置可充电锂离子电池,无需相机供电。
    zhusx123 2025-05-11 14:14 94浏览
  • 在 AI 浪潮席卷下,厨电行业正经历着深刻变革。AWE 2025期间,万得厨对外首次发布了wan AiOS 1.0组织体超智能系统——通过AI技术能够帮助全球家庭实现从健康检测、膳食推荐,到食材即时配送,再到一步烹饪、营养总结的个性化健康膳食管理。这一创新之举并非偶然的个案,而是整个厨电行业大步迈向智能化、数字化转型浪潮的一个关键注脚,折射出全行业对 AI 赋能的热切渴求。前有标兵后有追兵,万得厨面临着高昂的研发成本与技术迭代压力,稍有懈怠便可能被后来者赶
    用户1742991715177 2025-05-11 22:44 73浏览
  • 1.概述MYD-YG2LX采用瑞萨RZ/G2L作为核心处理器,该处理器搭载双核Cortex-A55@1.2GHz+Cortex-M33@200MHz处理器,其内部集成高性能3D加速引擎Mail-G31 GPU(500MHz)和视频处理单元(支持H.264硬件编解码),16位的DDR4-1600 / DDR3L-1333内存控制器、千兆以太网控制器、USB、CAN、SD卡、MIPI-CSI等外设接口,在工业、医疗、电力等行业都得到广泛的应用。米尔基于瑞萨RZ/G2L开发板本文主要介绍基于MYD-Y
    米尔电子嵌入式 2025-05-09 17:38 25浏览
  • 蓝牙耳机是长这个样子,如下图。背部图,如下图。拆开L耳的一侧,有NFC和电池包(501230 3.7V 150mAh)如下图。电池包(501230 3.7V 150mAh)如下图。NFC正面,如下图。NFC背面,如下图。如何理解NFC的工作原理呢,搜集一下相关的资料,如下图。拆开R耳的一侧,PCB正面,如下图。PCB背面,如下图。有两组红黑的线,一组连接到了喇叭,另一组连接到了MIC头上,MIC头参数如下图。蓝牙模块(CSR 8635),有蛇形PCB走线做成天线,节约了天线成本,如下图。该IC介
    liweicheng 2025-05-10 00:45 39浏览
  •   定制软件开发公司推荐清单   在企业数字化转型加速的2025年,定制软件开发需求愈发多元复杂。不同行业、技术偏好与服务模式的企业,对开发公司的要求大相径庭。以下从技术赛道、服务模式及行业场景出发,为您提供适配的定制软件开发公司推荐及选择建议。   华盛恒辉科技有限公司:是一家专注于高端软件定制开发服务和高端建设的服务机构,致力于为企业提供全面、系统的开发制作方案。在部队政企开发、建设到运营推广领域拥有丰富经验,在教育,工业,医疗,APP,管理,商城,人工智能,部队软件、工业软件、数字化转
    华盛恒辉l58ll334744 2025-05-12 15:55 109浏览
  • 在工业自动化领域中,PLC(可编程逻辑控制器)和 DCS(分布式控制系统)是两种最为常见的控制技术。它们凭借着高可靠性、高灵活性与高自动化程度等显著优势,在工业自动化行业中发挥着不可替代的作用,并已被广泛应用于机械臂自动装配、发电机功率调节、石油炼制、化工生产、交通信号控制与地铁轻轨控制等众多工控场景之中。一种典型的现代工业总线控制系统而数字隔离器,作为工业自动化领域中低压控制系统与高压设备进行信息传输的“安全桥梁”,其不仅能有效阻断高压电气向低压控制系统的传导路径,保障操作人员与控制系统的安全
    华普微HOPERF 2025-05-09 17:08 23浏览
  • 文/Leon编辑/cc孙聪颖‍在新能源汽车赛道的残酷洗牌中,威马、爱驰等数十个品牌黯然退场,极越、哪吒汽车也深陷经营困局,“跨界造车” 早已褪去曾经的光环,成为吞噬企业资金与精力的风险泥潭,尤其对上市公司而言,稍有不慎便会被拖入业绩泥沼。当行业共识已清晰显现 —— 新能源汽车市场这片红海正上演着惨烈的生存之战,石头科技创始人昌敬却逆势入局,掌舵极石汽车,其押注造车的抉择,正让本就面临挑战的石头科技主业雪上加霜。2025 年 4 月中旬,昌敬突然清空微博、抖音等社交媒体账号的举动,迅速引爆舆论场。
    华尔街科技眼 2025-05-09 20:53 30浏览
  •         信创产业含义的“信息技术应用创新”一词,最早公开信息见于2019年3月26日,在江苏南京召开的信息技术应用创新研讨会。本次大会主办单位为江苏省工业和信息化厅和中国电子工业标准化技术协会安全可靠工作委员会。        2019年5月16日,美国将华为列入实体清单,在未获得美国商务部许可的情况下,美国企业将无法向华为供应产品。       2019年6
    天涯书生 2025-05-11 10:41 128浏览
  •   基于 2025 年行业权威性与时效性,以下梳理国内知名软件定制开发企业,涵盖综合型、垂直领域及特色技术服务商:   华盛恒辉科技有限公司:是一家专注于高端软件定制开发服务和高端建设的服务机构,致力于为企业提供全面、系统的开发制作方案。在部队政企开发、建设到运营推广领域拥有丰富经验,在教育,工业,医疗,APP,管理,商城,人工智能,部队软件、工业软件、数字化转型、新能源软件、光伏软件、汽车软件,ERP,系统二次开发,CRM等领域有很多成功案例。   五木恒润科技有限公司:是一家专业的部队信
    华盛恒辉l58ll334744 2025-05-12 16:13 77浏览
  • 递交招股书近一年后,曹操出行 IPO 进程终于迎来关键节点。从 2024 年 4 月首次递表,到 2025 年 4 月顺利通过中国证监会境外发行上市备案,并迅速更新招股书。而通过上市备案也标志着其赴港IPO进程进入实质性推进阶段,曹操出行最快有望于2025年内完成港股上市,成为李书福商业版图中又一关键落子。行路至此,曹操出行面临的挑战依然不容忽视。当下的网约车赛道,早已不是当年群雄逐鹿的草莽时代,市场渐趋饱和,竞争近乎白热化。曹操出行此时冲刺上市,既是背水一战,也是谋篇布局。其招股书中披露的资金
    用户1742991715177 2025-05-10 21:18 57浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦