Swift與硬件打交道封裝的方法二

之前寫過一篇Swift與硬件打交道封裝的方法惦费,現(xiàn)在又開始做硬件開發(fā)了,主要方向是音頻(AudioUnit)視頻的采集隘世、播放栈雳、合成、以及藍(lán)牙蝉娜、socket通訊唱较,這里繼續(xù)記錄下開發(fā)中封裝的一些方法,以便以后再次使用召川。

獲取AudioBuffer中音頻數(shù)據(jù)的平均分貝值

這里是AudioBuffer南缓,且采樣位數(shù)為16bit,如果是Data類型荧呐,也可以按照下文的方法將Data轉(zhuǎn)為UInt16數(shù)組再計(jì)算(如果采樣位數(shù)為8bit汉形,需要用Int8計(jì)算)
(DBSPL)

    private func getVolumeValue(buffer: AudioBuffer) {
        var pcmAll: Int = 0
        let bufferPoint = UnsafeMutableBufferPointer<Int16>.init(buffer)
        let bufferArray = Array(bufferPoint)
        let len = bufferArray.count
        for index in 0..<len {
            let value = bufferArray[index]
            pcmAll += Int(value) * Int(value)
        }
        let mean: Double = Double(pcmAll) / Double(len)
        let volume: Double = 10 * log10(mean)
        guard "\(volume)" != "nan" else { return }
        print(volume)
        //0-42 42-97
        //42分貝以下認(rèn)為是靜音,97分貝認(rèn)為是最大
       //如果需要轉(zhuǎn)換為0-1
        let db = min(volume - 42, 97-42)
        var value = 0.0
        if db > 0 {
            value = db/(97-42)
        }
        print(value)
    }

附DBFS計(jì)算(電平表顯示)

private func getVolumeValue(buffer: AudioBuffer) {
        let bufferPoint = UnsafeMutableBufferPointer<Int16>.init(buffer)
        let originalArray = Array(bufferPoint)
        let dic = [
            64: 8,
            128: 16,
            256: 24,
            512: 32,
            1024: 64,
            2048: 128
        ]
        let base = dic[originalArray.count] ?? 16
        var index = base
        var bufferArray = [Int16]()
        while index < originalArray.count {
            bufferArray.append(originalArray[index])
            index += base
        }
        DispatchQueue.global().async {
            let maxValue = bufferArray.max() ?? 0
            let minValue = bufferArray.min() ?? 0
            let value = max(maxValue, abs(minValue))
            let volume: Double = 20 * log10(Double(value)/65535)
            guard !volume.isNaN else { return }
            self.delegate?.audioRecorder(recorder: self, didUpdate: volume)
        }
    }

根據(jù)PCM數(shù)據(jù)長度生成WAV音頻的頭

func getWavHeader(asbd: WSASBD, pcmDataLen: Int) -> Data {
        var wavHeader = Data(count: 44)
        // RIFF
        wavHeader[0 ... 3] = "RIFF".data(using: .ascii)!

        // 04~07 文件長度倍阐,暫時不填

        // wave
        wavHeader[8 ... 11] = "WAVE".data(using: .ascii)!

        // fmt
        wavHeader[12 ... 15] = "fmt ".data(using: .ascii)!

        // 過濾字節(jié) 00000010
        wavHeader[16 ... 19] = withUnsafeBytes(of: UInt32(littleEndian: 16)) { Data($0) }

        // 格式種類(值為1時,表示數(shù)據(jù)為線性pcm編碼)
        wavHeader[20] = 1
        wavHeader[21] = 0

        // chanel
        wavHeader[22] = UInt8(asbd.channels)
        wavHeader[23] = 0

        let samplerate = UInt32(littleEndian: UInt32(asbd.sampleRate))
        wavHeader[24 ... 27] = withUnsafeBytes(of: samplerate) { Data($0) }

        let bitRate = UInt32(littleEndian: UInt32(asbd.sampleRate) * asbd.channels * asbd.mBitsPerChannel / 8)
        wavHeader[28 ... 31] = withUnsafeBytes(of: bitRate) { Data($0) }

        let sampleBit = UInt16(littleEndian: UInt16(asbd.channels * asbd.mBitsPerChannel / 8))
        wavHeader[32 ... 33] = withUnsafeBytes(of: sampleBit) { Data($0) }

        wavHeader[34] = UInt8(asbd.mBitsPerChannel)
        wavHeader[35] = 0

        // data
        wavHeader[36 ... 39] = "data".data(using: .ascii)!

        // 04~07 文件長度
        wavHeader[4 ... 7] = withUnsafeBytes(of: UInt32(littleEndian: UInt32(pcmDataLen + 44 - 8))) { Data($0) }
        // 40-43 PCM數(shù)據(jù)大小
        wavHeader[40 ... 43] = withUnsafeBytes(of: UInt32(littleEndian: UInt32(pcmDataLen))) { Data($0) }
        return wavHeader
    }

