簡體   English   中英

無法在 Qt 中將音頻從麥克風錄制到 QFile

[英]Unable to record audio from microphone to QFile in Qt

我正在嘗試從桌面中的默認麥克風錄制音頻並將其存儲在 QFile 中。 最后我想使用 windows 媒體播放器播放錄制的文件。 問題是 QAudioInput 有一個方法 bytesReady() 在我的情況下總是返回 0 。 即使音頻狀態是 QAudio::ActiveState,bytesReady() 也會顯示 0 字節,這意味着沒有要從麥克風讀取的音頻字節。 系統麥克風工作正常。 我的代碼得到了完美的編譯,我還在指定位置獲得了一個 80KB 大小的文件,但即使保存為 .wav,該文件也無法與 windows 媒體播放器一起播放,它返回一個錯誤。 在我的情況下,輸出窗口每次都將 bytesReady 顯示為 0。 我懷疑 QAudioInput 無法從麥克風讀取音頻數據。

你能幫我找出代碼中的錯誤嗎?

 `class mikeDemoClass : public QWidget
    {
        Q_OBJECT

    public:
        mikeDemoClass(QWidget *parent = 0, Qt::WFlags flags = 0);
        ~mikeDemoClass();

    public slots:

        void startRecording();
        void browseFiles();
        void stopRecording();
        void handleAudioInputState(QAudio::State);
        void notified();

    private:
        Ui::mikeDemoWidget ui;
        QAudioInput *audioInput;
        QFile *recordFile;
        QTimer *testTimer;
        int audio_state;

    };

// cpp file starts here

    #include <QIODevice>
    #include "mic_demo.h"

    mikeDemoClass::mikeDemoClass(QWidget *parent, Qt::WFlags flags)
        : QWidget(parent, flags)
    {
        ui.setupUi(this);
        audioInput = NULL;
        recordFile = NULL;
        audio_state = -1;

        connect(ui.browseButton,SIGNAL(clicked()),this,SLOT(browseFiles()));
        connect(ui.recordingButton,SIGNAL(clicked()), this,SLOT(startRecording()));
    }

    mikeDemoClass::~mikeDemoClass()
    {
        if(recordFile)
        {
            delete recordFile;
            recordFile = NULL;
        }
    }

    void mikeDemoClass::browseFiles()
    {
        QString FileName = QFileDialog::getSaveFileName(this, tr("Browse Files"), "D:/", tr("Media Files (*.raw)"));
        if(!FileName.isEmpty())
        {
            recordFile = new QFile(FileName);
            QTextDocument *textDoc = new QTextDocument(FileName);
            ui.textEdit->setDocument(textDoc);
        }

    }

    void mikeDemoClass::startRecording()
    {
        bool status = recordFile->open(QIODevice::WriteOnly);
        if(!status)
        {
            qDebug() <<"Error opening the file";
        }


        QString default_deviceName = "";
        QAudioFormat preferred_format;

        QList<QAudioDeviceInfo> device_list = QAudioDeviceInfo::availableDevices(QAudio::AudioInput);
        int count = device_list.count();

        if(device_list.empty())
        {
            qDebug() <<"The Audio Input Devices is empty";
        }
        else
        {

        foreach(QAudioDeviceInfo device_info, device_list)
        {
            QString device_name = device_info.deviceName();
            qDebug() << "device_name:" << device_name.toLatin1();
        }

        QAudioDeviceInfo info = QAudioDeviceInfo::defaultInputDevice();
        default_deviceName = info.deviceName();

        if(!default_deviceName.isEmpty())
        {
            preferred_format = info.preferredFormat();
            QString codec = preferred_format.codec();
            int sampleRate = preferred_format.sampleRate();
            int sampleSize = preferred_format.sampleSize();
            int channelCount = preferred_format.channelCount();
            int sampleType = preferred_format.sampleType();
            int byteOrder =  preferred_format.byteOrder();
            qDebug() << "codec:" << codec.toLatin1() << "sampleRate :" << sampleRate << "sampleSize:" << sampleSize << "channel Count:" << channelCount << "sample type:" <<sampleType
                << "byteOrder:" << byteOrder;
        }

    }

    QAudioFormat format;
    format.setSampleRate(8000);
    format.setChannels(1);
    format.setSampleSize(8);
    format.setCodec("audio/PCM");
    format.setByteOrder(QAudioFormat::LittleEndian);
    format.setSampleType(QAudioFormat::UnSignedInt);

    QAudioDeviceInfo info = QAudioDeviceInfo::defaultInputDevice();
    if(!info.isFormatSupported(format))
    {
        qDebug() <<"Default format not supported, try to use nearest format";
        format = info.nearestFormat(format);
    }

    audioInput = new QAudioInput(info, format, this);

     connect(audioInput, SIGNAL(notify()), this, SLOT(notified()));
    connect(audioInput,SIGNAL(stateChanged(QAudio::State)),this, SLOT(handleAudioInputState(QAudio::State)));
    QTimer::singleShot(10000, this, SLOT(stopRecording()));

    //audioInput->setBufferSize(4096);
    qDebug() << "platform buffer size:" << audioInput->bufferSize();
    audioInput->start(recordFile);
}

void mikeDemoClass::stopRecording()
{
    testTimer->stop();
    audioInput->stop();
    recordFile->close();
    delete audioInput;
}

void  mikeDemoClass::handleAudioInputState(QAudio::State state)
{
    qDebug() << "Audio State:" << state;

    audio_state = state;

    if(state == QAudio::StoppedState)
    {
        qDebug() << "Error State:" << audioInput->error();

        if(audioInput->error() != QAudio::NoError)
        {
            qDebug() << "QAudioInput error:" << audioInput->error();
        }
    }
}

void  mikeDemoClass::notified()
{
    if(audio_state ==  QAudio::ActiveState)
    {
        qDebug() << "Error State:" << audioInput->error();

        qDebug() << "platform buffer size after calling QAudioInput start():" << audioInput->bufferSize();

        qDebug() << "bytesReady = " << audioInput->bytesReady()
        << ", " << "elapsedUSecs = " <<audioInput->elapsedUSecs()
        << ", " << "processedUSecs = "<<audioInput->processedUSecs();
    }

}`

