版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)
文檔簡介
第python目標檢測yolo3詳解預測及代碼復現(xiàn)目錄學習前言實現(xiàn)思路1、yolo3的預測思路(網(wǎng)絡構(gòu)建思路)2、利用先驗框?qū)W(wǎng)絡的輸出進行解碼3、進行得分排序與非極大抑制篩選實現(xiàn)結(jié)果
學習前言
對yolo2解析完了之后當然要講講yolo3,yolo3與yolo2的差別主要在網(wǎng)絡的特征提取部分,實際的解碼部分其實差距不大
代碼下載
本次教程主要基于github中的項目點擊直接下載,該項目相比于yolo3-Keras的項目更容易看懂一些,不過它的許多代碼與yolo3-Keras相同。
我保留了預測部分的代碼,在實際可以通過執(zhí)行detect.py運行示例。
鏈接:/s/1_xLeytnjBBSL2h2Kj4-YEw
取碼:i3hi
實現(xiàn)思路
1、yolo3的預測思路(網(wǎng)絡構(gòu)建思路)
YOLOv3相比于之前的yolo1和yolo2,改進較大,主要改進方向有:
1、使用了殘差網(wǎng)絡Residual,殘差卷積就是進行一次3X3、步長為2的卷積,然后保存該卷積layer,再進行一次1X1的卷積和一次3X3的卷積,并把這個結(jié)果加上layer作為最后的結(jié)果,殘差網(wǎng)絡的特點是容易優(yōu)化,并且能夠通過增加相當?shù)纳疃葋硖岣邷蚀_率。其內(nèi)部的殘差塊使用了跳躍連接,緩解了在深度神經(jīng)網(wǎng)絡中增加深度帶來的梯度消失問題。
2、提取多特征層進行目標檢測,一共提取三個特征層,特征層的shape分別為(13,13,75),(26,26,75),(52,52,75),最后一個維度為75是因為該圖是基于voc數(shù)據(jù)集的,它的類為20種,yolo3只有針對每一個特征層存在3個先驗框,所以最后維度為3x25;
如果使用的是coco訓練集,類則為80種,最后的維度應該為255=3x85,三個特征層的shape為(13,13,255),(26,26,255),(52,52,255)
3、其采用反卷積UmSampling2d設(shè)計,逆卷積相對于卷積在神經(jīng)網(wǎng)絡結(jié)構(gòu)的正向和反向傳播中做相反的運算,其可以更多更好的提取出特征。
其實際情況就是,輸入N張416x416的圖片,在經(jīng)過多層的運算后,會輸出三個shape分別為(N,13,13,255),(N,26,26,255),(N,52,52,255)的數(shù)據(jù),對應每個圖分為13x13、26x26、52x52的網(wǎng)格上3個先驗框的位置。
實現(xiàn)代碼如下,其在實際調(diào)用時,會調(diào)用其中的yolo_inference函數(shù),此時獲得三個特征層的內(nèi)容。
def_darknet53(self,inputs,conv_index,training=True,norm_decay=0.99,norm_epsilon=1e-3):
Introduction
------------
構(gòu)建yolo3使用的darknet53網(wǎng)絡結(jié)構(gòu)
Parameters
----------
inputs:模型輸入變量
conv_index:卷積層數(shù)序號,方便根據(jù)名字加載預訓練權(quán)重
weights_dict:預訓練權(quán)重
training:是否為訓練
norm_decay:在預測時計算movingaverage時的衰減率
norm_epsilon:方差加上極小的數(shù),防止除以0的情況
Returns
-------
conv:經(jīng)過52層卷積計算之后的結(jié)果,輸入圖片為416x416x3,則此時輸出的結(jié)果shape為13x13x1024
route1:返回第26層卷積計算結(jié)果52x52x256,供后續(xù)使用
route2:返回第43層卷積計算結(jié)果26x26x512,供后續(xù)使用
conv_index:卷積層計數(shù),方便在加載預訓練模型時使用
withtf.variable_scope('darknet53'):
#416,416,3-416,416,32
conv=self._conv2d_layer(inputs,filters_num=32,kernel_size=3,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
#416,416,32-208,208,64
conv,conv_index=self._Residual_block(conv,conv_index=conv_index,filters_num=64,blocks_num=1,training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
#208,208,64-104,104,128
conv,conv_index=self._Residual_block(conv,conv_index=conv_index,filters_num=128,blocks_num=2,training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
#104,104,128-52,52,256
conv,conv_index=self._Residual_block(conv,conv_index=conv_index,filters_num=256,blocks_num=8,training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
#route1=52,52,256
route1=conv
#52,52,256-26,26,512
conv,conv_index=self._Residual_block(conv,conv_index=conv_index,filters_num=512,blocks_num=8,training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
#route2=26,26,512
route2=conv
#26,26,512-13,13,1024
conv,conv_index=self._Residual_block(conv,conv_index=conv_index,filters_num=1024,blocks_num=4,training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
#route3=13,13,1024
returnroute1,route2,conv,conv_index
#輸出兩個網(wǎng)絡結(jié)果
#第一個是進行5次卷積后,用于下一次逆卷積的,卷積過程是1X1,3X3,1X1,3X3,1X1
#第二個是進行5+2次卷積,作為一個特征層的,卷積過程是1X1,3X3,1X1,3X3,1X1,3X3,1X1
def_yolo_block(self,inputs,filters_num,out_filters,conv_index,training=True,norm_decay=0.99,norm_epsilon=1e-3):
Introduction
------------
yolo3在Darknet53提取的特征層基礎(chǔ)上,又加了針對3種不同比例的featuremap的block,這樣來提高對小物體的檢測率
Parameters
----------
inputs:輸入特征
filters_num:卷積核數(shù)量
out_filters:最后輸出層的卷積核數(shù)量
conv_index:卷積層數(shù)序號,方便根據(jù)名字加載預訓練權(quán)重
training:是否為訓練
norm_decay:在預測時計算movingaverage時的衰減率
norm_epsilon:方差加上極小的數(shù),防止除以0的情況
Returns
-------
route:返回最后一層卷積的前一層結(jié)果
conv:返回最后一層卷積的結(jié)果
conv_index:conv層計數(shù)
conv=self._conv2d_layer(inputs,filters_num=filters_num,kernel_size=1,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
conv=self._conv2d_layer(conv,filters_num=filters_num*2,kernel_size=3,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
conv=self._conv2d_layer(conv,filters_num=filters_num,kernel_size=1,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
conv=self._conv2d_layer(conv,filters_num=filters_num*2,kernel_size=3,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
conv=self._conv2d_layer(conv,filters_num=filters_num,kernel_size=1,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
route=conv
conv=self._conv2d_layer(conv,filters_num=filters_num*2,kernel_size=3,strides=1,name="conv2d_"+str(conv_index))
conv=self._batch_normalization_layer(conv,name="batch_normalization_"+str(conv_index),training=training,norm_decay=norm_decay,norm_epsilon=norm_epsilon)
conv_index+=1
conv=self._conv2d_layer(conv,filters_num=out_filters,kernel_size=1,strides=1,name="conv2d_"+str(conv_index),use_bias=True)
conv_index+=1
returnroute,conv,conv_index
#返回三個特征層的內(nèi)容
defyolo_inference(self,inputs,num_anchors,num_classes,training=True):
Introduction
------------
構(gòu)建yolo模型結(jié)構(gòu)
Parameters
----------
inputs:模型的輸入變量
num_anchors:每個gridcell負責檢測的anchor數(shù)量
num_classes:類別數(shù)量
training:是否為訓練模式
conv_index=1
#route1=52,52,256、route2=26,26,512、route3=13,13,1024
conv2d_26,conv2d_43,conv,conv_index=self._darknet53(inputs,conv_index,training=training,norm_decay=self.norm_decay,norm_epsilon=self.norm_epsilon)
withtf.variable_scope('yolo'):
#--------------------------------------#
#獲得第一個特征層
#--------------------------------------#
#conv2d_57=13,13,512,conv2d_59=13,13,255(3x(80+5))
conv2d_57,conv2d_59,conv_index=self._yolo_block(conv,512,num_anchors*(num_classes+5),conv_index=conv_index,training=training,norm_decay=self.norm_decay,norm_epsilon=self.norm_epsilon)
#--------------------------------------#
#獲得第二個特征層
#--------------------------------------#
conv2d_60=self._conv2d_layer(conv2d_57,filters_num=256,kernel_size=1,strides=1,name="conv2d_"+str(conv_index))
conv2d_60=self._batch_normalization_layer(conv2d_60,name="batch_normalization_"+str(conv_index),training=training,norm_decay=self.norm_decay,norm_epsilon=self.norm_epsilon)
conv_index+=1
#unSample_0=26,26,256
unSample_0=tf.image.resize_nearest_neighbor(conv2d_60,[2*tf.shape(conv2d_60)[1],2*tf.shape(conv2d_60)[1]],name='upSample_0')
#route0=26,26,768
route0=tf.concat([unSample_0,conv2d_43],axis=-1,name='route_0')
#conv2d_65=52,52,256,conv2d_67=26,26,255
conv2d_65,conv2d_67,conv_index=self._yolo_block(route0,256,num_anchors*(num_classes+5),conv_index=conv_index,training=training,norm_decay=self.norm_decay,norm_epsilon=self.norm_epsilon)
#--------------------------------------#
#獲得第三個特征層
#--------------------------------------#
conv2d_68=self._conv2d_layer(conv2d_65,filters_num=128,kernel_size=1,strides=1,name="conv2d_"+str(conv_index))
conv2d_68=self._batch_normalization_layer(conv2d_68,name="batch_normalization_"+str(conv_index),training=training,norm_decay=self.norm_decay,norm_epsilon=self.norm_epsilon)
conv_index+=1
#unSample_1=52,52,128
unSample_1=tf.image.resize_nearest_neighbor(conv2d_68,[2*tf.shape(conv2d_68)[1],2*tf.shape(conv2d_68)[1]],name='upSample_1')
#route1=52,52,384
route1=tf.concat([unSample_1,conv2d_26],axis=-1,name='route_1')
#conv2d_75=52,52,255
_,conv2d_75,_=self._yolo_block(route1,128,num_anchors*(num_classes+5),conv_index=conv_index,training=training,norm_decay=self.norm_decay,norm_epsilon=self.norm_epsilon)
return[conv2d_59,conv2d_67,conv2d_75]
2、利用先驗框?qū)W(wǎng)絡的輸出進行解碼
yolo3的先驗框生成與yolo2的類似,如果不明白先驗框是如何生成的,可以看我的上一篇博文
python目標檢測yolo2詳解及其預測代碼復現(xiàn)
其實yolo3的解碼與yolo2的解碼過程一樣,只是對于yolo3而言,其需要對三個特征層進行解碼,三個特征層的shape分別為(N,13,13,255),(N,26,26,255),(N,52,52,255)的數(shù)據(jù),對應每個圖分為13x13、26x26、52x52的網(wǎng)格上3個先驗框的位置。
此處需要用到一個循環(huán)。
1、將第一個特征層reshape成[-1,13,13,3,80+5],代表169個中心點每個中心點的3個先驗框的情況。
2、將80+5的5中的xywh分離出來,0、1是xy相對中心點的偏移量;2、3是寬和高的情況;4是置信度。
3、建立13x13的網(wǎng)格,代表圖片進行13x13處理后網(wǎng)格的中心點。
4、利用計算公式計算實際的bbox的位置。
5、置信度乘上80+5中的80(這里的80指的是類別概率)得到得分。
6、將第二個特征層reshape成[-1,26,26,3,80+5],重復2到5步。將第三個特征層reshape成[-1,52,52,3,80+5],重復2到5步。
單個特征層的解碼部分代碼如下:
#獲得單個特征層框的位置和得分
defboxes_and_scores(self,feats,anchors,classes_num,input_shape,image_shape):
Introduction
------------
將預測出的box坐標轉(zhuǎn)換為對應原圖的坐標,然后計算每個box的分數(shù)
Parameters
----------
feats:yolo輸出的featuremap
anchors:anchor的位置
class_num:類別數(shù)目
input_shape:輸入大小
image_shape:圖片大小
Returns
-------
boxes:物體框的位置
boxes_scores:物體框的分數(shù),為置信度和類別概率的乘積
#獲得特征
box_xy,box_wh,box_confidence,box_class_probs=self._get_feats(feats,anchors,classes_num,input_shape)
#尋找在原圖上的位置
boxes=self.correct_boxes(box_xy,box_wh,input_shape,image_shape)
boxes=tf.reshape(boxes,[-1,4])
#獲得置信度box_confidence*box_class_probs
box_scores=box_confidence*box_class_probs
box_scores=tf.reshape(box_scores,[-1,classes_num])
returnboxes,box_scores
#單個特征層的解碼過程
def_get_feats(self,feats,anchors,num_classes,input_shape):
Introduction
------------
根據(jù)yolo最后一層的輸出確定boundingbox
Parameters
----------
feats:yolo模型最后一層輸出
anchors:anchors的位置
num_classes:類別數(shù)量
input_shape:輸入大小
Returns
-------
box_xy,box_wh,box_confidence,box_class_probs
num_anchors=len(anchors)
anchors_tensor=tf.reshape(tf.constant(anchors,dtype=tf.float32),[1,1,1,num_anchors,2])
grid_size=tf.shape(feats)[1:3]
predictions=tf.reshape(feats,[-1,grid_size[0],grid_size[1],num_anchors,num_classes+5])
#這里構(gòu)建13*13*1*2的矩陣,對應每個格子加上對應的坐標
grid_y=tf.tile(tf.reshape(tf.range(grid_size[0]),[-1,1,1,1]),[1,grid_size[1],1,1])
grid_x=tf.tile(tf.reshape(tf.range(grid_size[1]),[1,-1,1,1]),[grid_size[0],1,1,1])
grid=tf.concat([grid_x,grid_y],axis=-1)
grid=tf.cast(grid,tf.float32)
#將x,y坐標歸一化,相對網(wǎng)格的位置
box_xy=(tf.sigmoid(predictions[...,:2])+grid)/tf.cast(grid_size[::-1],tf.float32)
#將w,h也歸一化
box_wh=tf.exp(predictions[...,2:4])*anchors_tensor/tf.cast(input_shape[::-1],tf.float32)
box_confidence=tf.sigmoid(predictions[...,4:5])
box_class_probs=tf.sigmoid(predictions[...,5:])
returnbox_xy,box_wh,box_confidence,box_class_probs
該函數(shù)被其它函數(shù)調(diào)用,用于完成三個特征層的解碼:
defeval(self,yolo_outputs,image_shape,max_boxes=20):
Introduction
------------
根據(jù)Yolo模型的輸出進行非極大值抑制,獲取最后的物體檢測框和物體檢測類別
Parameters
----------
yolo_outputs:yolo模型輸出
image_shape:圖片的大小
max_boxes:最大box數(shù)量
Returns
-------
boxes_:物體框的位置
scores_:物體類別的概率
classes_:物體類別
#每一個特征層對應三個先驗框
anchor_mask=[[6,7,8],[3,4,5],[0,1,2]]
boxes=[]
box_scores=[]
#inputshape是416x416
#image_shape是實際圖片的大小
input_shape=tf.shape(yolo_outputs[0])[1:3]*32
#對三個特征層的輸出獲取每個預測box坐標和box的分數(shù),score=置信度x類別概率
#---------------------------------------#
#對三個特征層解碼
#獲得分數(shù)和框的位置
#---------------------------------------#
foriinrange(len(yolo_outputs)):
_boxes,_box_scores=self.boxes_and_scores(yolo_outputs[i],self.anchors[anchor_mask[i]],len(self.class_names),input_shape,image_shape)
boxes.append(_boxes)
box_scores.append(_box_scores)
#放在一行里面便于操作
boxes=tf.concat(boxes,axis=0)
box_scores=tf.concat(box_scores,axis=0)
3、進行得分排序與非極大抑制篩選
這一部分基本上是所有目標檢測通用的部分。不過該項目的處理方式與其它項目不同。其對于每一個類進行判別。
1、取出每一類得分大于self.obj_threshold的框和得分。
2、利用框的位置和得分進行非極大抑制。
實現(xiàn)代碼如下:
#---------------------------------------#
#1、取出每一類得分大于self.obj_threshold
#的框和得分
#2、對得分進行非極大抑制
#---------------------------------------#
#對每一個類進行判斷
forcinrange(len(self.class_names)):
#取出所有類為c的box
class_boxes=tf.boolean_mask(boxes,mask[:,c])
#取出所有類為c的分數(shù)
class_box_scores=tf.boolean_mask(box_scores[:,c],mask[:,c])
#非極大抑制
nms_index=tf.image.non_max_suppression(class_boxes,class_box_scores,max_boxes_tensor,iou_threshold=self.nms_threshold)
#獲取非極大抑制的結(jié)果
class_boxes=tf.gather(class_boxes,nms_index)
class_box_scores=tf.gather(class_box_scores,nms_index)
classes=tf.ones_like(class_box_scores,'int32')*c
boxes_.append(class_boxes)
scores_.append(class_box_scores)
classes_.append(classes)
boxes_=tf.concat(boxes_,axis=0)
scores_=tf.concat(scores_,axis=0)
classes_=tf.concat(classes_,axis=0)
實際上該部分與第二部分在一個函數(shù)里,完成輸出的解碼和篩選,完成預測過程。
defeval(self,yolo_outputs,image_shape,max_boxes=20):
Introduction
------------
根據(jù)Yolo模型的輸出進行非極大值抑制,獲取最后的物體檢測框和物體檢測類別
Parameters
----------
yolo_outputs:yolo模型輸出
image_shape:圖片的大小
max_boxes:最大box數(shù)量
Returns
-------
boxes_:物體框的位置
scores_:物體類別的概率
classes_:物體類別
#每一個特征層對應三個先驗框
anchor_mask=[[6,7,8],[3,4,5],[0,1,2]]
boxes=[]
box_scores=[]
#inputshape是416x416
#image_shape是實際圖片的大小
input_shape=tf.shape(yolo_outputs[0])[1:3]*32
#對三個特征層的輸出獲取每個預測box坐標和box的分數(shù),score=置信度x類別概率
#---------------------------------------#
#對三
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 中學學生社團活動經(jīng)費預算制度
- 養(yǎng)老院定期體檢制度
- 企業(yè)內(nèi)部保密工作責任追究制度
- 公共交通車輛安全檢查制度
- 2026年工業(yè)制造中的6S管理與效率提升題解
- 2026年公務員考試模擬題行政職業(yè)能力測驗與申論練習
- 2026年航空航天基礎(chǔ)知識學習入門與實踐題庫
- 2026年體育訓練與健康管理測試題目
- 2026年企業(yè)管理實務能力測試題
- 2026年版權(quán)侵權(quán)監(jiān)測協(xié)議(實時·發(fā)現(xiàn)版)
- 2026中國電信四川公用信息產(chǎn)業(yè)有限責任公司社會成熟人才招聘備考題庫及答案詳解(奪冠系列)
- 學堂在線 雨課堂 學堂云 莊子哲學導讀 章節(jié)測試答案
- 23G409先張法預應力混凝土管樁
- 街道(鄉(xiāng)鎮(zhèn))區(qū)域養(yǎng)老服務中心建設(shè)驗收指標總分表及驗收標準
- 國家衛(wèi)生部《綜合醫(yī)院分級管理標準》
- 預防兩癌知識講座
- 人教版九年級數(shù)學第二十四章《圓》單元知識點總結(jié)
- 西班牙語專業(yè)本科論文模板
- GB/T 42288-2022電化學儲能電站安全規(guī)程
- 地質(zhì)災害治理工程用表格(完整資料)
- GB/T 9254.1-2021信息技術(shù)設(shè)備、多媒體設(shè)備和接收機電磁兼容第1部分: 發(fā)射要求
評論
0/150
提交評論