欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android實現(xiàn)錄音靜音降噪

 更新時間:2022年08月09日 14:49:35   作者:suwen8100  
這篇文章主要為大家詳細介紹了Android實現(xiàn)錄音靜音降噪,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文實例為大家分享了Android實現(xiàn)錄音靜音降噪的具體代碼,供大家參考,具體內(nèi)容如下

需求:

客戶反饋產(chǎn)品的錄音里面很多雜音(因為我們把Codec的錄音增益調(diào)至最大,且電路上沒有專用的音頻處理芯片、CPU直接接MIC(有包地))。在外殼、硬件不能修改的情況下,軟件得想想辦法嘗試解決問題。

首先想到的是雙麥降噪,原理大概是:一個主麥克風用來做通話,另一個收集環(huán)境噪音,對音頻波形分析和相位操作,疊加到主麥克風的采樣波形上,形成相位抵消,就降噪了。缺點是,兩個麥克風不能距離太近,并且兩個麥克風距離說話人的距離不能太遠,太遠了角度就很小了,根本無法分辨出來,另外,根據(jù)產(chǎn)品使用情況,上下麥克風各自都有幾率稱為主麥克風。所以實驗測試出來的結(jié)果并沒有很好。

考慮到錄音噪音在有“人聲”的時候是分辨不出來的、或者說影響很小,而在靜音時有明顯的環(huán)境噪聲,因此想使用靜音降噪的方法來規(guī)避問題。

本文只是簡單的靜音降噪,原理如下:考慮到啟動錄音時,要等待一段時間(比如0.5s)才會有人聲,可根據(jù)這0.5s時間來預測噪聲的大?。ㄩ撝担缓笠源藶榛A來檢測“人聲”的起始點。在人聲到來前,把所有音頻數(shù)據(jù)設置為0,也就是做靜音處理,所以這里叫靜音降噪。而人聲到來時,返回實際的音頻數(shù)據(jù)(包括里面的噪聲數(shù)據(jù))。計算閾值的方法只是簡單的求和平均。

下面代碼在RK平臺上hardware/alsa_sound/AudioStreamInALSA.cpp實現(xiàn)。

#define MUTE_NOISE_REDUCTION
#ifdef MUTE_NOISE_REDUCTION
bool enable_reduction_noise = false;?? ?//由屬性sys.is.audiorecord.only控制

int threshold_def = 0x400;?? ?//默認閾值
int threshold = 0;?? ?//自適應噪聲閾值
int threshold_count = 0;?? ?//計數(shù),超過THRESHOLD_COUNT則使用threshold來檢測“人聲”
#define THRESHOLD_COUNT 10

#define MUTE_DELAY_COUNT 15?? ??? ?//播放人聲后保留的音頻幀數(shù)、不靜音

#define AUDIO_BUFFER_NUM 4?? ??? ?//緩存音頻數(shù)據(jù)的幀數(shù)
#define AUDIO_BUFFER_SIZE 1024?? ?//一幀的音頻數(shù)據(jù)大小
char *audio_buffer[AUDIO_BUFFER_NUM];?? ?//audio_buffer用于緩存音頻數(shù)據(jù)
char *audio_buffer_temp;?? ?//用于交互音頻數(shù)據(jù)
int audio_buffer_pos=0;
#endif

