女人被狂躁到高潮视频免费无遮挡,内射人妻骚骚骚,免费人成小说在线观看网站,九九影院午夜理论片少妇,免费av永久免费网址

當前位置:首頁 > 單片機 > 碼農(nóng)愛學習
[導讀]本編利用Qt實現(xiàn)一個網(wǎng)絡攝像頭功能,包含一個服務端和一個客戶端,服務端用于將USB攝像頭轉(zhuǎn)換為一個IP攝像頭,當有客戶端連接時,將其捕獲到的圖像通過TCP發(fā)送出去;客戶端運行在Linux板子上,用于查看攝像頭的實時畫面。


本編利用Qt實現(xiàn)一個網(wǎng)絡攝像頭功能,包含一個服務端和一個客戶端,服務端用于將USB攝像頭轉(zhuǎn)換為一個IP攝像頭,當有客戶端連接時,將其捕獲到的圖像通過TCP發(fā)送出去;客戶端運行在Linux板子上,用于查看攝像頭的實時畫面。

1 必備基礎(chǔ)知識

本篇需要編寫一個服務器和客戶端,關(guān)于TCP服務器/客戶端的基礎(chǔ)知識,可參考這篇:socket套接字基礎(chǔ)

注意,Qt中對Socket的操作進行了進一步的封裝,其基本思想還是一樣的。

下面就來看一下Qt中如何實現(xiàn)TCP Socket通信。

1.1 QTcpSocket與QTcpServer

  • QTcpSocket,在Qt中,Socket被封裝成了QTcpSocket,可以用它實現(xiàn)TCP客戶端的功能,以及服務端接收到客戶端后,對客戶端的處理。

  • QTcpServer,對于TCP服務端的功能,可以使用QTcpServer來完成。

這里整理Qt中TCP Socket的使用方法,配合Qt的信號與槽機制,即可實現(xiàn)服務端/客戶端數(shù)據(jù)的收發(fā)處理。

1.2 QCamera相關(guān)

  • QCamer,獲取當前系統(tǒng)可用的攝像頭 類似獲取串口

  • QCamerInfo,獲取當前系統(tǒng)可用的攝像頭 類似獲取串口

  • QCameraViewfinder,取景框類,攝像頭的實時畫面顯示到這個里面

  • QCameraImageCapture,圖像錄制類,與QCamer 配合使用可進行拍照

2 Win平臺上測試

首先在Windows平臺上用Qt Creator編寫服務端和客戶端程序,并運行測試。

2.1 服務器端

先來看下服務器端的最終效果:

  • 左側(cè)是攝像頭的顯示界面
  • 可以切換不同的攝像頭作為視頻源(筆記本自帶的攝像頭與USB外接的攝像頭)
  • 可以切換攝像頭的顯示分辨率
  • 可以選擇開啟或關(guān)閉攝像頭的IP服務

2.1.1 攝像頭畫面顯示

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);

    QComboBox *pCamType = new QComboBox();
    m_pComboBox = ui->cbBox_resolution;
    pCamType = ui->cbBox_cameras;
    pCamType->clear();

    cameraList = QCameraInfo::availableCameras();
    foreach(const QCameraInfo &cameraInfo, cameraList)
    {
        qDebug() << "CameraInfo:" << cameraInfo; pCamType->addItem(cameraInfo.description());
    }

    m_pCamViewFind = new QCameraViewfinder(this);
    m_pCamViewFind->setGeometry(10, 10, W, H);
    m_pCamViewFind->show();

    m_pCam = new QCamera(this);
    m_pCam->setViewfinder(m_pCamViewFind);
    m_pCam->start();
}

2.1.2 創(chuàng)建Socket服務

void Widget::on_btn_IPServer_toggled(bool checked) { if (checked)
    {
        m_pServer = new QTcpServer(this); if (!m_pServer->listen(QHostAddress::Any, 12345))
        {
            QMessageBox::critical(this, "error", "listen port failed"); exit(0);
        }
        qDebug() << "start IP server";

        m_pTimer = new QTimer(this);
        connect(m_pServer, SIGNAL(newConnection()), this, SLOT(new_client()));
        connect(m_pTimer, SIGNAL(timeout()), this, SLOT(timer_slot()));
        m_pTimer->start(100);

        ui->btn_IPServer->setText("關(guān)閉IP服務");
    } else {
        qDebug() << "stop IP server";
        m_pServer->close(); delete m_pServer;

        ui->btn_IPServer->setText("開啟IP服務");
    }
}

2.1.3 讀取圖像并發(fā)送給客戶端

先定義一下圖像傳送結(jié)構(gòu)體和傳送狀態(tài):

