IOS和Android音頻開發總結

来源:http://www.cnblogs.com/jymblog/archive/2016/06/17/5592047.html
-Advertisement-
Play Games

最近因為項目需要對聲音進行變聲,所以邊學習邊做,發現音頻的處理思路並不難,但是做起來還是有些繁瑣的(比預期的) 趁著腦子還發熱,趕緊把思路總結一下,記錄下來。 主要講三個部分 1,如何變聲2,安卓實現變聲3,ios實現變聲 1. 要想自己寫一個變聲的函數或者庫出來,談何容易,所以採用了大家普遍採用的 ...


最近因為項目需要對聲音進行變聲,所以邊學習邊做,發現音頻的處理思路並不難,但是做起來還是有些繁瑣的(比預期的)

趁著腦子還發熱,趕緊把思路總結一下,記錄下來。

主要講三個部分

1,如何變聲2,安卓實現變聲3,ios實現變聲

1.

要想自己寫一個變聲的函數或者庫出來,談何容易,所以採用了大家普遍採用的庫SoundTouch。

該庫可以實現改變聲音的速度,節拍,音調(這個最重要,可以把聲音的音調調高調低,使之變成男生女生,可以參照湯姆貓)

使用的思路為把整個庫放到不同平臺的底層,使用時只需包含頭文件soundtouch.h即可.

SoundTouch類提供了許多方法,其中最重要的就是setPitch,setRate這幾個調節聲音參數的方法,具體使用時自行設置參數。

但是在使用前需要預先設置一下其中的幾個函數的參數如下:

                mSoundTouchInstance->setSetting(SETTING_USE_QUICKSEEK, 0);
                mSoundTouchInstance->setSetting(SETTING_USE_AA_FILTER, !(0));
                mSoundTouchInstance->setSetting(SETTING_AA_FILTER_LENGTH, 32);
                mSoundTouchInstance->setSetting(SETTING_SEQUENCE_MS, 40);
                mSoundTouchInstance->setSetting(SETTING_SEEKWINDOW_MS, 16);
                mSoundTouchInstance->setSetting(SETTING_OVERLAP_MS, 8);

 然後設置需要的參數

                mSoundTouchInstance->setChannels(2);

                mSoundTouchInstance->setSampleRate(8000);

                mSoundTouchInstance->setPitch(2);

這裡解釋一下音頻處理的幾個參數,很重要。

聲道:channals,可以是單聲道和雙聲道,分別對應1,2

採樣率:SampleRate  8000-44100不等,一般是常用的幾個值,安卓裡面好像44100是所有設備都支持的,所以設置成44100比較保險吧

每個聲道的位數:bitsPerChannel 一般設置為16

每個幀的聲道數 ChannelsPerFrame    對於pcm數據來說,這個是1

還有幾個參數,對於安卓和ios可能說法不太一樣,以上幾個是都要用到的,比較重要,必須得掌握

 

2.Android中實現變聲

因為項目要求錄音要實時播放,所以需要採用讀取音頻數據流(PCM格式)來播放,採用的api是AudioRecorder和AudioTrack。

具體的使用方法相關資料較多,官方文檔也比較詳細。大致思路就是先初始化:

        //initilize
     trbusize=AudioTrack.getMinBufferSize(RECORDER_SAMPLERATE,AudioFormat.CHANNEL_OUT_STEREO, AudioFormat.ENCODING_PCM_16BIT); mAudioTrack = new AudioTrack(AudioManager.STREAM_MUSIC,RECORDER_SAMPLERATE, AudioFormat.CHANNEL_IN_STEREO, AudioFormat.ENCODING_PCM_16BIT, trbusize, AudioTrack.MODE_STREAM); rebusize = AudioRecord.getMinBufferSize(RECORDER_SAMPLERATE, AudioFormat.CHANNEL_IN_STEREO,AudioFormat.ENCODING_PCM_16BIT); mAudioRecord= new AudioRecord(MediaRecorder.AudioSource.MIC,RECORDER_SAMPLERATE,AudioFormat.CHANNEL_IN_STEREO, AudioFormat.ENCODING_PCM_16BIT, rebusize);

這裡因為不同設備支持的參數可能不同,需要是可以寫一個迴圈把所有可能的參數全試一遍。

之後是錄音和播放,可以分別放到兩個線程裡面。一般來說都是把錄音數據保存到文件中,然後再進行播放,這樣可以應付一般的錄音需求。但不足之處在於,錄音時間久了,文件會很大,如果在網路上實時播放的話這樣肯定不行。解決方法就是將錄音的數據傳到一個緩衝區,然後播放時直接從緩衝區取走數據即可。這個緩衝區可以考慮用迴圈隊列或者在java裡面可以直接用一個LinkedList實現。

