micomia

Blog

技術記事

OpenCV × YOLOで作る車両カウントシステム

OpenCV × YOLOで作る車両カウントシステム

micomia株式会社でAIエンジニアをしている松久保です。

本記事では、OpenCVとYOLOを用いて構築した車両カウントシステムについてご紹介します。どのような技術を使い、どのような仕組みで車両を検出・追跡しカウントしているのか、実装のポイントも含めてわかりやすく解説していきます。




1. はじめに

今回作成したシステムは、ノートパソコン 上でも動作する軽量構成であり、主に Python・Ultralytics YOLO・OpenCV というシンプルな技術スタックで実現しています。

車両検出 & 分類 → 追跡 → ライン交差によるカウント という処理フローで動作し、交通量調査やインフラ監視などの用途にも応用できる実用的な仕組みになっています。



2. 使用技術について

本システムは Python をベースに、主に YOLO(Ultralytics YOLO11)OpenCV を組み合わせて構築しています。

YOLOは軽量・高速・高精度な物体検出モデルで、車両の検出 (Detection)と 追跡 (Tracking) を一貫して行うことができます。

OpenCVは、動画の読み込み、描画、ライン交差のチェック、そして最終的な動画出力など、処理全体の制御を行っています。



アーキテクチャ

本システムは「入力動画 → 検出・追跡 → ライン交差判定 → カウント」というシンプルな処理パイプラインで構成されています。


  1. 入力動画のフレーム分割

    Python と OpenCV を用いて入力動画を読み込み、1フレームずつ画像として取り出します。以降の処理はすべてフレーム単位で行われます。

  2. YOLOによる車両検出・トラッキング

    各フレームに対して YOLO を適用し、車・バイク・バス・トラックを検出・分類します。track モードを使用することで、各車両に対して位置(バウンディングボックス)、クラスID、トラッキングIDが付与され、同一車両をフレーム間で一貫して追跡できます。

  3. カウントライン交差の判定

    画像中に1本のカウントライン(本実装では画像中央の水平線)を設置し、各トラッキングIDごとに「前フレームの位置」と「現在フレームの位置」を比較します。車両の位置がカウントラインの上側から下側へ移動したタイミングで、その車両をカウントします。



3. コード・実装の解説

ライブラリのインポートと基本設定

必要なライブラリのインポートと、モデルや入力・出力ファイルのパスなどの基本設定を行います。

import cv2
from ultralytics import YOLO

# Config
MODEL = "yolo11l.pt"
INPUT_VIDEO = "input_path"
OUTPUT_VIDEO = "output_path"

モデル読み込みと動画入出力の初期化

YOLOモデルをロードし、OpenCVで動画の読み込み・書き込みの準備をします。

model = YOLO(MODEL)
cap = cv2.VideoCapture(INPUT_VIDEO)
if not cap.isOpened():
    print("Error: Could not open video.")
    exit()

fps = int(cap.get(cv2.CAP_PROP_FPS))
w = int(cap.get(cv2.CAP_PROP_FRAME_WIDTH))
h = int(cap.get(cv2.CAP_PROP_FRAME_HEIGHT))

fourcc = cv2.VideoWriter_fourcc(*'mp4v')
out = cv2.VideoWriter(OUTPUT_VIDEO, fourcc, fps, (w, h))

カウント用の変数定義

車種ごとのカウンタや、トラッキングIDの履歴を管理するための変数を定義します。

class_counts = {
    'car': 0,
    'motorcycle': 0,
    'bus': 0,
    'truck': 0
}
CLASS_NAMES = {2: 'car', 3: 'motorcycle', 5: 'bus', 7: 'truck'}
counted_ids = {}
track_history = {}

メインループ:フレームごとの処理

動画を1フレームずつ処理し、検出 → 追跡 → ライン交差判定 → カウント表示を行います。

while True:
    ret, frame = cap.read()
    if not ret:
        print("End of video stream.")
        break
    result = model.track(frame, classes=[2, 3, 5, 7], persist=True, tracker="bytetrack.yaml")
    annotated_frame = result[0].plot()

ライン交差の判定とカウントロジック

各トラッキングIDごとに「前フレームの位置」と「現在の位置」を比較し、ラインより上側 → ラインより下側に移動したタイミングでカウントします。

    line_y = h // 2
    cv2.line(annotated_frame, (0, line_y), (w, line_y), (0, 255, 255), 2)

    if result[0].boxes.id is not None:
        boxes = result[0].boxes.xywh.cpu()
        track_ids = result[0].boxes.id.int().cpu().tolist()
        class_ids = result[0].boxes.cls.int().cpu().tolist()

        for box, track_id, class_id in zip(boxes, track_ids, class_ids):
            class_name = CLASS_NAMES.get(class_id)
            if not class_name:
                continue
            car_position_y = int(box[1])
            current_side = 'top' if car_position_y < line_y else 'down'
            if track_id in counted_ids:
                continue
            if track_id in track_history:
                prev_side = track_history[track_id]
                if prev_side == 'top' and current_side == 'down':
                    class_counts[class_name] += 1
                    counted_ids[track_id] = class_name
            track_history[track_id] = current_side

