利用AidLux實(shí)現(xiàn)電離目標(biāo)檢測(cè)與實(shí)時(shí)鎖定演示-lesson4文章

基于yolov8實(shí)現(xiàn)目標(biāo)檢測(cè)功能匪傍,結(jié)合sort算法賦予id政溃,實(shí)現(xiàn)電力目標(biāo)鎖定功能。

部分代碼如下:

# aidlux相關(guān)

from cvs import *

import aidlite_gpu

from utils import detect_postprocess, preprocess_img, draw_detect_res, scale_boxes

import time

import cv2

import os

import numpy as np

import glob

import argparse

from filterpy.kalman import KalmanFilter

np.random.seed(0)

def linear_assignment(cost_matrix):

? ? try:

? ? ? ? import lap

? ? ? ? _, x, y = lap.lapjv(cost_matrix, extend_cost=True)

? ? ? ? return np.array([[y[i],i] for i in x if i >= 0]) #

? ? except ImportError:

? ? ? ? from scipy.optimize import linear_sum_assignment

? ? ? ? x, y = linear_sum_assignment(cost_matrix)

? ? ? ? return np.array(list(zip(x, y)))

def iou_batch(bb_test, bb_gt):

? ? """

? ? From SORT: Computes IOU between two bboxes in the form [x1,y1,x2,y2]

? ? """

? ? bb_gt = np.expand_dims(bb_gt, 0)

? ? bb_test = np.expand_dims(bb_test, 1)


? ? xx1 = np.maximum(bb_test[..., 0], bb_gt[..., 0])

? ? yy1 = np.maximum(bb_test[..., 1], bb_gt[..., 1])

? ? xx2 = np.minimum(bb_test[..., 2], bb_gt[..., 2])

? ? yy2 = np.minimum(bb_test[..., 3], bb_gt[..., 3])

? ? w = np.maximum(0., xx2 - xx1)

? ? h = np.maximum(0., yy2 - yy1)

? ? wh = w * h

? ? o = wh / ((bb_test[..., 2] - bb_test[..., 0]) * (bb_test[..., 3] - bb_test[..., 1])? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?

? ? ? ? + (bb_gt[..., 2] - bb_gt[..., 0]) * (bb_gt[..., 3] - bb_gt[..., 1]) - wh)? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?

? ? return(o)

def convert_bbox_to_z(bbox):

? ? """

? ? Takes a bounding box in the form [x1,y1,x2,y2] and returns z in the form

? ? ? ? [x,y,s,r] where x,y is the centre of the box and s is the scale/area and r is

? ? ? ? the aspect ratio

? ? """

? ? w = bbox[2] - bbox[0]

? ? h = bbox[3] - bbox[1]

? ? x = bbox[0] + w/2.

? ? y = bbox[1] + h/2.

? ? s = w * h? ? #scale is just area

? ? r = w / float(h)

? ? return np.array([x, y, s, r]).reshape((4, 1))

def convert_x_to_bbox(x, score=None):

? ? """

? ? Takes a bounding box in the centre form [x, y, s, r] and returns it in the form

? ? ? ? [x1,y1,x2,y2] where x1, y1 is the top left and x2, y2 is the bottom right

? ? """

? ? w = np.sqrt(x[2] * x[3])

? ? h = x[2] / w

? ? if(score==None):

? ? ? ? return np.array([x[0]-w/2., x[1]-h/2., x[0]+w/2., x[1]+h/2.]).reshape((1,4))

? ? else:

? ? ? ? return np.array([x[0]-w/2., x[1]-h/2., x[0]+w/2., x[1]+h/2., score]).reshape((1,5))

class KalmanBoxTracker(object):

? ? """

? ? This class represents the internal state of individual tracked objects observed as bbox.

? ? """

? ? count = 0

? ? def __init__(self,bbox):

? ? ? ? """

? ? ? ? Initialises a tracker using initial bounding box.

? ? ? ? """

? ? ? ? #define constant velocity model

? ? ? ? self.kf = KalmanFilter(dim_x=7, dim_z=4)

? ? ? ? self.kf.F = np.array([[1,0,0,0,1,0,0],[0,1,0,0,0,1,0],[0,0,1,0,0,0,1],[0,0,0,1,0,0,0],? [0,0,0,0,1,0,0],[0,0,0,0,0,1,0],[0,0,0,0,0,0,1]])

? ? ? ? self.kf.H = np.array([[1,0,0,0,0,0,0],[0,1,0,0,0,0,0],[0,0,1,0,0,0,0],[0,0,0,1,0,0,0]])

