yolo.py 10.7 KB
Newer Older
J
JiaQi Xu 已提交
1 2 3 4 5
#-------------------------------------#
#       创建YOLO类
#-------------------------------------#
import colorsys
import os
B
Bubbliiiing 已提交
6 7 8

import cv2
import numpy as np
J
JiaQi Xu 已提交
9 10
import torch
import torch.backends.cudnn as cudnn
B
Bubbliiiing 已提交
11 12
import torch.nn as nn
from PIL import Image, ImageDraw, ImageFont
J
JiaQi Xu 已提交
13
from torch.autograd import Variable
B
Bubbliiiing 已提交
14 15 16 17 18

from nets.yolo4 import YoloBody
from utils.utils import (DecodeBox, bbox_iou, letterbox_image,
                         non_max_suppression, yolo_correct_boxes)

J
JiaQi Xu 已提交
19

B
Bubbliiiing 已提交
20 21 22
#--------------------------------------------#
#   使用自己训练好的模型预测需要修改2个参数
#   model_path和classes_path都需要修改!
B
Bubbliiiing 已提交
23 24
#   如果出现shape不匹配,一定要注意
#   训练时的model_path和classes_path参数的修改
B
Bubbliiiing 已提交
25
#--------------------------------------------#
J
JiaQi Xu 已提交
26 27
class YOLO(object):
    _defaults = {
B
Bubbliiiing 已提交
28 29 30 31 32 33
        "model_path"        : 'model_data/yolo4_weights.pth',
        "anchors_path"      : 'model_data/yolo_anchors.txt',
        "classes_path"      : 'model_data/coco_classes.txt',
        "model_image_size"  : (416, 416, 3),
        "confidence"        : 0.5,
        "iou"               : 0.3,
B
Bubbliiiing 已提交
34 35 36 37 38 39
        "cuda"              : True,
        #---------------------------------------------------------------------#
        #   该变量用于控制是否使用letterbox_image对输入图像进行不失真的resize,
        #   在多次测试后,发现关闭letterbox_image直接resize的效果更好
        #---------------------------------------------------------------------#
        "letterbox_image"   : False,
J
JiaQi Xu 已提交
40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
    }

    @classmethod
    def get_defaults(cls, n):
        if n in cls._defaults:
            return cls._defaults[n]
        else:
            return "Unrecognized attribute name '" + n + "'"

    #---------------------------------------------------#
    #   初始化YOLO
    #---------------------------------------------------#
    def __init__(self, **kwargs):
        self.__dict__.update(self._defaults)
        self.class_names = self._get_class()
        self.anchors = self._get_anchors()
        self.generate()
B
Bubbliiiing 已提交
57

J
JiaQi Xu 已提交
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
    #---------------------------------------------------#
    #   获得所有的分类
    #---------------------------------------------------#
    def _get_class(self):
        classes_path = os.path.expanduser(self.classes_path)
        with open(classes_path) as f:
            class_names = f.readlines()
        class_names = [c.strip() for c in class_names]
        return class_names
    
    #---------------------------------------------------#
    #   获得所有的先验框
    #---------------------------------------------------#
    def _get_anchors(self):
        anchors_path = os.path.expanduser(self.anchors_path)
        with open(anchors_path) as f:
            anchors = f.readline()
        anchors = [float(x) for x in anchors.split(',')]
        return np.array(anchors).reshape([-1, 3, 2])[::-1,:,:]

    #---------------------------------------------------#
B
Bubbliiiing 已提交
79
    #   生成模型
J
JiaQi Xu 已提交
80 81
    #---------------------------------------------------#
    def generate(self):
B
Bubbliiiing 已提交
82 83 84 85 86 87 88 89
        #---------------------------------------------------#
        #   建立yolov4模型
        #---------------------------------------------------#
        self.net = YoloBody(len(self.anchors[0]), len(self.class_names)).eval()

        #---------------------------------------------------#
        #   载入yolov4模型的权重
        #---------------------------------------------------#
