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フレームずつ画像として取り出します。以降の処理はすべてフレーム単位で行われます。

video splited frames

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

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

detect & tracking

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株式会社を設立しました。

関連記事

ユーザーが迷わない画面体験と運営の管理画面|メディカルサークルのUI/UX②
開発Tips

ユーザーが迷わない画面体験と運営の管理画面|メディカルサークルのUI/UX②

医学部生向けノートアプリ「メディカルサークル」の画面 UX と管理画面設計。アップロード導線、ファイル種別の視認性、ゲスト→会員導線、退会フロー、ボトムナビと FAB の配置、React 製管理画面の俯瞰性を解説します。

RevenueCat でサブスクを Firestore と同期する|メディカルサークル Pro の課金実装
開発Tips

RevenueCat でサブスクを Firestore と同期する|メディカルサークル Pro の課金実装

医学部生向けノートアプリ「メディカルサークル」の有料プラン実装。RevenueCat の Entitlement Identifier の落とし穴、Firestore との二重反映、一元化された課金プロバイダ、購入の復元の検証フローまで解説します。

通報・ブロック・非表示で安心を設計する|メディカルサークルのコミュニティ機能
開発Tips

通報・ブロック・非表示で安心を設計する|メディカルサークルのコミュニティ機能

医学部生向けノートアプリ「メディカルサークル」のコミュニティ設計。通報・ブロック・コンテンツ非表示の3機能を別コレクションで分離し、ストリーム監視やセキュリティルールで安全性とパフォーマンスを両立した実装を紹介します。

医療×学術の信頼感を作るデザインシステム|メディカルサークルのUI設計
開発Tips

医療×学術の信頼感を作るデザインシステム|メディカルサークルのUI設計

医学部生向けノートアプリ「メディカルサークル」のデザインシステム。余白・角丸・色数のルール化、メディカルブルーの配色、Noto Sans JP の段階設計、共通ウィジェットの先行構築、空状態・エラー UI の作り方を解説します。

恋愛系マッチングアプリを作りたいと思ったら読む記事|開発会社が教える、作る前に詰めるべきこと
開発Tips

恋愛系マッチングアプリを作りたいと思ったら読む記事|開発会社が教える、作る前に詰めるべきこと

恋愛系マッチングアプリを作りたい方へ。開発相談を多数受けてきた開発会社の視点で、作る前に知っておくべき「アイデアの詰めが甘い」6つの失敗パターン、それでも作る価値がある条件、事前に詰めるべき3点を解説します。

省人化とは?意味・読み方と中小企業のバックオフィス業務で進める具体的な方法
DX

省人化とは?意味・読み方と中小企業のバックオフィス業務で進める具体的な方法

省人化の読み方・意味から、業務効率化・自動化との違い、中小企業のバックオフィス業務で実現する具体的な4つのパターンと3ステップの進め方、ツール選定の罠までを一本で解説します。

SNSアプリの作り方完全ガイド|開発費用・作成手順・必要機能・成功事例まとめ
開発Tips

SNSアプリの作り方完全ガイド|開発費用・作成手順・必要機能・成功事例まとめ

SNSアプリの作り方を「パッケージ開発」と「オーダーメイド開発」で徹底比較。依頼前に整理すべき機能・予算・ターゲットのポイントと、micomiaの開発実績を交えてわかりやすく解説します。

【これ一本で丸わかり】FlutterFlowとは?できること・料金・日本語対応・iOS/Android開発までわかりやすく解説
FlutterFlow

【これ一本で丸わかり】FlutterFlowとは?できること・料金・日本語対応・iOS/Android開発までわかりやすく解説

FlutterFlowとは何か、できること・料金プラン・日本語対応・信頼性をわかりやすく解説。iOS/Android/Webアプリをノーコードで開発できるローコードツールの基本と、開発実績80記事を持つmicomiaが解説します。

ノーコードでアプリ開発はどこまでできる?Adalo→FlutterFlow移行の実例で限界と本番化を解説
開発Tips

ノーコードでアプリ開発はどこまでできる?Adalo→FlutterFlow移行の実例で限界と本番化を解説

ノーコードツールでのアプリ開発の実態を解説。Adalo・Click・Glideなど無料で使えるノーコードツールの特徴やメリット・デメリット、初心者がつまずきやすいポイントを紹介します。

システム受託開発とは?依頼前に知るべき流れ・契約形態・費用相場
開発Tips

システム受託開発とは?依頼前に知るべき流れ・契約形態・費用相場