#ifdef MUTE_NOISE_REDUCTION
? ? {
? ? ? ? unsigned int value = 0;
? ? ? ? int is_voice = 0;
? ? ? ? static int is_mute_delay_count;
? ? ? ? //ALOGE("in_begin_swip_num:%d in_begin_narrow_num=%d",in_begin_swip_num,in_begin_narrow_num);?? ??? ?

? ? ? ? ?if(enable_reduction_noise && bytes > AUDIO_BUFFER_SIZE){
? ? ? ? ? ? bytes = AUDIO_BUFFER_SIZE;
? ? ? ? }

? ? ? ? if(enable_reduction_noise){
? ? ? ? ? ? unsigned char * buffer_temp=(unsigned char *)buffer;
? ? ? ? ? ? unsigned int total = 0;
? ? ? ? ? ? unsigned int total_count=0;
?? ??? ??? ?unsigned int total_temp = 0;
? ? ? ? ? ? short data16;
? ? ? ? ? ? int j = 0;
? ? ? ? ? ? for(j=0; j<bytes; j=j+2){
? ? ? ? ? ? ? ? value = buffer_temp[j+1];?? ?//第二個字節(jié)為高位數(shù)據(jù)
? ? ? ? ? ? ? ? value = (value<<8)+buffer_temp[j];?? ?//獲得一個16bit的音頻數(shù)據(jù)
? ? ? ? ? ? ? ? data16 = value&0xFFFF;
? ? ? ? ? ? ? ? if( (data16 & 0x8000) == 0){//正數(shù)
?? ??? ??? ??? ??? ?total +=data16;?? ??? ?//思考:會不會溢出
?? ??? ??? ??? ??? ?total_count++;?? ??? ?//計數(shù)
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }

?? ??? ??? ?total_temp = total/total_count;
? ? ? ? ? ? if(total_temp > threshold_def){
? ? ? ? ? ? ? ? is_voice++;?? ??? ?//檢測到人聲
? ? ? ? ? ? }else {?? ?//is noise
?? ??? ??? ??? ?if(threshold_count == 0){
?? ??? ??? ??? ??? ?threshold = total_temp;
?? ??? ??? ??? ?}else{
?? ??? ??? ??? ??? ?threshold = (threshold+total_temp)/2;
?? ??? ??? ??? ?}
?? ??? ??? ??? ?threshold_count++;
?? ??? ??? ??? ?if(threshold_count >= THRESHOLD_COUNT){
?? ??? ??? ??? ??? ?threshold_def = threshold*2;?? ?//更新閾值,這里的2要對產(chǎn)品實驗來確定。
?? ??? ??? ??? ??? ?threshold_count = THRESHOLD_COUNT;?? ?//此后一直用新閾值,直到停止錄音
?? ??? ??? ??? ?}
?? ??? ??? ?}

?? ??? ??? ?//is_mute_delay_count的意義是,如果前面播放了人聲,那再停止說話之后繼續(xù)保留MUTE_DELAY_COUNT的音頻數(shù)據(jù),這樣不會“戛然而止”。
?? ? ? ? ? ?if( is_voice != 0 ){
?? ? ? ? ? ? ? ?is_mute_delay_count=MUTE_DELAY_COUNT;
?? ? ? ? ? ?}else{
?? ? ? ? ? ? ? ?if(is_mute_delay_count != 0)
?? ? ? ? ? ? ? ? ? ?is_mute_delay_count--;
?? ? ? ? ? ?}

?? ??? ??? ?//audio_buffer的意義:檢測到人聲,要返回說話前的一小段音頻數(shù)據(jù),否則聲音從靜音到人聲有個POP聲的跳躍。
?? ??? ??? ?//這里用audio_buffer來緩存AUDIO_BUFFER_NUM幀數(shù)據(jù)。
?? ? ? ? ? ?if(is_mute_delay_count == 0){//Mute in order to remove noise
?? ? ? ? ? ? ? ?memcpy(audio_buffer[audio_buffer_pos], (char *)buffer, bytes);?? ?//緩存音頻
?? ? ? ? ? ? ? ?memset(buffer, 0, bytes);?? ?//返回靜音數(shù)據(jù)
?? ? ? ? ? ?}else {
?? ? ? ? ? ? ? ?memcpy(audio_buffer_temp, (char *)buffer, bytes);
?? ? ? ? ? ? ? ?memcpy((char *)buffer, audio_buffer[audio_buffer_pos], bytes);?? ?//返回舊的音頻數(shù)據(jù)
?? ? ? ? ? ? ? ?memcpy(audio_buffer[audio_buffer_pos], (char *)audio_buffer_temp, bytes); ?? ?//保存新的音頻數(shù)據(jù)
?? ? ? ? ? ?}
?? ??? ??? ?audio_buffer_pos++;
?? ??? ??? ?if(audio_buffer_pos>=AUDIO_BUFFER_NUM)
? ? ? ? ? ? ? ? audio_buffer_pos=0;
? ? ? ? }
? ? }
#endif

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Android不顯示開機向?qū)Ш烷_機氣泡問題

    Android不顯示開機向?qū)Ш烷_機氣泡問題

    這篇文章主要介紹了Android不顯示開機向?qū)Ш烷_機氣泡問題,需要的朋友可以參考下
    2019-05-05
  • HandlerThread的使用場景和用法詳解

    HandlerThread的使用場景和用法詳解

    這篇文章主要介紹了HandlerThread的使用場景和用法詳解,HandlerThread是Android中的一個線程類,它是Thread的子類,并且內(nèi)部封裝了Looper和Handler,提供了更方便的消息處理和線程操作,需要的朋友可以參考下
    2023-07-07
  • Android中實現(xiàn)布局背景模糊化處理的方法

    Android中實現(xiàn)布局背景模糊化處理的方法

    這篇文章主要介紹了Android中實現(xiàn)布局背景模糊化處理的方法,需要的朋友可以參考下
    2015-04-04
  • 使用Kotlin開發(fā)Android應用教程

    使用Kotlin開發(fā)Android應用教程

    這篇文章主要為大家詳細介紹了使用Kotlin開發(fā)Android應用的教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • Android CardView+ViewPager實現(xiàn)ViewPager翻頁動畫的方法

    Android CardView+ViewPager實現(xiàn)ViewPager翻頁動畫的方法

    本篇文章主要介紹了Android CardView+ViewPager實現(xiàn)ViewPager翻頁動畫的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-06-06
  • 從源碼剖析Android中的Intent組件

    從源碼剖析Android中的Intent組件

    這篇文章主要介紹了Android中的Intent組件,作者從源碼剖析了Intent如何實現(xiàn)組件間的切換,需要的朋友可以參考下
    2016-03-03
  • Android自定義View用切圖顯示字符串

    Android自定義View用切圖顯示字符串

    這篇文章主要為大家詳細介紹了Android自定義View用切圖顯示字符串,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-01-01
  • android app icon 圖標大小尺寸

    android app icon 圖標大小尺寸

    應用程序圖標 (Icon)應當是一個 Alpha 通道透明的32位 PNG 圖片。由于安卓設備眾多,一個應用程序圖標需要設計幾種不同大小。
    2016-05-05
  • Android開發(fā)之廣播機制淺析

    Android開發(fā)之廣播機制淺析

    這篇文章主要介紹了Android開發(fā)之廣播機制淺析,主要包括了發(fā)布、接收及配置廣播的實例,需要的朋友可以參考下
    2014-08-08
  • Android自定義View實現(xiàn)彈性小球效果

    Android自定義View實現(xiàn)彈性小球效果

    前段時間看到一個功能,是一個小球沿著固定軌跡彈動的效果,那么這篇文章小編給大家分享在Android中如何自定義View來實現(xiàn)彈性小球的效果,有需要的可以參考借鑒。
    2016-09-09

最新評論