J
JiaQi Xu 已提交
90
        print('Loading weights into state dict...')
B
Bubbliiiing 已提交
91
        device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')
B
Bubbliiiing 已提交
92 93
        state_dict = torch.load(self.model_path, map_location=device)
        self.net.load_state_dict(state_dict)
B
Bubbliiiing 已提交
94
        print('Finished!')
J
JiaQi Xu 已提交
95
        
J
JiaQi Xu 已提交
96
        if self.cuda:
J
JiaQi Xu 已提交
97 98
            os.environ["CUDA_VISIBLE_DEVICES"] = '0'
            self.net = nn.DataParallel(self.net)
J
JiaQi Xu 已提交
99 100
            self.net = self.net.cuda()

B
Bubbliiiing 已提交
101 102 103
        #---------------------------------------------------#
        #   建立三个特征层解码用的工具
        #---------------------------------------------------#
J
JiaQi Xu 已提交
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
        self.yolo_decodes = []
        for i in range(3):
            self.yolo_decodes.append(DecodeBox(self.anchors[i], len(self.class_names),  (self.model_image_size[1], self.model_image_size[0])))


        print('{} model, anchors, and classes loaded.'.format(self.model_path))
        # 画框设置不同的颜色
        hsv_tuples = [(x / len(self.class_names), 1., 1.)
                      for x in range(len(self.class_names))]
        self.colors = list(map(lambda x: colorsys.hsv_to_rgb(*x), hsv_tuples))
        self.colors = list(
            map(lambda x: (int(x[0] * 255), int(x[1] * 255), int(x[2] * 255)),
                self.colors))

    #---------------------------------------------------#
    #   检测图片
    #---------------------------------------------------#
    def detect_image(self, image):
        image_shape = np.array(np.shape(image)[0:2])

B
Bubbliiiing 已提交
124 125
        #---------------------------------------------------------#
        #   给图像增加灰条,实现不失真的resize
B
Bubbliiiing 已提交
126
        #   也可以直接resize进行识别
B
Bubbliiiing 已提交
127
        #---------------------------------------------------------#
B
Bubbliiiing 已提交
128 129 130 131 132
        if self.letterbox_image:
            crop_img = np.array(letterbox_image(image, (self.model_image_size[1],self.model_image_size[0])))
        else:
            crop_img = image.convert('RGB')
            crop_img = crop_img.resize((self.model_image_size[1],self.model_image_size[0]), Image.BICUBIC)
B
Bubbliiiing 已提交
133
        photo = np.array(crop_img,dtype = np.float32) / 255.0
J
JiaQi Xu 已提交
134
        photo = np.transpose(photo, (2, 0, 1))
B
Bubbliiiing 已提交
135 136 137 138
        #---------------------------------------------------------#
        #   添加上batch_size维度
        #---------------------------------------------------------#
        images = [photo]
J
JiaQi Xu 已提交
139 140

        with torch.no_grad():
B
Bubbliiiing 已提交
141
            images = torch.from_numpy(np.asarray(images))
J
JiaQi Xu 已提交
142 143
            if self.cuda:
                images = images.cuda()
B
Bubbliiiing 已提交
144 145 146 147

            #---------------------------------------------------------#
            #   将图像输入网络当中进行预测!
            #---------------------------------------------------------#
J
JiaQi Xu 已提交
148
            outputs = self.net(images)
B
Bubbliiiing 已提交
149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
            output_list = []
            for i in range(3):
                output_list.append(self.yolo_decodes[i](outputs[i]))

            #---------------------------------------------------------#
            #   将预测框进行堆叠,然后进行非极大抑制
            #---------------------------------------------------------#
            output = torch.cat(output_list, 1)
            batch_detections = non_max_suppression(output, len(self.class_names),
                                                    conf_thres=self.confidence,
                                                    nms_thres=self.iou)

            #---------------------------------------------------------#
            #   如果没有检测出物体,返回原图
            #---------------------------------------------------------#
            try:
                batch_detections = batch_detections[0].cpu().numpy()
            except:
                return image