struct WSASBD {
    /// 聲道數(shù)
    var channels: UInt32 = 1
    /// 采樣率:
    var sampleRate: Double = 48000
    /// 位深
    var mBitsPerChannel: UInt32 = 16
}

檢測是否有耳機(jī)設(shè)備

監(jiān)聽設(shè)備變化的通知名字AVAudioSession.routeChangeNotification

private func getHeadPhoneState(){
        var state = false
        for outPort in AVAudioSession.sharedInstance().currentRoute.outputs {
            if outPort.portType == .headphones ||
                outPort.portType == .bluetoothA2DP ||
                outPort.portType == .bluetoothHFP
            {
                state = true
                break
            }
        }
        hasHeadPhone = state
    }

Double 類型與 Data 類型互轉(zhuǎn)

這里舉例Double類型概疆,其他類型可以以此類推

let value = Date().timeIntervalSince1970
//Double轉(zhuǎn)data
let data = withUnsafeBytes(of: value) { Data($0) }
//data轉(zhuǎn)Double
var value2 = 0.0
let _ = withUnsafeMutableBytes(of: &value2, { data.copyBytes(to: $0)} )
print(value2)

Data 轉(zhuǎn)Int16數(shù)組

let data = Data([0, 7, 3, 2, 1, 0, 0, 4])
let int16array = data.withUnsafeBytes {
     Array($0.bindMemory(to: Int16.self)).map(Int16.init(bigEndian:))
}
print(int16array)

Data 截取及移除

extension Data{
    func getSubData(start: Int, num: Int) -> Data?{
        guard start >= 0 && num >= 0 else { return nil }
        guard self.count >= start + num else {
            return nil
        }
        let startIndex = self.index(self.startIndex, offsetBy: start)
        let endIndex = self.index(self.startIndex, offsetBy: start + num)
        let range = startIndex..<endIndex
        return self[range]
    }

//或者
  func getSubData2(start: Int, num: Int) -> Data?{
        guard start >= 0 && num >= 0 else { return nil }
        guard self.count >= start + num else {
            return nil
        }
        let byte = [UInt8](self)
        return Data(byte[start...start+num])
    }
}

private func removeSubData(data: inout Data, start: Int, num: Int){
        guard start >= 0 && num >= 0 else { return }
        guard data.count >= start + num else { return }
        let startIndex = data.index(data.startIndex, offsetBy: start)
        let endIndex = data.index(data.startIndex, offsetBy: start + num)
        let range = startIndex..<endIndex
        data.removeSubrange(range)
    }

十六進(jìn)制數(shù)據(jù)Data互轉(zhuǎn) 以及大小端模式

在轉(zhuǎn)大小端時,需要先確定轉(zhuǎn)出的數(shù)據(jù)需要占幾個字節(jié)峰搪,例如需要占2字節(jié)則使用UInt16岔冀、需要占4字節(jié)用Uint32,以此類推概耻,例如我要將0x06轉(zhuǎn)為2字節(jié)的Data

let bigEndianData = withUnsafeBytes(of: UInt16(bigEndian: UInt16(0x06))) { Data($0) } //大端 0x0006
var lc: UInt16 = 0
(bigEndianData as NSData).getBytes(&lc, range: NSMakeRange(0, bigEndianData.count))
let bigEndianValue = Int(UInt16(bigEndian: lc)) //6
let littleEndianData = withUnsafeBytes(of: UInt16(littleEndian: UInt16(0x06))) { Data($0) } //小端 0x0600
(littleEndianData as NSData).getBytes(&lc, range: NSMakeRange(0, littleEndianData.count))
let littleEndianValue = Int(UInt16(littleEndian: lc)) //6

獲取ip地址使套、子網(wǎng)掩碼罐呼、廣播地址

en0: WiFi
bridge100:熱點(diǎn)開啟方