? ? ? ? self.kf.R[2:,2:] *= 10.

? ? ? ? self.kf.P[4:,4:] *= 1000. #give high uncertainty to the unobservable initial velocities

? ? ? ? self.kf.P *= 10.

? ? ? ? self.kf.Q[-1,-1] *= 0.01

? ? ? ? self.kf.Q[4:,4:] *= 0.01

? ? ? ? self.kf.x[:4] = convert_bbox_to_z(bbox)

? ? ? ? self.time_since_update = 0

? ? ? ? self.id = KalmanBoxTracker.count

? ? ? ? KalmanBoxTracker.count += 1

? ? ? ? self.history = []

? ? ? ? self.hits = 0

? ? ? ? self.hit_streak = 0

? ? ? ? self.age = 0

? ? def update(self,bbox):

? ? ? ? """

? ? ? ? Updates the state vector with observed bbox.

? ? ? ? """

? ? ? ? self.time_since_update = 0

? ? ? ? self.history = []

? ? ? ? self.hits += 1

? ? ? ? self.hit_streak += 1

? ? ? ? self.kf.update(convert_bbox_to_z(bbox))

? ? def predict(self):

? ? ? ? """

? ? ? ? Advances the state vector and returns the predicted bounding box estimate.

? ? ? ? """

? ? ? ? if((self.kf.x[6]+self.kf.x[2])<=0):

? ? ? ? ? ? self.kf.x[6] *= 0.0

? ? ? ? self.kf.predict()

? ? ? ? self.age += 1

? ? ? ? if(self.time_since_update>0):

? ? ? ? ? ? self.hit_streak = 0

? ? ? ? self.time_since_update += 1

? ? ? ? self.history.append(convert_x_to_bbox(self.kf.x))

? ? ? ? return self.history[-1]

? ? def get_state(self):

? ? ? ? """

? ? ? ? Returns the current bounding box estimate.

? ? ? ? """

? ? ? ? return convert_x_to_bbox(self.kf.x)

def associate_detections_to_trackers(detections,trackers,iou_threshold = 0.3):

? ? """

? ? Assigns detections to tracked object (both represented as bounding boxes)

? ? Returns 3 lists of matches, unmatched_detections and unmatched_trackers

? ? """

? ? if(len(trackers)==0):

? ? ? ? return np.empty((0,2),dtype=int), np.arange(len(detections)), np.empty((0,5),dtype=int)

? ? iou_matrix = iou_batch(detections, trackers)

? ? if min(iou_matrix.shape) > 0:

? ? ? ? a = (iou_matrix > iou_threshold).astype(np.int32)

? ? ? ? if a.sum(1).max() == 1 and a.sum(0).max() == 1:

? ? ? ? ? ? matched_indices = np.stack(np.where(a), axis=1)

? ? ? ? else:

? ? ? ? ? ? matched_indices = linear_assignment(-iou_matrix)

? ? else:

? ? ? ? matched_indices = np.empty(shape=(0,2))

? ? unmatched_detections = []

? ? for d, det in enumerate(detections):

? ? ? ? if(d not in matched_indices[:,0]):

? ? ? ? ? ? unmatched_detections.append(d)

? ? unmatched_trackers = []

? ? for t, trk in enumerate(trackers):

? ? ? ? if(t not in matched_indices[:,1]):

? ? ? ? ? ? unmatched_trackers.append(t)

? ? #filter out matched with low IOU

? ? matches = []

? ? for m in matched_indices:

? ? ? ? if(iou_matrix[m[0], m[1]]<iou_threshold):

? ? ? ? ? ? unmatched_detections.append(m[0])

? ? ? ? ? ? unmatched_trackers.append(m[1])

? ? ? ? else:

? ? ? ? ? ? matches.append(m.reshape(1,2))

? ? if(len(matches)==0):

? ? ? ? matches = np.empty((0,2),dtype=int)

? ? else:

? ? ? ? matches = np.concatenate(matches,axis=0)

? ? return matches, np.array(unmatched_detections), np.array(unmatched_trackers)

class Sort(object):

? ? def __init__(self, max_age=1, min_hits=3, iou_threshold=0.3):

? ? ? ? """

? ? ? ? Sets key parameters for SORT

? ? ? ? """

? ? ? ? self.max_age = max_age? # time_since_update > max_age, track被清除

? ? ? ? self.min_hits = min_hits