J
JiaQi Xu 已提交
168
            
B
Bubbliiiing 已提交
169 170 171 172 173 174 175 176 177 178 179 180 181 182
            #---------------------------------------------------------#
            #   对预测框进行得分筛选
            #---------------------------------------------------------#
            top_index = batch_detections[:,4] * batch_detections[:,5] > self.confidence
            top_conf = batch_detections[top_index,4]*batch_detections[top_index,5]
            top_label = np.array(batch_detections[top_index,-1],np.int32)
            top_bboxes = np.array(batch_detections[top_index,:4])
            top_xmin, top_ymin, top_xmax, top_ymax = np.expand_dims(top_bboxes[:,0],-1),np.expand_dims(top_bboxes[:,1],-1),np.expand_dims(top_bboxes[:,2],-1),np.expand_dims(top_bboxes[:,3],-1)

            #-----------------------------------------------------------------#
            #   在图像传入网络预测前会进行letterbox_image给图像周围添加灰条
            #   因此生成的top_bboxes是相对于有灰条的图像的
            #   我们需要对其进行修改,去除灰条的部分。
            #-----------------------------------------------------------------#
B
Bubbliiiing 已提交
183 184 185 186 187 188 189 190 191
            if self.letterbox_image:
                boxes = yolo_correct_boxes(top_ymin,top_xmin,top_ymax,top_xmax,np.array([self.model_image_size[0],self.model_image_size[1]]),image_shape)
            else:
                top_xmin = top_xmin / self.model_image_size[1] * image_shape[1]
                top_ymin = top_ymin / self.model_image_size[0] * image_shape[0]
                top_xmax = top_xmax / self.model_image_size[1] * image_shape[1]
                top_ymax = top_ymax / self.model_image_size[0] * image_shape[0]
                boxes = np.concatenate([top_ymin,top_xmin,top_ymax,top_xmax], axis=-1)
                
J
JiaQi Xu 已提交
192 193
        font = ImageFont.truetype(font='model_data/simhei.ttf',size=np.floor(3e-2 * np.shape(image)[1] + 0.5).astype('int32'))

B
Bubbliiiing 已提交
194
        thickness = max((np.shape(image)[0] + np.shape(image)[1]) // self.model_image_size[0], 1)
J
JiaQi Xu 已提交
195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215

        for i, c in enumerate(top_label):
            predicted_class = self.class_names[c]
            score = top_conf[i]

            top, left, bottom, right = boxes[i]
            top = top - 5
            left = left - 5
            bottom = bottom + 5
            right = right + 5

            top = max(0, np.floor(top + 0.5).astype('int32'))
            left = max(0, np.floor(left + 0.5).astype('int32'))
            bottom = min(np.shape(image)[0], np.floor(bottom + 0.5).astype('int32'))
            right = min(np.shape(image)[1], np.floor(right + 0.5).astype('int32'))

            # 画框框
            label = '{} {:.2f}'.format(predicted_class, score)
            draw = ImageDraw.Draw(image)
            label_size = draw.textsize(label, font)
            label = label.encode('utf-8')
B
Bubbliiiing 已提交
216
            print(label, top, left, bottom, right)
J
JiaQi Xu 已提交
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233
            
            if top - label_size[1] >= 0:
                text_origin = np.array([left, top - label_size[1]])
            else:
                text_origin = np.array([left, top + 1])

            for i in range(thickness):
                draw.rectangle(
                    [left + i, top + i, right - i, bottom - i],
                    outline=self.colors[self.class_names.index(predicted_class)])
            draw.rectangle(
                [tuple(text_origin), tuple(text_origin + label_size)],
                fill=self.colors[self.class_names.index(predicted_class)])
            draw.text(text_origin, str(label,'UTF-8'), fill=(0, 0, 0), font=font)
            del draw
        return image