func getWiFiAddress() -> String? {
        var address: String?
        var ifaddr: UnsafeMutablePointer<ifaddrs>?
        guard getifaddrs(&ifaddr) == 0 else { return nil }
        guard let firstAddr = ifaddr else { return nil }
        for ifptr in sequence(first: firstAddr, next: { $0.pointee.ifa_next }) {
            let interface = ifptr.pointee
            let addrFamily = interface.ifa_addr.pointee.sa_family
            if addrFamily == UInt8(AF_INET) {
                let name = String(cString: interface.ifa_name)
                if name == "en0" || name == "bridge100" || name == "en1" {
                    var hostname = [CChar](repeating: 0, count: Int(NI_MAXHOST))
                    getnameinfo(interface.ifa_addr, socklen_t(interface.ifa_addr.pointee.sa_len),
                                &hostname, socklen_t(hostname.count),
                                nil, socklen_t(0), NI_NUMERICHOST)
                    address = String(cString: hostname)
                    var broadcast = "error"
                    ifptr.pointee.ifa_dstaddr.withMemoryRebound(to: sockaddr_in.self, capacity: 1) {
                        sockaddr in
                        if let addressCString = inet_ntoa(sockaddr.pointee.sin_addr) {
                            broadcast = String(cString: addressCString)
                        }
                    }
                    var net = interface.ifa_netmask.pointee
                    var netmaskName = [CChar](repeating: 0, count: Int(NI_MAXHOST))
                    getnameinfo(&net, socklen_t(net.sa_len), &netmaskName, socklen_t(netmaskName.count),
                                nil, socklen_t(0), NI_NUMERICHOST)
                    let netmask = String(cString: netmaskName)
                    print("address: \(address), name:\(name), 廣播地址: \(broadcast), 子網(wǎng)掩碼: \(netmask)")
                }
            }
        }
        freeifaddrs(ifaddr)
        return address
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市侦高,隨后出現(xiàn)的幾起案子嫉柴,更是在濱河造成了極大的恐慌,老刑警劉巖奉呛,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件计螺,死亡現(xiàn)場離奇詭異,居然都是意外死亡侧馅,警方通過查閱死者的電腦和手機(jī)危尿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來馁痴,“玉大人谊娇,你說我怎么就攤上這事÷拊危” “怎么了济欢?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長小渊。 經(jīng)常有香客問我法褥,道長,這世上最難降的妖魔是什么酬屉? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任半等,我火速辦了婚禮,結(jié)果婚禮上呐萨,老公的妹妹穿的比我還像新娘杀饵。我一直安慰自己,他們只是感情好谬擦,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布切距。 她就那樣靜靜地躺著,像睡著了一般惨远。 火紅的嫁衣襯著肌膚如雪谜悟。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天北秽,我揣著相機(jī)與錄音葡幸,去河邊找鬼。 笑死羡儿,一個胖子當(dāng)著我的面吹牛礼患,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼缅叠,長吁一口氣:“原來是場噩夢啊……” “哼悄泥!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起肤粱,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤弹囚,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后领曼,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體鸥鹉,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年庶骄,在試婚紗的時候發(fā)現(xiàn)自己被綠了毁渗。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡单刁,死狀恐怖灸异,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情羔飞,我是刑警寧澤肺樟,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站逻淌,受9級特大地震影響么伯,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜卡儒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一田柔、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧骨望,春花似錦凯楔、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽邻遏。三九已至糠亩,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間准验,已是汗流浹背赎线。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留糊饱,地道東北人垂寥。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親滞项。 傳聞我的和親對象是個殘疾皇子狭归,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內(nèi)容

  • Data轉(zhuǎn)HexString 或者 HexString轉(zhuǎn)Data hexString與Data互轉(zhuǎn)OC版 進(jìn)制轉(zhuǎn)換...
    T92閱讀 1,540評論 0 3
  • 本章的主要內(nèi)容有: 2.1節(jié) 信息的存儲 2.2節(jié) 整數(shù)的表示 2.3節(jié) 整數(shù)的運(yùn)算 2.4節(jié) 浮點(diǎn)數(shù) 2.5節(jié) ...
    橡樹人閱讀 344評論 0 3
  • ArrayBuffer對象、TypedArray視圖和DataView視圖是 JavaScript 操作二進(jìn)制數(shù)據(jù)...
    硅谷干貨閱讀 274評論 0 0
  • 僅為方便個人查詢使用來源:http://es6.ruanyifeng.com/#docs/arraybuffer ...
    zqyadam閱讀 216評論 0 0
  • 關(guān)鍵詞/關(guān)鍵類 1.#import跟#include有什么區(qū)別文判,@class呢过椎,#import<>跟#import...
    丶逐漸閱讀 1,425評論 0 1