然後是變聲部分,安卓裡面要想用c++庫的話,只能通過jni來實現,可以寫幾個函數。

                while(isInstancePlaying){
                    if(l<21){
                        byte[] mbyte=new byte[64];
                        mAudioRecord.read(mbyte,0,64);
                        SoundTouch.getSoundTouch().putSamples(mbyte,0,INPUT_LENGTH);
                        SoundTouch.getSoundTouch().setPitchSemiTones(pitchTone);
                        SoundTouch.getSoundTouch().receiveSamples(mbyte,INPUT_LENGTH);
                        byteArray.add(mbyte);
                        l=byteArray.size();
                    }
                    else{
                            mAudioTrack.write(byteArray.getFirst(),0,64);
                            byteArray.removeFirst();
                            l=byteArray.size();
                    }

代碼中有三個函數putSamples,setPitchSemiTones,receiveSamples.這三個都是native方法,在SoundTouch庫中分別通過SoundTouch類提供的對應函數實現,比較簡單,通過這幾個函數即可實現聲音的變聲。

l變數是LinkedList(代碼中的byteArray)的長度,當小於20時添加到byteArray的末尾,同時AudioTrack不斷讀取數組中的第一個元素來播放然後刪除該元素。

最後播放完要記得釋放mAudioTrack和mAudioRecorder。通過stop和release方法實現。

 

3.IOS實現變聲

 

因為本人之前沒接觸過ios所以做起來遇到了不少問題,還好最後解決了。

ios裡面的音頻處理比起安卓來說感覺要麻煩一些,用到的核心api就是AudioQueue,正在使用之前一定要好好理解一下它的原理,跟安卓不同的是ios播放和錄音都是用的這個api。就相當於它一個東西實現了安卓中AudioRecorder和AudioTrack的功能,只不過在播放和錄音過程中內部的流程有所變化。

核心思想:

Audio裡面有自帶的一個隊列,首先用戶創建若幹個(3-6個左右都行)緩衝器用來裝填音頻數據,在自帶隊列中播放或錄音完後使用用戶自定義的回調函數進行處理,使得緩衝器能夠被重新利用,並且可以在回調函數中實現用戶自定義的一些功能,比如變聲,寫入文件等等操作。官方給了說明圖比較詳細,需要著重理解一下。

首先是錄音的流程圖:

 

 

然後是播放的流程圖:

如何變聲呢?

ios的變聲不需要安卓的jni,因為oc語言可以和c++混編,所以這點相對來說要簡單許多。流程如下:

首先在你的程式中實例化一個SoundTouch類,然後在初始化時將它的參數設置好(setSetting),之後在上面所述的回調函數裡面就可以將錄音得到的數據流進行處理然後選擇保存到文件或者直接播放。思路就是這樣,但是裡面的函數的參數相對還是比較繁瑣的,前面原理沒理解的話這邊就很難做下去了。

實時播放?

思路同Android,可以寫一個迴圈隊列用來緩存音頻數據,然後邊錄音往裡面傳數據邊播放,跟安卓不同的是這些操作需要放到相應的回調函數裡面來實現,有個簡單的辦法是在錄音的回調函數裡面直接播放pcm數據。因為數據是一塊一塊的進來的,每使用完一次緩衝器才會調用一次回調函數,可以直接在回調函數裡面進行播放。

 

以上就是兩個平臺上實現錄音和實時播放的簡單介紹,這裡面的東西還是蠻多的,值得深入研究。

 

 

 

  

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 說在前面: 嗯...博主現在是大一,前段時間犯腦癌想開一個技術博,記錄一下學習過程,也是這麼做的。 為什麼要學前端呢?無非就是為了走在發家致富,迎娶白富美的康莊大道上。那幾周來對前端開發已經入門了,(鼓掌,鼓掌)非常有意思。但還是感覺道阻且長。不懂的東西越來越多......不過還好,有好多人幫我呀, ...
  • 先來個相容性說明,洗洗腦: div{transform:rotate(7deg);-ms-transform:rotate(7deg); /* IE 9 */-moz-transform:rotate(7deg); /* Firefox */-webkit-transform:rotate(7deg ...
  • 遞歸函數: function factorical(num){ if(num<=1){ return 1; } else{ return num*factorical(num-1); } } factorial(2)//2 這個遞歸函數就是用函數來調用函數本身,但是這樣真的好嗎,好 接下來看這裡 v ...
  • 從jQuery API 文檔中可以知道,jQuery自定義動畫的函數.animate( properties [, duration] [, easing] [, complete] )有四個參數: properties:一組包含作為動畫屬性和終值的樣式屬性和及其值的集合 duration(可選): ...
  • 還記得之前的javascript入門裡的講的confirm 消息對話框嗎?不記得也沒關係,我們先來回顧一下,然後在詳細講它。 複習: confirm 消息對話框通常用於允許用戶做選擇的動作,如:“你對嗎?”等。彈出對話框(包括一個確定按鈕和一個取消按鈕)。 語法: 參數說明: str:在消息對話框中 ...
  • 當初始化SpaceShip原型時,我們尚未創建任何能作為第一個參數來傳遞的場景。並且SpaceShip的原型加入到場景的註冊表中,而這絕不是我們想做的。這是一種使用子類時常用的方法。應當僅僅在子類構造函數中調用父類構造函數,而不是當創建子類原型時調用它。 一旦創建了SpaceShip的原型對象,我們... ...
  • 當頁面使用 UITabBarController + UINavigationController 框架的時候,當跳轉到詳情頁面的時候,如果 UITabBar 仍然存在的話就會造成邏輯混亂,用戶體驗也會下降,因此我們就有一個在詳情頁將 UITabBar 隱藏的需求,當然,在其他的一些情況也可能有隱藏 ...
  • 用類對象作為ArrayAdapter綁定的基本數據類型(和SimpleAdater效果類似) 一般ArrayAdapter綁定的基本數據類型是String,接下來介紹一下類對象作為基本數據類型; 首先,新建一個類News,這個類作為基本的數據類型 接下來先把listView的item的佈局確定下來, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...