Java實(shí)現(xiàn)人臉識別登錄、注冊等功能(最新完整版)
前言:
這段時(shí)間由于學(xué)校實(shí)行靜態(tài)化管理,寢室門和校門都是用了人臉識別的裝置,每次經(jīng)過都會(huì)激發(fā)我的好奇心,也想自己搞一個(gè)人臉識別玩玩,隨著開始查找資料來研究這方面的信息,還好有好幾家公司都有提供這方面的免費(fèi)API,也是省下來很多功夫。一開始采用的是face++,但是在執(zhí)行到最后一步人臉?biāo)阉鲿r(shí)出現(xiàn)問題,一直提示INVALID_OUTER_ID,跟著官方文檔,一步步抽離再封裝,最終還是以失敗告終,無奈只能選擇放棄。接著輾轉(zhuǎn)第二家 百度AI ,這次還是比較順利的,中間只出現(xiàn)過一次錯(cuò)誤 ,而且官方大大還給出了解決方案,很是貼心,最終還是實(shí)現(xiàn)了開始的預(yù)想:成功使用人臉來實(shí)現(xiàn)注冊和登錄功能。
本系列項(xiàng)目從設(shè)計(jì)到實(shí)現(xiàn)源碼全部開源免費(fèi)學(xué)習(xí)使用,一起追向理想,歡迎各位大佬監(jiān)督打卡開發(fā)!
難度分析:
由于使用的是百度云AI
/face++
這兩個(gè)平臺的現(xiàn)有的Api
接口,所以分別對二者的難度進(jìn)行分析介紹:
- face++
? 曠視face++
的難度在于需要自己對官方提供的代碼段進(jìn)行獨(dú)立封裝,這就比較考驗(yàn)開發(fā)者的代碼抽取能力和 自我獨(dú)立封裝的技能,如果不能很好地進(jìn)行抽取封裝,就會(huì)造成代碼的冗余以及內(nèi)存的浪費(fèi),這樣封裝的效 果并不能達(dá)到最佳。比較耗費(fèi)時(shí)間,在抽取face++
平臺的代碼段時(shí)需要對IO流有著比較好的理解,從而讀取 讀取配置文件中的內(nèi)容,這里就不多做介紹。
- 百度云AI
? 百度云AI
可以讓開發(fā)者對開發(fā)難度有所選擇,因?yàn)樵撈脚_提供了兩種方式:第一種方式類似于face++
需要抽 取代碼然后進(jìn)行代碼封裝,難度分析見上述face++
對分析;第二種方式則是百度云提供了Maven
倉庫,可以 直接導(dǎo)入依賴,直接調(diào)用相應(yīng)的Api
進(jìn)行開發(fā)即可!有較好的選擇性!
前端使用JQuery
調(diào)用本地?cái)z像頭進(jìn)行拍攝(自我感覺這是最難的部分)
項(xiàng)目回顧(百度云AI)
最終效果演示:
技術(shù)選型:
- SpringBoot
- BootStrap
- Thymeleaf
- 百度云AI / Face++
項(xiàng)目需求分析
為了用戶登錄的便捷,不再輸入賬號密碼進(jìn)行手動(dòng)登錄與注冊,而是使用JQuery
調(diào)用本機(jī)的攝像頭進(jìn)行拍攝照片,然后調(diào)用人臉識別接口將人臉信息自動(dòng)注冊進(jìn)所使用的平臺(百度云/Face++)
后,用戶即可進(jìn)行人臉掃描實(shí)現(xiàn)登錄操作。
項(xiàng)目搭建
1. 前期準(zhǔn)備 ① 進(jìn)入百度云的人臉識別控制臺
如果沒有百度賬號可以使用手機(jī)號快速進(jìn)行注冊進(jìn)入百度云AI控制臺
②創(chuàng)建人臉識別應(yīng)用
【1】創(chuàng)建應(yīng)用
【2】可以選擇自己想要使用的接口:比如人臉識別、語音技術(shù)等,本次項(xiàng)目采用的是人臉識別,官方也默認(rèn)選擇了人臉識別的全部接口,所以也不需要做改動(dòng),只需要填寫應(yīng)用名稱即可!
【3】獲取秘鑰
2. 測試百度云API
① 導(dǎo)入依賴
<dependency> <groupId>com.baidu.aip</groupId> <artifactId>java-sdk</artifactId> <version>4.9.0</version> </dependency>
② 測試部分API(人臉注冊、人臉檢測、人臉?biāo)阉鞯龋?/p>
人臉注冊
用于從人臉庫中新增用戶,可以設(shè)定多個(gè)用戶所在組及組內(nèi)用戶的人臉圖片
典型應(yīng)用場景:構(gòu)建屬于自己人臉庫,比如:會(huì)員人臉注冊、已有用戶補(bǔ)全人臉信息
測試代碼
//人臉注冊 @Test public void testFaceRegister() throws IOException { //1. 創(chuàng)建Java代碼和百度云交互的Client對象 AipFace client = new AipFace("AppId","Api_key","Api_secret"); //2. 參數(shù)設(shè)置(示例下表格對參數(shù)進(jìn)行介紹) HashMap<String,String> map = new HashMap<>(); map.put("quality_control","NORMAL");//圖片質(zhì)量 map.put("liveness_control","LOW");//活體檢測 //3. 構(gòu)造圖片 String path = "本地圖片路徑"; //上傳的圖片 兩種格式:url地址 Base64字符串形式 byte[] bytes = Files.readAllBytes(Paths.get(path)); String encode = Base64Util.encode(bytes); //4.調(diào)用api方法完成人臉注冊 /** * 參數(shù)1:圖片的url或者base64字符串 * 參數(shù)2:圖片形式(URL,BASE64) * 參數(shù)3:組Id(固定一個(gè)字符串) * 參數(shù)4:用戶Id * 參數(shù)5:hashMap基本參數(shù)配置 */ JSONObject res = client.addUser(encode, "BASE64", "pdx", "1000", map); System.out.println(res.toString()); }
測試結(jié)果:只要最后error_code為0則表示測試成功,后續(xù)封裝代碼也是需要判斷error_code的值
參數(shù)介紹:
參數(shù)名稱 | 是否必選 | 類型 | 默認(rèn)值 | 說明 |
---|---|---|---|---|
image | 是 | String | 圖片信息(總數(shù)據(jù)大小應(yīng)小于10M),圖片上傳方式根據(jù)image_type來判斷 | |
image_type | 是 | String | 圖片類型BASE64:圖片的base64值,base64編碼后的圖片數(shù)據(jù),編碼后的圖片大小不超過2M;URL:圖片的 URL地址( 可能由于網(wǎng)絡(luò)等原因?qū)е孪螺d圖片時(shí)間過長);FACE_TOKEN: 人臉圖片的唯一標(biāo)識,調(diào)用人臉檢測接口時(shí),會(huì)為每個(gè)人臉圖片賦予一個(gè)唯一的FACE_TOKEN,同一張圖片多次檢測得到的FACE_TOKEN是同一個(gè)。 | |
face_field | 否 | string | 包age,expression,face_shape,gender,glasses,landmark,landmark150,quality,eye_status,emotion,face_type,mask,spoofing信息逗號分隔. 默認(rèn)只返回face_token、人臉框、概率和旋轉(zhuǎn)角度 | |
max_face_num | 否 | uint32 | 最多處理人臉的數(shù)目,默認(rèn)值為1,根據(jù)人臉檢測排序類型檢測圖片中排序第一的人臉(默認(rèn)為人臉面積最大的人臉),最大值120 | |
face_type | 否 | string | 人臉的類型LIVE表示生活照:通常為手機(jī)、相機(jī)拍攝的人像圖片、或從網(wǎng)絡(luò)獲取的人像圖片等 IDCARD表示身份證芯片照:二代身份證內(nèi)置芯片中的人像照片WATERMARK表示帶水印證件照:一般為帶水印的小圖,如公安網(wǎng)小圖CERT表示證件照片:如拍攝的身份證、工卡、護(hù)照、學(xué)生證等證件圖片默認(rèn)LIVE |
人臉檢測
判斷圖片是否具有面部信息
測試代碼
@Test public void testFaceCheck() throws IOException { //1. 創(chuàng)建Java代碼和百度云交互的Client對象 AipFace client = new AipFace("AppId","Api_key","Api_secret"); //2. 構(gòu)造圖片 String path = "本地圖片路徑"; //上傳的圖片 兩種格式:url地址 Base64字符串形式 byte[] bytes = Files.readAllBytes(Paths.get(path)); String encode = Base64Util.encode(bytes); //調(diào)用Api方法進(jìn)行人臉檢測 /** * 參數(shù)1:圖片的url或者base64字符串 * 參數(shù)2:圖片形式(URL,BASE64) * 參數(shù)3:hashMap中的基本參數(shù)配置(null:使用默認(rèn)配置) */ JSONObject res = client.detect(encode, "BASE64", null); System.out.println(res.toString(2)); }
測試結(jié)果:
參數(shù)介紹:
參數(shù)名稱 | 是否必選 | 類型 | 默認(rèn)值 | 說明 |
---|---|---|---|---|
image | 是 | String | 圖片信息(總數(shù)據(jù)大小應(yīng)小于10M),圖片上傳方式根據(jù)image_type來判斷 | |
image_type | 是 | String | 圖片類型BASE64:圖片的base64值,base64編碼后的圖片數(shù)據(jù),編碼后的圖片大小不超過2M;URL:圖片的 URL地址( 可能由于網(wǎng)絡(luò)等原因?qū)е孪螺d圖片時(shí)間過長);FACE_TOKEN: 人臉圖片的唯一標(biāo)識,調(diào)用人臉檢測接口時(shí),會(huì)為每個(gè)人臉圖片賦予一個(gè)唯一的FACE_TOKEN,同一張圖片多次檢測得到的FACE_TOKEN是同一個(gè)。 | |
face_field | 否 | string | 包age,expression,face_shape,gender,glasses,landmark,landmark150,quality,eye_status,emotion,face_type,mask,spoofing信息逗號分隔. 默認(rèn)只返回face_token、人臉框、概率和旋轉(zhuǎn)角度 | |
max_face_num | 否 | uint32 | 最多處理人臉的數(shù)目,默認(rèn)值為1,根據(jù)人臉檢測排序類型檢測圖片中排序第一的人臉(默認(rèn)為人臉面積最大的人臉),最大值120 | |
face_type | 否 | string | 人臉的類型LIVE表示生活照:通常為手機(jī)、相機(jī)拍攝的人像圖片、或從網(wǎng)絡(luò)獲取的人像圖片等 IDCARD表示身份證芯片照:二代身份證內(nèi)置芯片中的人像照片WATERMARK表示帶水印證件照:一般為帶水印的小圖,如公安網(wǎng)小圖CERT表示證件照片:如拍攝的身份證、工卡、護(hù)照、學(xué)生證等證件圖片默認(rèn)LIVE |
人臉?biāo)阉?/strong>
根據(jù)用戶上傳的圖片和指定人臉庫中的所有人臉進(jìn)行比較,獲取相似度最高的一個(gè)或者某幾個(gè)的評分
返回值(數(shù)據(jù),只需要第一條,相似度最高的數(shù)據(jù))score
:相似度評分(80分以上可以認(rèn)為是同一個(gè)人)
測試代碼:
@Test public void testFaceSearch() throws IOException { //1. 創(chuàng)建Java代碼和百度云交互的Client對象 AipFace client = new AipFace("AppId","Api_key","Api_secret"); //2. 構(gòu)造圖片 String path = "本地圖片路徑"; //上傳的圖片 兩種格式:url地址 Base64字符串形式 byte[] bytes = Files.readAllBytes(Paths.get(path)); String encode = Base64Util.encode(bytes); //人臉?biāo)阉? JSONObject res = client.search(encode, "BASE64", "pdx", null); System.out.println(res.toString(2)); }
測試結(jié)果:
小結(jié):
整體測試下來,和平常對數(shù)據(jù)庫的操作類似,都是增刪改查的操作,除了這些Api
,還有其他的就不再贅述了,操作類似!所有測試代碼也會(huì)同項(xiàng)目案例一并上傳到Gitee倉庫中!
3. 搭建項(xiàng)目(使用Thymeleaf模板引擎)
① 創(chuàng)建Maven項(xiàng)目
引入相關(guān)依賴,構(gòu)建所需文件目錄
② 編寫properties配置文件
ai.appId="api_id" ai.apiKey="api_key" ai.secretKey="api_secret" ai.imageType=BASE64 ai.groupId="自定義組" server.max-http-header-size=1000KB spring.thymeleaf.cache=false
③ 展示對上述部分Api接口進(jìn)行二次封裝
private AipFace client; private HashMap<String,String> map = new HashMap<>(); private BaiduAiUtils(){ map.put("quality_control","NORMAL");//圖片質(zhì)量 map.put("liveness_control","LOW");//活體檢測 } @PostConstruct public void init(){ client = new AipFace(APP_ID,API_KEY,SECRET_KEY); } /** * 人臉注冊,將用戶照片存入人臉庫中 * @param userId * @param image * @return */ public Boolean faceRegister(String userId,String image){ //人臉注冊 JSONObject res = client.addUser(image, IMAGE_TYPE, groupId, userId, map); Integer errorCode = res.getInt("error_code"); return errorCode == 0? true : false; } /** * 人臉更新,更新人臉庫中的用戶照片 * @param userId * @param image * @return */ public Boolean faceUpdate(String userId,String image){ //人臉更新 JSONObject res = client.updateUser(image, IMAGE_TYPE, groupId, userId, map); Integer errorCode = res.getInt("error_code"); return errorCode == 0 ? true : false; }
想要了解全部封裝代碼請移步Gittee倉庫
④ 編寫Controller前端控制器代碼
首選創(chuàng)建一個(gè)FaceLoginController
類
跳轉(zhuǎn)到人臉登錄頁面的控制器
@RequestMapping("/") public String toLogin(){ return "index"; }
實(shí)現(xiàn)人臉登錄邏輯
/** * 人臉登錄 * @return * @throws Exception */ @RequestMapping("/face-login") @ResponseBody public String searchface(@RequestBody @RequestParam(name = "imagebast64") StringBuffer imagebast64, HttpServletRequest request) throws Exception { String userId = faceLoginService.loginByFace(imagebast64); request.getSession().setAttribute("userId",userId); request.getSession().setAttribute("username","派大星"); return userId; }
實(shí)現(xiàn)人臉登錄業(yè)務(wù)邏輯層
由于在封裝接口時(shí)設(shè)定的imageType為Base64
,所以在實(shí)現(xiàn)前端拍照時(shí)使用的是Canvas Api
提供了toDataURL()方法
將畫布中的圖形轉(zhuǎn)換為圖片,而默認(rèn)情況下,toDataURL()
方法把圖形轉(zhuǎn)變成Base64
編碼格式的png
,其格式為
,而,
后面的內(nèi)容才是接口中需要的部分,所以需要對字符串進(jìn)行切割處理。前端部分具體見下圖:
注意事項(xiàng)(項(xiàng)目搭建前了解)
① 出現(xiàn)qps不足如何解決
Open api qps request limit reached
這個(gè)問題官方也給出了具體的解決方案,在百度云控制臺可以免費(fèi)領(lǐng)取測試額度,也是對開發(fā)者的一個(gè)福利。
② oauth 獲取錯(cuò)誤
0 [main] WARN com.baidu.aip.client.BaseClient - oauth get error, current state: STATE_TRUE_CLOUD_USER
{
"error_msg": "IAM Certification failed",
"error_code": 14
}
具體原因:
仔細(xì)檢查:APP_ID、API_KEY、SECRET_KEY
字符串中是否有空格
項(xiàng)目總結(jié)
使用百度云AI
把項(xiàng)目整體搭建下來,可以很完美的實(shí)現(xiàn)最終的效果,人臉的識別速度也是相當(dāng)迅速的,一些細(xì)節(jié)處理的特別到位,比如:在進(jìn)行人臉識別的過程中眼睛必須對準(zhǔn)攝像頭,當(dāng)你的眼睛有所阻礙時(shí),會(huì)提示的某眼處有阻礙等等。但是在并發(fā)方面支持的并不是很到位!
新鮮出爐的代碼將會(huì)及時(shí)更新到Gitee
倉庫
以上代碼屬于部分實(shí)現(xiàn),想要了解完整版請移步派大星的Gitee倉庫
到此這篇關(guān)于Java實(shí)現(xiàn)人臉登錄、注冊等功能的文章就介紹到這了,更多相關(guān)Java人臉登錄、注冊內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring?boot前后端交互之?dāng)?shù)據(jù)格式轉(zhuǎn)換問題
這篇文章主要介紹了spring?boot前后端交互之?dāng)?shù)據(jù)格式轉(zhuǎn)換,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-01-01springboot集成mybatis?plus和dynamic-datasource注意事項(xiàng)說明
這篇文章主要介紹了springboot集成mybatis?plus和dynamic-datasource注意事項(xiàng)說明,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01Java 實(shí)戰(zhàn)項(xiàng)目錘煉之在線蛋糕商城系統(tǒng)的實(shí)現(xiàn)
讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+jsp+jdbc+mysql實(shí)現(xiàn)一個(gè)在線蛋糕商城系統(tǒng),大家可以在過程中查缺補(bǔ)漏,提升水平2021-11-11Java?實(shí)現(xiàn)訂單未支付超時(shí)自動(dòng)取消功能(京東商城為例)
本文以京東網(wǎng)上商城為例,給大家介紹商品在下單后沒有支付的情況下,超時(shí)自動(dòng)取消功能,超過24小時(shí),就會(huì)自動(dòng)取消訂單,下面使用 Java 定時(shí)器實(shí)現(xiàn)超時(shí)取消訂單功能,感興趣的朋友一起看看吧2022-01-01淺析JavaMail發(fā)送郵件后再通過JavaMail接收格式問題
這篇文章主要介紹了JavaMail發(fā)送郵件后再通過JavaMail接收格式問題 ,本文通過代碼實(shí)例給大家詳細(xì)解說,需要的朋友可以參考下2019-06-06搭建Spring MVC和Vue3的應(yīng)用程序的實(shí)現(xiàn)
本文主要介紹了搭建Spring MVC和Vue3的應(yīng)用程序的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11