Car Lane Detection === 現今無人駕駛技術已經越來越成熟及普及,從現在路上最常跑的電動車 Tesla 就含有很多自動駕駛的技術在裡面,上次看到這影片時[Tesla Navigating](https://www.youtube.com/watch?v=rKG3eDppzm4&t=94s&ab_channel=AIDRIVR)、[Tesla autopilot recognition of roadside structures](https://www.youtube.com/watch?v=7ztK5AhShqU&ab_channel=greentheonly)、[CA through the eyes of FSD BETA](https://www.youtube.com/watch?v=REcBsg93YIM&ab_channel=AIDRIVR),整個特斯拉能看到的場景很豐富,當然也對於最基礎的車道偵測也是必備功能。Tesla 是藉由很多 sensor 去提供資訊再去做運算與判斷不同場景,那本次想從基礎車道偵測去下手,藉由影像處理的方法去辨識車道線,再去判斷車子左右偏移狀況。 [加強版車道偵測在這~~](https://hackmd.io/@yoch/Sy8H_v4vO) ## 基礎車道辨識 ### 基本架構 整體架構如下,首先是由 camera 影像進來,那要對單張單張影像進行處理與判別。 ```flow st=>start: Image e=>end: Draw Line op=>operation: To Gary & Reduce Noice op2=>operation: Edge Detect op3=>operation: Interes Region op4=>operation: Hough Line Detection st->op->op2->op3->op4 op4->e ``` ### 方法 由上面架構圖可以看到使用了哪些方法,這部分就來講解一下。 首先做影像處理幾乎都是轉為單通道去做處理比較方便也好理解,整體過程都直接採用 OpenCV Library 進行,首先將 GBR 轉為 Gray ![](https://i.imgur.com/6EU6NiF.jpg =500x) 之後再透過 Gaussian Filter 進行低通濾波將影像中的雜訊給濾除掉, ![](https://i.imgur.com/HHkane6.jpg =500x) 接著進行邊緣偵測 ![](https://i.imgur.com/QjVGZBE.png =500x) 那這邊比較一下有無濾波的差異 沒有濾波 ![](https://i.imgur.com/rKClZwg.png) 有濾波 ![](https://i.imgur.com/c36wL04.png) 從上面可以看出來有些雜訊是被濾除掉的 那再來要偵測車道線,由於行車紀錄器都是固定位置,那可以直接抓出我們感興趣的位置進行偵測,由下圖可以看到只顯示出一部分要去做處理 ![](https://i.imgur.com/vKbHAvy.png) 從剛剛上面的圖再透過 hough transform 進行 line detection,當然會偵測到很多點然後顯示一堆線,所以必須採取一個步驟就是去限制這些點,然後把最需要的資訊再抓取出來,這邊透過高中所學的直線斜率簡單的知識就可以了! 大家應該都還記得兩點可以成一直線,那一直線就可以算出它的角度與斜率了吧,這邊附上一張圖讓各位回憶一下xd ![](https://i.imgur.com/JMa7vw9.png =300x) ![](https://i.imgur.com/nFhUMcR.png =400x) 那麼斜率 K 就等於 $$K ={y_2 - y_1 \over x_2 - x_1}$$ 那可以透過斜率方便尋找出我們需要的直線,這邊採用的方法是去將所有點的斜率去做一個限制與平均,可能這樣說會不懂直接上程式碼。 ```python= def make_coordinate(parameter, y_max, y_min): x1_mean = int((y_max - parameter[1]) / parameter[0]) x2_mean = int((y_min - parameter[1]) / parameter[0]) return np.array([x1_mean, y_max, x2_mean, y_min]) def mean_coordinate(img, lines): left_fit = [] right_fit = [] y_max = img.shape[0] y_min = img.shape[0] for line in lines: x1, y1, x2, y2 = line[0] y_min = min(min(y1, y2), y_min) parameter = np.polyfit((x1, x2), (y1, y2), 1) ## y = ax + b, 兩點求直線公式 slope = parameter[0] intercept = parameter[1] if abs(slope) < 0.5: continue if slope < 0: ## 判斷斜率來表示左右 left_fit.append((slope, intercept)) else: right_fit.append((slope, intercept)) if len(left_fit) > 0 and len(right_fit) > 0: left_fit_mean = np.mean(left_fit, axis=0) right_fit_mean = np.mean(right_fit, axis=0) left_coordinate = make_coordinate(left_fit_mean, y_max, y_min) right_coordinate = make_coordinate(right_fit_mean, y_max, y_min) return np.array([left_coordinate, right_coordinate]) return ``` ### Demo 可以偵測車道線,但是效果並不這麼理想,因此又想說做一個加強版的來讓車道偵測再完美一些。 ![](https://i.imgur.com/2d0vtuB.gif) ![](https://imgur.com/mFOsjvH.gif) ## 參考 >Lane Detection:https://github.com/davidawad/Lane-Detection >Hands-On Tutorial on Real-Time Lane Detection using OpenCV:https://www.analyticsvidhya.com/blog/2020/05/tutorial-real-time-lane-detection-opencv/