enum TransStatus{
    TS_IDLE, //空閑(圖像數(shù)據(jù)可以更新) TS_RUNNING, //圖像數(shù)據(jù)傳輸中(還不可以更新圖像數(shù)據(jù)) TS_FIRST_DATA, //需要發(fā)出圖像數(shù)據(jù)的第一部分 }; class ImgData { public: char data[LEN] = {0}; //圖像數(shù)據(jù) int totalLen = 0; //圖像大小 int hasSentLen = 0; //已發(fā)出的數(shù)據(jù)長度 TransStatus  stats = TS_IDLE; //工作狀態(tài) };

具體的實現(xiàn)過程:

void Widget::read_data() {
    QString str = m_pClient->readAll();
    ImgData *pData = (ImgData*)m_pClient->userData(0); QString s("newImage:%1"); if (str == "new_request")
    {
        qDebug() << "read_data, new_request, d->len:" << pData->totalLen << "d->stats:" << pData->stats; if ((pData->totalLen > 0) && (pData->stats==TS_IDLE)) //圖像大小不為0,表示已更新圖像數(shù)據(jù)了 {
            pData->stats = TS_RUNNING;
            m_pClient->write(s.arg(pData->totalLen).toUtf8());
            pData->hasSentLen = 0;
        } else //圖像數(shù)據(jù)還沒有更新 {
            pData->stats = TS_FIRST_DATA; //在定時器的槽函數(shù)里發(fā)出"newImage..." }
    } else if (str == "ack")
    { int len_send = P_LEN; //本次需要發(fā)送的長度 if (pData->hasSentLen >= pData->totalLen) //如果圖像已傳輸完畢 {
            qDebug() << "read_data, send done! lenSent:" << pData->hasSentLen << "len" << pData->totalLen; return;
        } // 最后1包數(shù)據(jù)(不滿P_LEN) if ((pData->hasSentLen + P_LEN) > pData->totalLen)
        {
            len_send = pData->totalLen - pData->hasSentLen;
        }

        qDebug() << "read_data, ack, write len:" << len_send; // 發(fā)送數(shù)據(jù) pData->hasSentLen += m_pClient->write(pData->data + pData->hasSentLen, len_send); if (pData->hasSentLen >= pData->totalLen)
        {
            pData->stats = TS_IDLE; //傳輸完畢后,把狀態(tài)改為可更新 pData->totalLen = 0;
        }
    }
}

需要注意的是,圖像是需要分包傳送的,最后一包一般都不是設定的最大長度,需要計算一下最后一包的數(shù)據(jù)長度。

2.2 客戶端

先來看下客戶端的最終效果:

  • 右側(cè)是攝像頭畫面的顯示框
  • 可以修改要連接的服務端的IP地址
  • 可以選擇開啟或關(guān)閉網(wǎng)絡攝像頭

2.2.1 創(chuàng)建Socket連接

void Widget::on_pushButton_toggled(bool checked) { if (checked)
    {
        QString ip = ui->lineEdit->text();
        m_pSocket->connectToHost(ip, 12345); if (!m_pSocket->waitForConnected(1000))
        {
            QMessageBox::critical(this, "error", "server connection failed"); return;
        }

        ui->pushButton->setText("關(guān)閉");
        m_iRecvLen = 0;
        m_pSocket->write("new_request");
        qDebug("on_bnt_connect_clicked, new_request");
    } else {
        m_pSocket->close();
        ui->pushButton->setText("打開");
    }
}

2.3.2 接收服務端的圖像

void Widget::read_data() { int ret;
    QTime qTime; static int i = 0;

    ret = m_pSocket->read(m_pData + m_iRecvLen, P_LEN); if (0 == strncmp("newImage", m_pData + m_iRecvLen, 8))
    {
        m_iImgLen = atoi(m_pData + m_iRecvLen + 9);
        i++;
    } else {
        m_iRecvLen += ret; if (m_iRecvLen >= m_iImgLen)
        {
            QString timestamp = QString::number(QDateTime::currentMSecsSinceEpoch());
            update(); return;
        }
    } //圖像傳輸完畢 m_pSocket->write("ack");
}

2.3.3 將圖像顯示出來

void Widget::paintEvent(QPaintEvent *event) {
    QPixmap map; if ((m_iRecvLen >= m_iImgLen) && (m_iImgLen > 0))
    { map.loadFromData((uchar *)m_pData, m_iImgLen); QPainter p(this);
        p.drawPixmap(140, 0, 640, 480, map);
        m_pSocket->write("new_request");
        m_iRecvLen = 0;
    }
}

3 嵌入式Linux平臺上測試

3.1 交叉編譯

將客戶端程序的源代碼拷貝到Ubunu中進行交叉編譯,具體編譯過程可參考之前的文章:

嵌入式Qt-動手編寫并運行自己的第1個ARM-Qt程序

本篇的實驗環(huán)境,繼續(xù)使用的是燒錄了野火i.MX6ULL自帶的系統(tǒng)固件Linux板子,需要通過SSH的方式將編譯的程序再發(fā)送到板子中,SSH傳輸文件的操作可參考上篇文章:

嵌入式Qt-控制硬件:滑動條控制RGB燈

3.2 實驗演示