カウント結果の描画と動画出力

最後に、フレーム上にカウント結果を描画し、動画として保存します。

    total_count = sum(class_counts.values())
    count_text = f"TOTAL: {total_count}"
    details_text = " | ".join([f"{k}: {v}" for k, v in class_counts.items()])
    cv2.putText(annotated_frame, count_text, (10, 50), cv2.FONT_HERSHEY_SIMPLEX, 1.2, (0, 0, 255), 3)
    cv2.putText(annotated_frame, details_text, (10, 100), cv2.FONT_HERSHEY_SIMPLEX, 0.7, (255, 255, 0), 2)
    cv2.imshow('YOLO11l Detection', annotated_frame)
    out.write(annotated_frame)
    if cv2.waitKey(1) & 0xFF == ord('q'):
        break

cap.release()
out.release()
cv2.destroyAllWindows()
print(f"Processing done. Saved to {OUTPUT_VIDEO}")


4. 実行結果

YOLO 画像認識 車

YOLO と OpenCV を組み合わせて、シンプルな構成で動作する車両カウントシステムを実装しました。

YOLOによる検出と ByteTrack を利用した追跡により、車両がカウントラインを通過するタイミングを高い精度で捉えることができました。

Python と軽量な技術スタックのみで動作するため、ローカル環境でも手軽に試せるほか、交通量調査や映像解析の自動化など、さまざまな用途に応用できます。

畑井駿佑

畑井駿佑

micomia株式会社の代表取締役です。 エンジニア、プロジェクトマネージャーを経験し、2024年にUI/UXにこだわった使いやすいシステム/アプリを開発するmicomia株式会社を設立しました。

関連記事

問い合わせフォームに届く営業メールが多い|迷惑を減らす対策とAI自動ブロック
AI

問い合わせフォームに届く営業メールが多い|迷惑を減らす対策とAI自動ブロック

問い合わせフォームに大量に届く営業メールに困っていませんか?営業メールが来る仕組み、従来の対策の限界、AIで自動ブロックできるFormGuardの仕組みまで、実務目線で解決策を解説します。

建設業向けマッチングアプリ開発|業界特化機能と費用感を解説
開発Tips

建設業向けマッチングアプリ開発|業界特化機能と費用感を解説

建設業向けマッチングアプリの開発について、業界特化の機能設計、案件マッチング・職人マッチングの違い、費用相場、開発期間、成功事例まで網羅。建設DX領域への参入を検討する事業者向けの実務ガイドです。

神戸でアプリ開発会社を選ぶ5つのポイント|失敗しない発注先の見極め方
開発Tips

神戸でアプリ開発会社を選ぶ5つのポイント|失敗しない発注先の見極め方

神戸・兵庫でアプリ開発会社を選ぶ際のチェックポイントを解説。地元対応力・実績・技術スタック・UI/UX品質・契約形態など、初めて発注する方でも失敗しないための判断軸を神戸特有の事情を踏まえて紹介します。

ユーザー視点になってアプリ開発 | micomiaでエンジニアとして働く
その他

ユーザー視点になってアプリ開発 | micomiaでエンジニアとして働く

観葉植物アプリ「でぃぐりーん」の開発事例をもとに、企画から実装・テスト・リリースまでのプロセスと、ユーザー体験を重視した開発の考え方を詳しく紹介します。

園芸のハードルを下げるには何が必要か グリラボ開発の出発点
開発Tips

園芸のハードルを下げるには何が必要か グリラボ開発の出発点

グリラボは、園芸初心者の不安を減らし、植物を育てる楽しさを広げるために生まれたアプリです。開発の出発点と狙いを紹介します。

「いつ水やりすればいいの?」にすぐ答える 育成ガイドを入れた理由
開発Tips

「いつ水やりすればいいの?」にすぐ答える 育成ガイドを入れた理由

グリラボの育成ガイドは、園芸初心者の小さな疑問にすぐ答えるための機能です。季節ごとのお手入れ支援をどう設計したかを紹介します。

初心者でも「自分にできそう」と思えること グリラボが目指した園芸体験の設計
開発Tips

初心者でも「自分にできそう」と思えること グリラボが目指した園芸体験の設計