? ? ? ? self.iou_threshold = iou_threshold

? ? ? ? self.trackers = []

? ? ? ? self.frame_count = 0

? ? def update(self, dets=np.empty((0, 5))):

? ? ? ? """

? ? ? ? Params:

? ? ? ? dets - a numpy array of detections in the format [[x1,y1,x2,y2,score],[x1,y1,x2,y2,score],...]

? ? ? ? Requires: this method must be called once for each frame even with empty detections (use np.empty((0, 5)) for frames without detections).

? ? ? ? Returns the a similar array, where the last column is the object ID.

? ? ? ? NOTE: The number of objects returned may differ from the number of detections provided.

? ? ? ? """

? ? ? ? self.frame_count += 1

? ? ? ? # get predicted locations from existing trackers.

? ? ? ? trks = np.zeros((len(self.trackers), 5))

? ? ? ? to_del = []

? ? ? ? ret = []

? ? ? ? for t, trk in enumerate(trks):

? ? ? ? ? ? pos = self.trackers[t].predict()[0]

? ? ? ? ? ? trk[:] = [pos[0], pos[1], pos[2], pos[3], 0]

? ? ? ? ? ? if np.any(np.isnan(pos)):

? ? ? ? ? ? ? ? to_del.append(t)

? ? ? ? trks = np.ma.compress_rows(np.ma.masked_invalid(trks))

? ? ? ? for t in reversed(to_del):

? ? ? ? ? ? self.trackers.pop(t)

? ? ? ? matched, unmatched_dets, unmatched_trks = associate_detections_to_trackers(dets, trks, self.iou_threshold)

? ? ? ? # update matched trackers with assigned detections

? ? ? ? for m in matched:

? ? ? ? ? ? self.trackers[m[1]].update(dets[m[0], :])

? ? ? ? # create and initialize new trackers for unmatched detections

? ? ? ? for i in unmatched_dets:

? ? ? ? ? ? trk = KalmanBoxTracker(dets[i,:])

? ? ? ? ? ? self.trackers.append(trk)

? ? ? ? i = len(self.trackers)

? ? ? ? for trk in reversed(self.trackers):

? ? ? ? ? ? d = trk.get_state()[0]

? ? ? ? ? ? if (trk.time_since_update < 1) and (trk.hit_streak >= self.min_hits or self.frame_count <= self.min_hits):

? ? ? ? ? ? ? ? ret.append(np.concatenate((d, [trk.id+1])).reshape(1,-1)) # +1 as MOT benchmark requires positive

? ? ? ? ? ? i -= 1

? ? ? ? ? ? # remove dead tracklet

? ? ? ? ? ? if(trk.time_since_update > self.max_age):

? ? ? ? ? ? ? ? self.trackers.pop(i)

? ? ? ? if(len(ret)>0):

? ? ? ? ? ? return np.concatenate(ret)

? ? ? ? return np.empty((0,5))

if __name__ == '__main__':

? ? mot_tracker = Sort(max_age = 1,? # time_since_update>max_age, 清楚在跟目標(biāo)

? ? ? ? ? ? ? ? ? ? ? min_hits = 3,? # hit_streak>min_hits, 轉(zhuǎn)為確認(rèn)態(tài)

? ? ? ? ? ? ? ? ? ? ? iou_threshold = 0.3) # create instance of the SORT tracker

? ? # tflite模型

? ? model_path = '/home/yolov8/models/8086_best_float32.tflite'

? ? # 定義輸入輸出shape

? ? in_shape = [1 * 640 * 640 * 3 * 4]? # HWC, float32

? ? out_shape = [1 * 8400 * 52 * 4]? # 8400: total cells, 52 = 48(num_classes) + 4(xywh), float32

? ? # AidLite初始化

? ? aidlite = aidlite_gpu.aidlite()

? ? # 載入模型

? ? res = aidlite.ANNModel(model_path, in_shape, out_shape, 4, 0)

? ? print(res)

? ? ''' 讀取手機(jī)后置攝像頭 '''

? ? cap = cvs.VideoCapture(0)

? ? frame_id = 0

? ? while True:

? ? ? ? frame = cap.read()

? ? ? ? if frame is None:

? ? ? ? ? ? continue

? ? ? ? frame_id += 1

? ? ? ? if frame_id % 3 != 0:

? ? ? ? ? ? continue

? ? ? ? time0 = time.time()

? ? ? ? # 預(yù)處理

? ? ? ? img = preprocess_img(frame, target_shape=(640, 640), div_num=255, means=None, stds=None)