4 總結(jié)

本篇介紹了如何用Qt實現(xiàn)一個網(wǎng)絡攝像頭功能,通過服務端將USB攝像頭轉(zhuǎn)換為一個IP攝像頭,Linux板子中的客戶端來連接服務器,將攝像頭的實時畫面顯示出來。

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

自2014年ST公司推出STM32CubeMX以來,這款圖形化配置工具憑借“一鍵生成初始化代碼”“跨IDE兼容”“中間件集成”等特性,迅速成為78%的STM32開發(fā)者首選工具。然而,伴隨其普及的爭議始終未息:STM32C...

關(guān)鍵字: STM32CubeMX ST公司

在工業(yè)自動化領(lǐng)域,Modbus協(xié)議憑借其開放性和易用性成為設備通信的"通用語言"。然而,當工程師面對Modbus RTU、ASCII和TCP三種變體時,如何根據(jù)具體場景做出最優(yōu)選擇?本文將從編碼機制、通信效率、錯誤檢測等...

關(guān)鍵字: Modbus協(xié)議 TCP

在工業(yè)自動化、能源管理等實時性要求嚴苛的場景中,Modbus通信系統(tǒng)的響應延遲直接關(guān)系到設備控制的精度與系統(tǒng)穩(wěn)定性。從智能電表的功率調(diào)節(jié)到機器人關(guān)節(jié)的同步控制,微秒級的響應偏差都可能引發(fā)連鎖故障。本文從硬件架構(gòu)、軟件設計...

關(guān)鍵字: Modbus 通信系統(tǒng)

在新能源發(fā)電、電動汽車、數(shù)據(jù)中心等直流供電系統(tǒng)中,過壓故障是導致設備損壞的主要誘因之一。據(jù)統(tǒng)計,電力電子設備故障中約35%與過壓事件相關(guān),其中直流側(cè)過壓占比達62%。本文以基于TVS二極管與MOSFET的復合型直流過壓保...

關(guān)鍵字: 直流過壓 保護電路

在工業(yè)物聯(lián)網(wǎng)(IIoT)與邊緣計算快速發(fā)展的背景下,Modbus協(xié)議憑借其輕量化特性成為微控制器(MCU)設備互聯(lián)的首選方案。然而,在資源受限的MCU(如STM32F0系列、ESP8266等,RAM通常小于32KB,F(xiàn)l...

關(guān)鍵字: 微控制器 Modbus 工業(yè)物聯(lián)網(wǎng)

在工業(yè)控制系統(tǒng)中,Modbus RTU協(xié)議的CRC校驗如同通信網(wǎng)絡的"免疫系統(tǒng)",某石化廠DCS系統(tǒng)曾因CRC計算錯誤導致0.3%的數(shù)據(jù)包丟失,引發(fā)連鎖控制故障。本文將深入解析CRC-16/MODBUS算法原理,對比軟件...

關(guān)鍵字: Modbus RTU CRC 算法

在工業(yè)自動化領(lǐng)域,Modbus協(xié)議憑借其簡潔高效的設計,已成為設備間通信的"通用語言"。某智能電網(wǎng)項目通過Modbus RTU協(xié)議實現(xiàn)2000臺電表的數(shù)據(jù)采集,通信成功率高達99.97%,這背后正是對消息結(jié)構(gòu)的精準把控。...

關(guān)鍵字: Modbus 工業(yè)自動化

在工業(yè)物聯(lián)網(wǎng)設備開發(fā)中,Modbus從站功能已成為微控制器(MCU)的標配能力。某智能電表項目通過在STM32上實現(xiàn)Modbus RTU從站,成功將設備接入現(xiàn)有SCADA系統(tǒng),開發(fā)周期縮短40%。本文將系統(tǒng)解析MCU實現(xiàn)...

關(guān)鍵字: 微控制器 Modbus 協(xié)議棧優(yōu)化

在嵌入式系統(tǒng)中,F(xiàn)lash存儲器因其非易失性、高密度和低成本特性,成為代碼存儲和關(guān)鍵數(shù)據(jù)保存的核心組件。然而,MCU驅(qū)動Flash讀寫時,開發(fā)者常因?qū)τ布匦岳斫獠蛔慊虿僮髁鞒淌韬觯萑胄阅芟陆?、?shù)據(jù)損壞甚至硬件損壞的陷...

關(guān)鍵字: MCU驅(qū)動 Flash

在嵌入式開發(fā)中,STM32的時鐘系統(tǒng)因其靈活性和復雜性成為開發(fā)者關(guān)注的焦點。然而,看似簡單的時鐘配置背后,隱藏著諸多易被忽視的陷阱,輕則導致系統(tǒng)不穩(wěn)定,重則引發(fā)硬件損壞。本文從時鐘源選擇、PLL配置、總線時鐘分配等關(guān)鍵環(huán)...

關(guān)鍵字: STM32 時鐘系統(tǒng)
關(guān)閉