swift - 如何在 macOS 中获取实时麦克风输入?

标签 swift macos core-audio microphone avaudioengine

我尝试使用以下代码访问实时麦克风数据:

import AVFoundation // for AVAudioEngine

class Mic
{
    public let audioEngine = AVAudioEngine()

    func startRecording() throws
    {
        // https://forums.developer.apple.com/thread/44833
        //audioEngine.mainMixerNode  // causes DIFFERENT crash!

        audioEngine.prepare()  // CRASHES


        let inputNode = audioEngine.inputNode
        if inputNode.inputFormat(forBus: 0).sampleRate == 0 {
            exit(0);
        }

        let recordingFormat = inputNode.outputFormat(forBus: 0)
        inputNode.installTap(onBus: 0, bufferSize: 1024, format: recordingFormat) { (buffer: AVAudioPCMBuffer, when: AVAudioTime) in
            print( "YES! Got some samples!")
        }

        audioEngine.prepare()

        try audioEngine.start()
    }

    //public
    func stopRecording()
    {
        audioEngine.stop()
    }
}

但是它崩溃了:

        audioEngine.prepare()  // CRASHES

2019-07-16 17:51:34.448107+0300 realtime_mic[8992:386743] [avae]

AVAEInternal.h:76 required condition is false:

[AVAudioEngineGraph.mm:1318:Initialize: (inputNode != nullptr || outputNode != nullptr)]

realtime_mic[8992:386743] required condition is false: inputNode != nullptr || outputNode != nullptr2019-07-16 17:51:34.449214+0300

可以看出,我尝试应用黑客/补丁:

        // https://forums.developer.apple.com/thread/44833
        audioEngine.mainMixerNode

但这会导致不同的崩溃:

2019-07-16 17:50:34.315005+0300 realtime_mic[8901:385699] [plugin] AddInstanceForFactory:

No factory registered for id F8BB1C28-BAE8-11D6-9C31-00039315CD46 2019-07-16

17:50:34.349337+0300 realtime_mic[8901:385699]

HALC_ShellDriverPlugIn::Open: Can't get a pointer to the Open routine

2019-07-16 17:50:34.354277+0300 realtime_mic[8901:385699] [ddagg]

AggregateDevice.mm:776 couldn't get default input device, ID = 0, err = 0!

我已发送权利以防万一:macOS Entitlements audio-input vs. microphone ——但无济于事。

正确的做法是什么?

测试用例位于:https://github.com/p-i-/macOS_rt_mic

最佳答案

在 testRecord.swift 中输入以下代码:

import Foundation
import AVFoundation 

print("starting")

public let audioEngine = AVAudioEngine()

var flag = 0

func startRecording() throws {

    let inputNode = audioEngine.inputNode
    let srate = inputNode.inputFormat(forBus: 0).sampleRate
    print("sample rate = \(srate)")
    if srate == 0 {
        exit(0);
    }

    let recordingFormat = inputNode.outputFormat(forBus: 0)
        inputNode.installTap(onBus: 0, 
        bufferSize: 1024, 
        format: recordingFormat) { 
            (buffer: AVAudioPCMBuffer, when: AVAudioTime) in
            let n = buffer.frameLength
            let c = buffer.stride
            if flag == 0 { 
                print( "num samples  = \(n)") ; 
                print( "num channels = \(c)") ; 
                flag = 1 
            }
        }

    try audioEngine.start()
}

func stopRecording() {
    audioEngine.stop()
}

do {
    try startRecording()
} catch {
    print("error?")
}

usleep(UInt32(1000*1000))         // sleep 1 second before quitting
stopRecording()
print("done")
exit(0)

在 macOS 10.14.5/Xcode 10.2.1 上使用 swiftc 编译了 testRecord.swift ;然后尝试从终​​端运行结果。第一次运行时,macOS 询问终端是否可以拥有麦克风权限。回复是,但是没有输出。

但是在随后的运行中它输出:

starting

sample rate = 44100.0

num samples = 4410

num channels = 1

done

因此,您可能需要在系统偏好设置:隐私:麦克风中授予您的应用一些权限

关于swift - 如何在 macOS 中获取实时麦克风输入?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57060130/

相关文章:

swift - 使用 Swift,为什么按下一个 UIStepper 会影响两个步进器?

macos - 从 Mac 部署 war 到生产环境 |等同于 WinSCP

cocoa - 接受 Dock 图标上的卷滴

ios - AudioBufferList 在 Swift 中分配

ios - 代码中有一个类对象,但其类型未声明,无法使用

ios - Xcode 11 SPM 身份验证失败,因为未提供凭据

ios - 在圆上使用 UIBezierPath

objective-c - Cocoa/OSX - NSWindow standardWindowButton 在复制并再次添加后表现异常

ios - 核心音频内存问题

core-audio - 我如何子类化 AVAudioUnit?