新聞中心
這篇文章主要講解了“Qt怎么實(shí)現(xiàn)人臉識(shí)別客戶(hù)端”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Qt怎么實(shí)現(xiàn)人臉識(shí)別客戶(hù)端”吧!
專(zhuān)注于為中小企業(yè)提供成都網(wǎng)站設(shè)計(jì)、做網(wǎng)站服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)綏江免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動(dòng)了上千家企業(yè)的穩(wěn)健成長(zhǎng),幫助中小企業(yè)通過(guò)網(wǎng)站建設(shè)實(shí)現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。
一、前言
人臉識(shí)別客戶(hù)端程序,不需要和人臉識(shí)別相關(guān)的庫(kù)在一起,而是通過(guò)協(xié)議通信來(lái)和人臉識(shí)別服務(wù)端通信交互,人臉識(shí)別客戶(hù)端和服務(wù)端程序框架,主要是為了提供一套通用的框架,按照定好的協(xié)議,實(shí)現(xiàn)人臉識(shí)別的相關(guān)處理,很多廠家都會(huì)有也都會(huì)做類(lèi)似的機(jī)制,以便第三方廠家或者自家的其他設(shè)備按照這個(gè)通信協(xié)議來(lái)處理,比如客戶(hù)端程序可以在PC機(jī)上,也可以是網(wǎng)頁(yè),還可以是安卓客戶(hù)端,前端設(shè)備比如人工訪客機(jī),訪客機(jī)本地是不需要做人臉識(shí)別等處理的,而是發(fā)送到服務(wù)端處理完以后再拿到結(jié)果進(jìn)行展示,這樣就可以利用服務(wù)端強(qiáng)大的運(yùn)算能力。
自定義人臉識(shí)別協(xié)議功能:
離線使用,同時(shí)支持百度的離線包和嵌入式linux人臉識(shí)別靜態(tài)庫(kù)。
支持多個(gè)連接并發(fā),自動(dòng)排隊(duì)處理,返回的時(shí)候帶上唯一標(biāo)識(shí)區(qū)分。
傳入單張圖片返回人臉區(qū)域。
傳入單張圖片返回人臉特征值。
傳入單張圖片或者多張圖片返回是否是活體。
傳入兩張圖片返回比對(duì)結(jié)果。
傳入兩個(gè)特征值返回比對(duì)結(jié)果。
傳入單張圖片添加人臉。
指定唯一標(biāo)識(shí)符刪除人臉。
傳入單張照片返回相似度最大的人臉信息。
修改人臉?lè)?wù)的配置參數(shù)比如是否快速查找、人臉占比等。
二、功能特點(diǎn)
支持的功能包括人臉識(shí)別、人臉比對(duì)、人臉?biāo)阉鳌⒒铙w檢測(cè)等。
在線版還支持身份證、駕駛證、行駛證、銀行卡等識(shí)別。
在線版的協(xié)議支持百度、曠視,離線版的支持百度,可定制。
除了支持X86架構(gòu),還支持嵌入式linux比如contex-A9、樹(shù)莓派等。
每個(gè)功能的執(zhí)行除了返回結(jié)果還返回執(zhí)行用時(shí)時(shí)間。
多線程處理,通過(guò)type控制當(dāng)前處理類(lèi)型。
支持單張圖片檢索相似度最高的圖片。
支持指定目錄圖片用來(lái)生成人臉特征值文件。
可設(shè)置等待處理圖片隊(duì)列中的數(shù)量。
每次執(zhí)行都有成功或者失敗的信號(hào)返回。
人臉?biāo)阉鞯姆祷亟Y(jié)果包含了原圖+最大相似度圖+相似度等。
人臉比對(duì)同時(shí)支持兩張圖片和兩個(gè)特征值比對(duì)。
相關(guān)功能自定義一套協(xié)議用于客戶(hù)端和服務(wù)端,可以通過(guò)TCP通信進(jìn)行交互。
自定義人臉識(shí)別協(xié)議非常適用于中心一臺(tái)服務(wù)器,現(xiàn)場(chǎng)若干設(shè)備請(qǐng)求的場(chǎng)景。
每個(gè)模塊全部是獨(dú)立的一個(gè)類(lèi),代碼整潔、注釋完善。
三、效果圖
四、核心代碼
FaceSdkClient::FaceSdkClient(QObject *parent) : QObject(parent) { isOk = false; timeout = 10; lastTime = QDateTime::currentDateTime(); //隨機(jī)生成IP用于測(cè)試 clientIP = QString("192.168.1.%1").arg(qrand() % 255); serverIP = "192.168.1.239"; serverPort = 6662; //實(shí)例化連接對(duì)象 tcpSocket = new QTcpSocket(this); connect(tcpSocket, SIGNAL(connected()), this, SLOT(connected())); connect(tcpSocket, SIGNAL(disconnected()), this, SLOT(disconnected())); connect(tcpSocket, SIGNAL(readyRead()), this, SLOT(readData())); //定時(shí)器處理數(shù)據(jù) timerData = new QTimer(this); connect(timerData, SIGNAL(timeout()), this, SLOT(checkData())); timerData->setInterval(100); //定時(shí)器處理重連 timerConn = new QTimer(this); connect(timerConn, SIGNAL(timeout()), this, SLOT(checkConn())); timerConn->setInterval(3000); //定時(shí)器發(fā)送心跳 timerHeart = new QTimer(this); connect(timerHeart, SIGNAL(timeout()), this, SLOT(sendHeart())); timerHeart->setInterval(2000); } void FaceSdkClient::connected() { isOk = true; sendHeart(); } void FaceSdkClient::disconnected() { isOk = false; emit receiveAnaly(serverIP, "服務(wù)器斷開(kāi)"); } void FaceSdkClient::checkConn() { QDateTime now = QDateTime::currentDateTime(); if (lastTime.secsTo(now) >= timeout) { isOk = false; start(); emit receiveAnaly(serverIP, "重連服務(wù)器"); } } void FaceSdkClient::readData() { lastTime = QDateTime::currentDateTime(); QMutexLocker locker(&mutex); buffer.append(tcpSocket->readAll()); } void FaceSdkClient::checkData() { QMutexLocker locker(&mutex); QDomDocument dom; if (!DeviceFun::getReceiveXmlData(buffer, dom, "IFACE:", 10)) { return; } //逐個(gè)取出節(jié)點(diǎn)判斷數(shù)據(jù) emit receiveData(serverIP, dom.toString()); QDomElement element = dom.documentElement(); if (element.tagName() == "FaceServer") { //查找時(shí)間字段屬性 QString nowTime = element.attribute("NowTime"); QString year, month, day, hour, min, sec; DeviceFun::getNowTime(nowTime, year, month, day, hour, min, sec); //QUIHelper::setSystemDateTime(year, month, day, hour, min, sec); //根據(jù)子節(jié)點(diǎn)處理 QDomNode childNode = element.firstChild(); QString name = childNode.nodeName(); QString value = element.text(); //qDebug() << TIMEMS << name << value; //取出faceID和耗時(shí) element = childNode.toElement(); QString faceID = element.attribute("FaceID"); int msec = element.attribute("TimeUsed").toInt(); if (name == "Msg") { if (value.startsWith("Init Face Finsh")) { emit receiveFaceFeatureFinsh(); } else if (value.startsWith("Find Face Rect Fail")) { emit receiveFaceRectFail(faceID); } else if (value.startsWith("Find Face Feature Fail")) { emit receiveFaceFeatureFail(faceID); } else if (value.startsWith("Find Face Compare Fail")) { emit receiveFaceCompareFail(faceID); } else if (value.startsWith("Find Face Live Fail")) { emit receiveFaceLiveFail(faceID); } else if (value.startsWith("Image Size Is Too Large")) { emit receiveAnaly(serverIP, "圖片尺寸過(guò)大"); } else { emit receiveAnaly(serverIP, value); } } else if (name == "FaceRect") { QStringList list = value.split(","); QRect faceRect(list.at(0).toInt(), list.at(1).toInt(), list.at(2).toInt(), list.at(3).toInt()); emit receiveFaceRect(faceID, faceRect, msec); } else if (name == "FaceFeature") { QStringList list = value.split("|"); QListfeature; for (int i = 0; i < 256; i++) { feature << list.at(i).toFloat(); } emit receiveFaceFeature(faceID, feature, msec); } else if (name == "FaceLive") { float result = element.attribute("Result").toFloat(); emit receiveFaceLive(faceID, result, msec); } else if (name == "FaceCompare") { float result = element.attribute("Result").toFloat(); emit receiveFaceCompare(faceID, result, msec); } else if (name == "FaceCompareOne") { QStringList list = value.split("|"); QImage image1 = DeviceFun::getImage(list.at(0)); QImage image2 = DeviceFun::getImage(list.at(1)); QString targetName = element.attribute("TargetName"); float result = element.attribute("Result").toFloat(); emit receiveFaceCompareOne(faceID, image1, image2, targetName, result, msec); } else if (name == "FaceConfig") { QString findFast = element.attribute("FindFast"); QString facePercent = element.attribute("FacePercent"); QString info = QString("模式: %1 比例: %2").arg(findFast == "true" ? "快速模式" : "正常模式").arg(facePercent); emit receiveAnaly(serverIP, "人臉配置 " + info); } else { emit receiveAnaly(serverIP, "心跳回復(fù)"); } } } void FaceSdkClient::setClientIP(const QString &clientIP) { this->clientIP = clientIP; } void FaceSdkClient::setServerIP(const QString &serverIP) { this->serverIP = serverIP; } void FaceSdkClient::setServerPort(int serverPort) { this->serverPort = serverPort; } bool FaceSdkClient::start() { tcpSocket->abort(); tcpSocket->connectToHost(serverIP, serverPort); timerData->start(); timerConn->start(); timerHeart->start(); return true; } void FaceSdkClient::stop() { tcpSocket->disconnectFromHost(); timerData->stop(); timerConn->stop(); timerHeart->stop(); } void FaceSdkClient::sendData(const QString &body) { if (!isOk) { return; } //構(gòu)建xml字符串 QStringList list; list.append(QString(" ").arg(clientIP)); list.append(body); list.append(" "); QString data = DeviceFun::getSendXmlData(list.join(""), "IFACE:"); tcpSocket->write(data.toUtf8()); emit sendData(clientIP, data); } void FaceSdkClient::sendHeart() { if (!isOk) { return; } sendData(""); emit sendAnaly(clientIP, "發(fā)送心跳"); } void FaceSdkClient::sendFindFace(const QString &faceID, const QImage &image) { if (!isOk) { return; } QString data = QString(" %2 ").arg(faceID).arg(DeviceFun::getImageData(image)); sendData(data); emit sendAnaly(clientIP, "請(qǐng)求人臉區(qū)域"); } void FaceSdkClient::sendFindFeature(const QString &faceID, const QImage &image) { if (!isOk) { return; } QString data = QString("%2 ").arg(faceID).arg(DeviceFun::getImageData(image)); sendData(data); emit sendAnaly(clientIP, "請(qǐng)求人臉特征"); } void FaceSdkClient::sendFindLive(const QString &faceID, const QImage &image) { if (!isOk) { return; } QString data = QString("%2 ").arg(faceID).arg(DeviceFun::getImageData(image)); sendData(data); emit sendAnaly(clientIP, "請(qǐng)求活體檢測(cè)"); } void FaceSdkClient::sendCompareByImage(const QString &faceID1, const QImage &image1, const QString &faceID2, const QImage &image2) { if (!isOk) { return; } QStringList list; list.append(""); list.append(QString(" "); sendData(list.join("")); emit sendAnaly(clientIP, "請(qǐng)求人臉比對(duì)"); } void FaceSdkClient::sendCompareByFeature(const QString &faceID1, const QString &feature1, const QString &faceID2, const QString &feature2) { if (!isOk) { return; } QStringList list; list.append("%2 ").arg(faceID1).arg(DeviceFun::getImageData(image1))); list.append(QString("%2 ").arg(faceID2).arg(DeviceFun::getImageData(image2))); list.append(""); list.append(QString(" "); sendData(list.join("")); emit sendAnaly(clientIP, "請(qǐng)求人臉比對(duì)"); } void FaceSdkClient::sendAppendFace(const QString &faceID, const QImage &image) { if (!isOk) { return; } QString data = QString("%2 ").arg(faceID1).arg(feature1)); list.append(QString("%2 ").arg(faceID2).arg(feature2)); list.append("%2 ").arg(faceID).arg(DeviceFun::getImageData(image)); sendData(data); emit sendAnaly(clientIP, "添加人臉"); } void FaceSdkClient::sendDeleteFace(const QString &faceID) { if (!isOk) { return; } QString data = QString("").arg(faceID); sendData(data); emit sendAnaly(clientIP, "刪除人臉"); } void FaceSdkClient::sendFindByImage(const QString &faceID, const QImage &image) { if (!isOk) { return; } QString data = QString(" %2 ").arg(faceID).arg(DeviceFun::getImageData(image)); sendData(data); emit sendAnaly(clientIP, "查找最大人臉"); } void FaceSdkClient::sendFindByFeature(const QString &faceID, const QString &feature) { if (!isOk) { return; } QString data = QString("%2 ").arg(faceID).arg(feature); sendData(data); emit sendAnaly(clientIP, "查找最大人臉"); }
感謝各位的閱讀,以上就是“Qt怎么實(shí)現(xiàn)人臉識(shí)別客戶(hù)端”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Qt怎么實(shí)現(xiàn)人臉識(shí)別客戶(hù)端這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!
文章名稱(chēng):Qt怎么實(shí)現(xiàn)人臉識(shí)別客戶(hù)端
分享路徑:http://www.dlmjj.cn/article/jieehc.html