? ? ? ? aidlite.setInput_Float32(img, 640, 640)

? ? ? ? # 推理

? ? ? ? aidlite.invoke()

? ? ? ? preds = aidlite.getOutput_Float32(0)

? ? ? ? preds = preds.reshape(1, 52, 8400)

? ? ? ? preds = detect_postprocess(preds, frame.shape, [640, 640, 3], conf_thres=0.25, iou_thres=0.45)

? ? ? ? print('1 batch takes {} s'.format(time.time() - time0))

? ? ? ? if len(preds) != 0:

? ? ? ? ? ? preds[:, :4] = scale_boxes([640, 640], preds[:, :4], frame.shape)

? ? ? ? ? ? ''' SORT鎖定 '''

? ? ? ? ? ? preds_out = preds[:, :5]? # 數(shù)據(jù)切片,得到格式如[x1, y1, x2, y2, conf]的ndarray誉结。

? ? ? ? ? ? trackers = mot_tracker.update(preds_out)? # predict -> associate -> update

? ? ? ? ? ? ''' 繪制結(jié)果 '''

? ? ? ? ? ? for d in trackers:

? ? ? ? ? ? ? ? cv2.putText(frame, str(int(d[4])), (int(d[0]), int(d[1])), cv2.FONT_HERSHEY_SIMPLEX, 0.7, (0, 0, 255), 1)

? ? ? ? ? ? ? ? cv2.rectangle(frame, (int(d[0]), int(d[1])), (int(d[2]), int(d[3])), (0, 0, 255),thickness = 2)?

? ? ? ? cvs.imshow(frame)

本視頻為基于AidLux的演示視頻,在安卓端實(shí)現(xiàn)yolov8實(shí)時(shí)檢測(cè)以及sort電力目標(biāo)鎖定功能券躁。

點(diǎn)擊下方演示視頻即可觀看惩坑。

演示視頻

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市也拜,隨后出現(xiàn)的幾起案子以舒,更是在濱河造成了極大的恐慌,老刑警劉巖慢哈,帶你破解...
    沈念sama閱讀 222,627評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蔓钟,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡卵贱,警方通過(guò)查閱死者的電腦和手機(jī)滥沫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門侣集,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人兰绣,你說(shuō)我怎么就攤上這事世分。” “怎么了缀辩?”我有些...
    開封第一講書人閱讀 169,346評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵臭埋,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我臀玄,道長(zhǎng)瓢阴,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 60,097評(píng)論 1 300
  • 正文 為了忘掉前任健无,我火速辦了婚禮荣恐,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘睬涧。我一直安慰自己募胃,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,100評(píng)論 6 398
  • 文/花漫 我一把揭開白布畦浓。 她就那樣靜靜地躺著痹束,像睡著了一般。 火紅的嫁衣襯著肌膚如雪讶请。 梳的紋絲不亂的頭發(fā)上祷嘶,一...
    開封第一講書人閱讀 52,696評(píng)論 1 312
  • 那天,我揣著相機(jī)與錄音夺溢,去河邊找鬼论巍。 笑死,一個(gè)胖子當(dāng)著我的面吹牛风响,可吹牛的內(nèi)容都是我干的嘉汰。 我是一名探鬼主播,決...
    沈念sama閱讀 41,165評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼状勤,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼鞋怀!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起持搜,我...
    開封第一講書人閱讀 40,108評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤密似,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后葫盼,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體残腌,經(jīng)...
    沈念sama閱讀 46,646評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,709評(píng)論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了抛猫。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蟆盹。...
    茶點(diǎn)故事閱讀 40,861評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖邑滨,靈堂內(nèi)的尸體忽然破棺而出日缨,到底是詐尸還是另有隱情钱反,我是刑警寧澤掖看,帶...
    沈念sama閱讀 36,527評(píng)論 5 351
  • 正文 年R本政府宣布,位于F島的核電站面哥,受9級(jí)特大地震影響哎壳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜尚卫,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,196評(píng)論 3 336
  • 文/蒙蒙 一归榕、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧吱涉,春花似錦刹泄、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至鳖链,卻和暖如春姆蘸,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背芙委。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工逞敷, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人灌侣。 一個(gè)月前我還...
    沈念sama閱讀 49,287評(píng)論 3 379
  • 正文 我出身青樓推捐,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親侧啼。 傳聞我的和親對(duì)象是個(gè)殘疾皇子牛柒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,860評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容