輸出顯示如下:

   device_name: "Microphone (High Definition Aud" 
device_name: "default" 
codec: "audio/pcm" sampleRate : 11025 sampleSize: 8 channel Count: 1 sample type: 1 byteOrder: 1 
Default format not supported, try to use nearest format 
platform buffer size: 0 
Audio State: 0 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  1003000 ,  processedUSecs =  1000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  1998000 ,  processedUSecs =  2000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  3003000 ,  processedUSecs =  3000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  4000000 ,  processedUSecs =  4000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  5006000 ,  processedUSecs =  5000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  6001000 ,  processedUSecs =  6000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  7005000 ,  processedUSecs =  7000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  8002000 ,  processedUSecs =  8000000 
Error State: 0 
platform buffer size after called QAudioInput start(): 1600 
bytesReady =  0 ,  elapsedUSecs =  8997000 ,  processedUSecs =  9000000 
Audio State: 2 
Error State: 0 

將您的代碼與Qt 文檔中的示例進行比較后,您使用通知信號來處理 QAudio::ActiveState,而示例代碼將插槽連接到 stateChanged 信號:-

void AudioInputExample::handleStateChanged(QAudio::State newState)
{
    switch (newState) {
        case QAudio::StoppedState:
            if (audio->error() != QAudio::NoError) {
                // Error handling
            } else {
                // Finished recording
            }
            break;

        case QAudio::ActiveState:
            // Started recording - read from IO device
            break;

        default:
            // ... other cases as appropriate
            break;
    }
}

正如notify() 所述:-

當 x ms 的音頻數據在 setNotifyInterval 設置的時間間隔內被處理時發出這個信號

間隔可以為零嗎?