システム受託開発の基本から、契約形態(請負・準委任)の違い、費用相場、依頼の流れ、失敗しないパートナー選びまで体系的に解説。発注を検討中のB2B担当者・経営者向けの実務ガイドです。

要件定義が曖昧でも相談してよいのか|アプリ開発の進め方をわかりやすく解説
開発Tips

要件定義が曖昧でも相談してよいのか|アプリ開発の進め方をわかりやすく解説

要件定義が曖昧でもアプリ開発会社に相談してOK。早い段階で専門家に相談するメリットやMVPアプローチの活用法を解説。micomiaではアイデア段階からのご相談を歓迎しています。

FlutterFlowとFlutterの違いとは?特徴・開発スピード・使い分けを徹底比較
FlutterFlow

FlutterFlowとFlutterの違いとは?特徴・開発スピード・使い分けを徹底比較

FlutterFlowとFlutterの違いを開発スピード・カスタマイズ性・必要スキルの観点で比較。プロジェクトに応じた使い分けの判断基準を解説します。

FlutterFlowとBubbleの違いとは?特徴・料金・選び方を徹底比較
FlutterFlow

FlutterFlowとBubbleの違いとは?特徴・料金・選び方を徹底比較

FlutterFlowとBubbleの違いを徹底比較。対応プラットフォーム・開発アプローチ・料金・パフォーマンスなど多角的に解説し、プロジェクトに合った選び方を紹介します。

開発後の保守運用で必要なこととは?コスト・体制・よくある課題を解説
開発Tips

開発後の保守運用で必要なこととは?コスト・体制・よくある課題を解説

開発後の保守運用で必要な業務内容・コスト目安・よくある課題を解説。障害対応やセキュリティ対策、属人化防止のポイントをmicomiaの経験をもとに紹介します。

FlutterFlowでStripe決済を導入する方法|設定手順・注意点をわかりやすく解説
FlutterFlow

FlutterFlowでStripe決済を導入する方法|設定手順・注意点をわかりやすく解説

Stripeとは何かを初心者向けにわかりやすく解説。FlutterFlowとの連携方法や決済の仕組み、導入手順、ビジネスでの活用事例まで詳しく紹介します。

Webアプリとネイティブアプリ、どっちが正解? 50個の事例から分析
開発Tips

Webアプリとネイティブアプリ、どっちが正解? 50個の事例から分析

Webアプリとネイティブアプリは、どちらが優れているかではなく、用途に対してどちらが適切かで決まります。大企業アプリ50件の分析フレームをもとに、選び方を整理します。

神戸でASO対策ならmicomia|App Store最適化でダウンロード数を増やす方法
開発Tips

神戸でASO対策ならmicomia|App Store最適化でダウンロード数を増やす方法

神戸でASO対策(App Store最適化)をお考えの方向けに、ASOの基本施策・効果測定方法・micomiaの支援内容をまとめて解説。アプリのダウンロード数を増やす実践的な手法を、神戸拠点の開発会社が紹介します。

サーバーサイドレンダリング(SSR)とは?
開発Tips

サーバーサイドレンダリング(SSR)とは?

サーバーサイドレンダリング(SSR)とは、Webページの描画をサーバー側で行い完成したHTMLを返す手法です。CSRとの違いやSEO効果、Next.jsなどのフレームワーク、ビジネス活用を初心者にもわかりやすく解説します。

関西のアプリ開発会社おすすめの選び方|大阪・神戸・京都で依頼する際のポイント
開発Tips

関西のアプリ開発会社おすすめの選び方|大阪・神戸・京都で依頼する際のポイント

関西エリア(大阪・神戸・京都)でアプリ開発会社を探している方向けに、選び方のポイントと地域特性をまとめました。神戸・兵庫拠点で開発を行うmicomiaの強みも紹介。地元企業との対面打ち合わせを重視したい方に。

事業計画書・補助金申請用のアプリ/システム開発見積もり|企画段階でも無料でお打ち合わせ
開発Tips

事業計画書・補助金申請用のアプリ/システム開発見積もり|企画段階でも無料でお打ち合わせ

事業計画書や補助金申請のためにアプリ・システム開発の見積もりが必要な方向けに、企画段階での見積もり対応や無料のお打ち合わせについて解説。IT導入補助金・ものづくり補助金の申請に間に合うスピード対応もご紹介します。

OpenCV × YOLOで作る車両カウントシステム | micomia株式会社