グリラボは、園芸初心者が「自分にもできそう」と思える体験を大切にしています。心理的ハードルを下げる設計思想を紹介します。

なぜ園芸アプリに参考価格機能を入れたのか 剪定・伐採・抜根の不安に向き合う設計
開発Tips

なぜ園芸アプリに参考価格機能を入れたのか 剪定・伐採・抜根の不安に向き合う設計

グリラボは、剪定・伐採・抜根の参考価格を確認できる機能を搭載しています。料金の不透明さに向き合った理由を紹介します。

アップデート前の今、あえて残しておきたい グリラボ現バージョンの設計と次の改善テーマ
開発Tips

アップデート前の今、あえて残しておきたい グリラボ現バージョンの設計と次の改善テーマ

グリラボはアップデートを見据えつつ、現バージョンにも大きな意味があります。ローコードからスクラッチへの転換と次の改善テーマを紹介します。

植物の管理を「楽しみ」に変えるための工夫
開発Tips

植物の管理を「楽しみ」に変えるための工夫

グリラボは、雑草スタンプラリーや図鑑登録で植物とのつながりを楽しくしています。管理を楽しみに変える設計思想を紹介します。

文字を詰め込まないことが、やさしさになる グリラボのデザイン設計
開発Tips

文字を詰め込まないことが、やさしさになる グリラボのデザイン設計

グリラボは、文字を詰め込まずイラストを活用した分かりやすいデザインを採用しています。初心者向けのUI/UX設計を紹介します。

園芸アプリにAIをどう入れるか グリラボがAI機能を豊富に展開した理由
開発Tips

園芸アプリにAIをどう入れるか グリラボがAI機能を豊富に展開した理由

グリラボは、AIチャット、病気判定、活力度チェック、剪定AIなどを備えた園芸アプリです。AI機能を豊富に展開した理由を紹介します。

APIとは?仕組み・種類・活用事例をわかりやすく解説|アプリ開発での重要性
開発Tips

APIとは?仕組み・種類・活用事例をわかりやすく解説|アプリ開発での重要性

APIとは何かを初心者向けにわかりやすく解説。仕組みやREST・GraphQLなどの種類、ビジネスでの活用例まで詳しく紹介します。

アプリ内課金(IAP)とは?仕組み・種類・導入方法をわかりやすく解説
開発Tips

アプリ内課金(IAP)とは?仕組み・種類・導入方法をわかりやすく解説

アプリ内課金(IAP)とは、アプリ内でコンテンツや機能を購入できる仕組みです。種類や実装方法、ビジネス活用をわかりやすく解説します。

仕様書とは?アプリ・システム開発における役割・種類・作り方をわかりやすく解説
開発Tips

仕様書とは?アプリ・システム開発における役割・種類・作り方をわかりやすく解説

仕様書とは何かを初心者向けにわかりやすく解説。役割や種類、作り方、開発プロジェクトでの重要性まで詳しく紹介します。

SaaSとは?仕組み・メリット・代表的なサービス例をわかりやすく解説
開発Tips

SaaSとは?仕組み・メリット・代表的なサービス例をわかりやすく解説

SaaSとは何かを初心者向けにわかりやすく解説。仕組みやメリット・デメリット、代表的なサービス例、ビジネスでの活用方法まで詳しく紹介します。

WebRTCとは?仕組み・活用事例・リアルタイム通信の実装をわかりやすく解説
開発Tips

WebRTCとは?仕組み・活用事例・リアルタイム通信の実装をわかりやすく解説

WebRTCとは、ブラウザ間でリアルタイムに音声・映像・データを直接やり取りできる技術です。仕組みやビジネス活用をわかりやすく解説します。

TypeScriptとは?特徴・JavaScriptとの違い・開発での活用をわかりやすく解説
開発Tips

TypeScriptとは?特徴・JavaScriptとの違い・開発での活用をわかりやすく解説

TypeScriptとは、JavaScriptに静的型付けを追加したプログラミング言語です。特徴やメリット、ビジネス活用をわかりやすく解説します。

受託開発とパッケージ開発の違いとは?特徴・費用・選び方をわかりやすく解説
費用

受託開発とパッケージ開発の違いとは?特徴・費用・選び方をわかりやすく解説

受託開発とパッケージ開発の違いを費用・期間・柔軟性の観点で比較。自社に最適な開発手法の選び方と、ノーコード活用の第3の選択肢を解説します。

UI/UXデザインとは?違い・重要性・アプリ開発での実践方法を解説
開発Tips

UI/UXデザインとは?違い・重要性・アプリ開発での実践方法を解説

UI/UXデザインの違いや重要性、デザインプロセスを初心者向けにわかりやすく解説。アプリ開発における実践方法やビジネスへの効果まで詳しく紹介します。