但是,我建議遵循 文檔中顯示的示例代碼並使用 stateChanged 信號,而不是通知。

將 PHDEBUG 或 PHDBG 視為 qDebug() 並查看以下內容(我認為您缺少的一點是作為緩沖區的QIODevice 。):

讀者.cpp :

#include "PhLtcReader.h"

PhLtcReader::PhLtcReader(QObject *parent) :
    QObject(parent),
    _input(NULL),
    _position(0),
    _buffer(NULL)
{
    _decoder = ltc_decoder_create(1920, 3840);
    PHDEBUG << "LTC Reader created";
}

bool PhLtcReader::init(QString input)
{
    QList<QAudioDeviceInfo> list = QAudioDeviceInfo::availableDevices(QAudio::AudioInput);

    if(list.isEmpty())
    {
        PHDEBUG << "No audio input device";
        return false;
    }

    QAudioDeviceInfo info = QAudioDeviceInfo::defaultInputDevice();

    foreach(QAudioDeviceInfo device, list)
    {
        if(device.deviceName() == input)
            info = device;
    }

    PHDEBUG << "LTC input device :" << info.deviceName();

    QAudioFormat format;
    format.setCodec("audio/pcm");
    format.setByteOrder(QAudioFormat::LittleEndian);
    format.setSampleRate(48000);
    format.setChannelCount(1);
    format.setSampleSize(8);
    format.setSampleType(QAudioFormat::SignedInt);

    if(!info.isFormatSupported(format))
    {
        PHDEBUG << "Unsupported audio format";
        return false;
    }

    _position = 0;
    _input = new QAudioInput(info, format);

    connect(_input, SIGNAL(notify()), this, SLOT(onNotify()));
    _buffer = _input->start();
    _input->setNotifyInterval(10);

    _pauseDetector.start();

    return true;
}

void PhLtcReader::close()
{
    if(_input)
    {
        _input->stop();
        delete _buffer;
        _buffer = NULL;
        delete _input;
        _input = NULL;
    }
}

QList<QString> PhLtcReader::inputList()
{
    QList<QString> names;
    QList<QAudioDeviceInfo> list = QAudioDeviceInfo::availableDevices(QAudio::AudioInput);

    foreach(QAudioDeviceInfo device, list)
        names.append(device.deviceName());

    return names;
}

void PhLtcReader::onNotify()
{

    QByteArray array = _buffer->readAll();
    char max = 0;
    for(int i = 0; i < array.count(); i++)
    {
        if(array.at(i) > max)
            max = array.at(i);
    }

    ltc_decoder_write(_decoder, (ltcsnd_sample_t*)array.data(), array.count(), _position);

    LTCFrameExt frame;
    unsigned int *hhmmssff = new unsigned int[4];
    while(ltc_decoder_read(_decoder, &frame))
    {
        PhFrame oldFrame = _clock.frame();
        hhmmssff[0] = frame.ltc.hours_tens * 10 + frame.ltc.hours_units;
        hhmmssff[1] = frame.ltc.mins_tens * 10 + frame.ltc.mins_units;
        hhmmssff[2] = frame.ltc.secs_tens * 10 + frame.ltc.secs_units;
        hhmmssff[3] = frame.ltc.frame_tens * 10 + frame.ltc.frame_units;
    }
    delete hhmmssff;

    _position += array.count();
}

讀者.h

class PhLtcReader : public QObject
{
    Q_OBJECT

public:
    explicit PhLtcReader(QObject *parent = 0);

    bool init(QString _input="");
    void close();

    static QList<QString> inputList();


private slots:
    void onNotify();

private:
    QAudioInput *_input;
    qint64 _position;
    QIODevice * _buffer;
};

#endif // PHLTCREADER_H

您的問題的答案可以在輸出中找到。

有一句話說

Default format not supported, try to use nearest format

這是問題所在。 您的平台不支持您使用的格式。 嘗試不同的格式或安裝編解碼器/驅動程序以支持該格式。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM