フリーキーズ | 独学プログラミング

データ構造入門!配列・リスト・スタック・キューを初心者にもわかりやすく解説

リンドくん

リンドくん

たなべ先生、「データ構造」って言葉をよく聞くんですけど、これって何なんですか?プログラミングで重要って聞くんですが...

たなべ

たなべ

データ構造は、プログラミングにおいてデータを効率的に整理・管理するための仕組みなんだ。
例えば、本を本棚にしまうとき、ジャンル別に分けたり、著者名順に並べたりするよね?それと同じように、コンピュータの中でデータを整理する方法がデータ構造なんだよ。

プログラミングを学び始めると、必ず出会うのが「データ構造」という概念です。
しかし、その重要性は理解していても、実際にどう使うのか、なぜ必要なのかがわからないという方も多いのではないでしょうか?

この記事では、プログラミングを始めたばかりの初心者の方でも理解できるよう、配列・リスト・スタック・キューという4つの基本的なデータ構造について、できるだけ分かりやすく解説していきます。

オンラインコミュニティ運営しています

HackATAは、IT技術を習得する人のために広く開かれたオンラインコミュニティです。 現在、無料コミュニティとして開放していますので、ご気軽に参加してください。

✓ 再立ち上げ

✓ コミュニティの方向性について意見募集中

HackATA公式Webサイト

データ構造とは?なぜ必要なのか

リンドくん

リンドくん

でも先生、普通に変数を使えばデータは保存できますよね?なぜわざわざデータ構造なんて考える必要があるんですか?

たなべ

たなべ

たしかにそう思うよね。でも、例えば100人分の学生の成績を管理するとしたら、score1score2...score100って変数を100個作るのは現実的じゃないよね?
そういうときに、データ構造を使うと効率的に管理できるんだよ。

データ構造の基本概念

データ構造とは、複数のデータを効率的に保存・整理・操作するための仕組みです。
プログラムにおいて、データをどう保存するかによって、以下のような違いが生まれます。

  • アクセスの速さ - 必要なデータをすぐに取り出せるか
  • メモリの効率 - 無駄なメモリを使っていないか
  • 操作のしやすさ - データの追加・削除・変更が簡単にできるか

なぜデータ構造が重要なのか

プログラミングにおいて、データ構造を理解することは次のような理由で重要です。

  1. パフォーマンスの向上 - 適切なデータ構造を選ぶことで、プログラムの実行速度が大きく変わります
  2. コードの可読性 - データの扱い方が明確になり、コードが理解しやすくなります
  3. 問題解決能力 - 複雑な問題を効率的に解決できるようになります
  4. 面接での必須知識 - エンジニアの採用面接では必ずといっていいほど問われます

例えば、1000件のデータから特定の項目を探す場合、データ構造の選び方次第で処理時間が数秒から一瞬に短縮できることもあるのです。

今回学ぶ4つのデータ構造

この記事では、最も基本的で重要な4つのデータ構造を学んでいきます。

  • 配列(Array) - 同じ型のデータを連続して保存
  • リスト(List) - データを順番に並べて管理
  • スタック(Stack) - 後入れ先出し(LIFO)の構造
  • キュー(Queue) - 先入れ先出し(FIFO)の構造

これらは、より複雑なデータ構造やアルゴリズムを理解する上での基礎中の基礎となります。しっかりマスターしていきましょう!

配列(Array)- データを連続して保存する

リンドくん

リンドくん

配列って何ですか?変数とどう違うんですか?

たなべ

たなべ

配列は、同じ種類のデータを箱に入れて並べたものと考えるとわかりやすいよ。
例えば、5人の学生の点数を管理するとき、scores[0], scores[1]...という風に、一つの名前で複数のデータを管理できるんだ。

配列の基本概念

配列は、同じデータ型の要素を連続したメモリ領域に格納するデータ構造です。
最も基本的で、ほぼすべてのプログラミング言語で利用できます。

配列の主な特徴は以下の通りです。

  • インデックスでアクセス - 0から始まる番号(インデックス)で各要素にアクセスできます
  • 固定サイズ - 多くの言語では、一度作成すると配列のサイズは変更できません
  • 高速なアクセス - インデックスを指定すれば、どの要素にも瞬時にアクセスできます
  • メモリ効率が良い - データが連続して配置されるため、メモリを効率的に使えます

Pythonでの配列の使い方

Pythonでは、リストを配列のように使用できます。

# 配列(リスト)の作成
scores = [85, 92, 78, 90, 88]

# 要素へのアクセス(インデックスは0から始まる)
print(scores[0])  # 85が表示される
print(scores[2])  # 78が表示される

# 要素の変更
scores[1] = 95
print(scores)  # [85, 95, 78, 90, 88]

# 要素の追加
scores.append(100)
print(scores)  # [85, 95, 78, 90, 88, 100]

# 配列の長さを取得
length = len(scores)
print(f"要素数: {length}")  # 要素数: 6

# 配列の全要素を表示
for i, score in enumerate(scores):
    print(f"{i}番目の点数: {score}")

配列の長所と短所

長所

  • インデックスによる高速なアクセス(O(1))
  • シンプルで理解しやすい
  • メモリ効率が良い

短所

  • サイズの変更が難しい(または非効率)
  • 要素の挿入・削除が遅い(O(n))
  • すべての要素が同じデータ型である必要がある

配列を使うべき場面

配列は以下のような場面で特に有効です。

  • 要素数が事前にわかっている場合 - 学生の成績管理など
  • ランダムアクセスが必要な場合 - 特定の位置の要素に頻繁にアクセスする
  • データの順序が重要な場合 - ゲームのスコアランキングなど

配列は最も基本的なデータ構造ですが、その分汎用性も高く、プログラミングにおいて欠かせない存在です。

リスト(List)- 柔軟にデータを管理する

リンドくん

リンドくん

配列とリストって何が違うんですか?同じように見えるんですけど...

たなべ

たなべ

確かに似ているね!大きな違いは柔軟性なんだ。
リストは配列と違って、要素の追加や削除が簡単で、サイズも自由に変更できるんだよ。実装方法も異なっていて、リンクという仕組みで要素同士がつながっているんだ。

リストの基本概念

リスト(特に連結リスト)は、各要素が次の要素への参照を持つデータ構造です。
これにより、配列よりも柔軟なデータ管理が可能になります。

リストには主に2種類あります。

  • 単方向リスト - 各要素が次の要素への参照のみを持つ
  • 双方向リスト - 各要素が前後の要素への参照を持つ

リストの主な特徴は以下です。

  • 動的なサイズ - 必要に応じて要素を追加・削除できます
  • 効率的な挿入・削除 - 特に先頭や中間への挿入・削除が効率的です
  • 順次アクセス - 先頭から順にアクセスする必要があります

Pythonでのリストの使い方

Pythonの標準リストは、実は配列とリストの両方の特性を持つ特殊な実装です。

# リストの作成
students = ["田中", "佐藤", "鈴木"]

# 要素の追加
students.append("高橋")  # 末尾に追加
students.insert(1, "伊藤")  # 1番目の位置に挿入
print(students)  # ['田中', '伊藤', '佐藤', '鈴木', '高橋']

# 要素の削除
students.remove("佐藤")  # 値を指定して削除
print(students)  # ['田中', '伊藤', '鈴木', '高橋']

deleted = students.pop()  # 末尾の要素を削除して取得
print(f"削除された要素: {deleted}")  # 削除された要素: 高橋
print(students)  # ['田中', '伊藤', '鈴木']

# リストの結合
other_students = ["山田", "渡辺"]
all_students = students + other_students
print(all_students)  # ['田中', '伊藤', '鈴木', '山田', '渡辺']

# リストのスライス
first_two = all_students[:2]
print(first_two)  # ['田中', '伊藤']

単純な連結リストの実装例

より本格的な連結リストの実装を見てみましょう。

# ノード(要素)のクラス
class Node:
    def __init__(self, data):
        self.data = data
        self.next = None

# 連結リストのクラス
class LinkedList:
    def __init__(self):
        self.head = None
    
    # 先頭に要素を追加
    def add_first(self, data):
        new_node = Node(data)
        new_node.next = self.head
        self.head = new_node
    
    # 末尾に要素を追加
    def add_last(self, data):
        new_node = Node(data)
        
        if self.head is None:
            self.head = new_node
            return
        
        current = self.head
        while current.next:
            current = current.next
        current.next = new_node
    
    # リストの内容を表示
    def display(self):
        elements = []
        current = self.head
        while current:
            elements.append(current.data)
            current = current.next
        print(" -> ".join(map(str, elements)))

# 使用例
linked_list = LinkedList()
linked_list.add_last(10)
linked_list.add_last(20)
linked_list.add_first(5)
linked_list.display()  # 5 -> 10 -> 20

リストの長所と短所

長所

  • 動的なサイズ変更が可能
  • 先頭への挿入・削除が高速(O(1))
  • メモリの効率的な利用

短所

  • ランダムアクセスが遅い(O(n))
  • 各要素が参照を保持するため、配列よりメモリを多く使う
  • 実装が複雑

リストを使うべき場面

リストは以下のような場面で特に有効です。

  • 頻繁な挿入・削除が必要な場合 - ToDo管理アプリなど
  • サイズが事前にわからない場合 - ユーザー入力の受付など
  • 順次処理が中心の場合 - ログファイルの処理など

スタック(Stack)- 後入れ先出しの構造

リンドくん

リンドくん

スタックって何ですか?配列やリストとどう違うんですか?

たなべ

たなべ

スタックは「後から入れたものが先に出てくる」という特殊なルールを持ったデータ構造なんだ。
お皿を積み重ねたときを想像してみて。一番上に載せたお皿が、取るときは一番最初に取れるよね?それがスタックなんだよ。

スタックの基本概念

スタックは、LIFO(Last In, First Out: 後入れ先出し)の原則に従うデータ構造です。
最後に追加された要素が最初に取り出されるという特徴があります。

スタックの基本操作は以下の3つです。

  • push(プッシュ) - スタックの一番上に要素を追加する
  • pop(ポップ) - スタックの一番上の要素を取り出して削除する
  • peek/top(ピーク/トップ) - スタックの一番上の要素を削除せずに参照する

Pythonでのスタックの実装

Pythonでは、リストを使って簡単にスタックを実装できます。

# リストを使ったスタックの実装
class Stack:
    def __init__(self):
        self.items = []
    
    # 要素を追加(push)
    def push(self, item):
        self.items.append(item)
        print(f"{item}をスタックに追加しました")
    
    # 要素を取り出し(pop)
    def pop(self):
        if not self.is_empty():
            return self.items.pop()
        else:
            return "スタックが空です"
    
    # 一番上の要素を確認(peek)
    def peek(self):
        if not self.is_empty():
            return self.items[-1]
        else:
            return "スタックが空です"
    
    # スタックが空かどうか確認
    def is_empty(self):
        return len(self.items) == 0
    
    # スタックのサイズを取得
    def size(self):
        return len(self.items)
    
    # スタックの内容を表示
    def display(self):
        print("スタックの内容(上が先頭):")
        for item in reversed(self.items):
            print(f"  {item}")

# 使用例
stack = Stack()
stack.push(10)
stack.push(20)
stack.push(30)
stack.display()
# スタックの内容(上が先頭):
#   30
#   20
#   10

print(f"一番上の要素: {stack.peek()}")  # 一番上の要素: 30
print(f"取り出した要素: {stack.pop()}")  # 取り出した要素: 30
print(f"取り出した要素: {stack.pop()}")  # 取り出した要素: 20
stack.display()
# スタックの内容(上が先頭):
#   10

スタックの活用例

スタックは実際のプログラミングで非常によく使われます。

# ブラウザの「戻る」機能のシミュレーション
class BrowserHistory:
    def __init__(self):
        self.history = Stack()
        self.current_page = None
    
    def visit(self, url):
        if self.current_page:
            self.history.push(self.current_page)
        self.current_page = url
        print(f"{url}にアクセスしました")
    
    def back(self):
        if not self.history.is_empty():
            self.current_page = self.history.pop()
            print(f"{self.current_page}に戻りました")
        else:
            print("これ以上戻れません")
    
    def show_current(self):
        print(f"現在のページ: {self.current_page}")

# 使用例
browser = BrowserHistory()
browser.visit("google.com")
browser.visit("youtube.com")
browser.visit("github.com")
browser.show_current()  # 現在のページ: github.com
browser.back()  # youtube.comに戻りました
browser.back()  # google.comに戻りました
browser.show_current()  # 現在のページ: google.com

スタックを使うべき場面

スタックは以下のような場面で特に有効です。

  • 関数呼び出しの管理 - プログラムの実行スタック
  • アンドゥ機能 - テキストエディタの操作履歴
  • 式の評価 - 数式の計算や括弧のチェック
  • 深さ優先探索(DFS) - グラフやツリーの探索アルゴリズム

スタックは、シンプルながら非常に強力なデータ構造で、多くのアルゴリズムやシステムの基盤となっています。

キュー(Queue)- 先入れ先出しの構造

リンドくん

リンドくん

キューはスタックと何が違うんですか?

たなべ

たなべ

スタックとは逆で、キューは「先に入れたものが先に出てくる」仕組みなんだ。
銀行の窓口やレジの行列を想像してみて。先に並んだ人が先に対応されるよね?それがキューなんだよ。

キューの基本概念

キューは、FIFO(First In, First Out: 先入れ先出し)の原則に従うデータ構造です。
最初に追加された要素が最初に取り出されるという特徴があります。

キューの基本操作は以下の3つです。

  • enqueue(エンキュー) - キューの末尾に要素を追加する
  • dequeue(デキュー) - キューの先頭から要素を取り出して削除する
  • peek/front(ピーク/フロント) - キューの先頭の要素を削除せずに参照する

Pythonでのキューの実装

Pythonでは、collections.dequeを使うと効率的なキューを実装できます。

from collections import deque

# dequeを使ったキューの実装
class Queue:
    def __init__(self):
        self.items = deque()
    
    # 要素を追加(enqueue)
    def enqueue(self, item):
        self.items.append(item)
        print(f"{item}をキューに追加しました")
    
    # 要素を取り出し(dequeue)
    def dequeue(self):
        if not self.is_empty():
            return self.items.popleft()
        else:
            return "キューが空です"
    
    # 先頭の要素を確認(peek)
    def peek(self):
        if not self.is_empty():
            return self.items[0]
        else:
            return "キューが空です"
    
    # キューが空かどうか確認
    def is_empty(self):
        return len(self.items) == 0
    
    # キューのサイズを取得
    def size(self):
        return len(self.items)
    
    # キューの内容を表示
    def display(self):
        print("キューの内容(左が先頭):")
        print(f"  {list(self.items)}")

# 使用例
queue = Queue()
queue.enqueue("田中")
queue.enqueue("佐藤")
queue.enqueue("鈴木")
queue.display()
# キューの内容(左が先頭):
#   ['田中', '佐藤', '鈴木']

print(f"先頭の要素: {queue.peek()}")  # 先頭の要素: 田中
print(f"処理した要素: {queue.dequeue()}")  # 処理した要素: 田中
print(f"処理した要素: {queue.dequeue()}")  # 処理した要素: 佐藤
queue.display()
# キューの内容(左が先頭):
#   ['鈴木']

キューの実践的な活用例

キューは待ち行列や順次処理で広く使われます。

# プリンタの印刷キューのシミュレーション
class PrintQueue:
    def __init__(self):
        self.queue = Queue()
    
    def add_job(self, document):
        self.queue.enqueue(document)
        print(f"印刷ジョブ '{document}' を追加しました")
    
    def process_job(self):
        if not self.queue.is_empty():
            doc = self.queue.dequeue()
            print(f"印刷中: {doc}")
            return doc
        else:
            print("印刷ジョブはありません")
            return None
    
    def show_queue(self):
        print("\n現在の印刷待ちリスト:")
        if self.queue.is_empty():
            print("  なし")
        else:
            self.queue.display()

# 使用例
printer = PrintQueue()
printer.add_job("報告書.pdf")
printer.add_job("請求書.docx")
printer.add_job("プレゼン資料.pptx")
printer.show_queue()

print("\n--- 印刷処理開始 ---")
printer.process_job()
printer.process_job()
printer.show_queue()

優先度付きキューの概念

通常のキューの応用として、優先度付きキューというものもあります。

import heapq

class PriorityQueue:
    def __init__(self):
        self.items = []
    
    def enqueue(self, item, priority):
        # 優先度が小さいほど先に処理される
        heapq.heappush(self.items, (priority, item))
        print(f"{item}(優先度{priority})を追加しました")
    
    def dequeue(self):
        if not self.is_empty():
            priority, item = heapq.heappop(self.items)
            return item
        return "キューが空です"
    
    def is_empty(self):
        return len(self.items) == 0

# 使用例: 緊急度による処理順序
tasks = PriorityQueue()
tasks.enqueue("メール返信", 3)
tasks.enqueue("緊急バグ修正", 1)
tasks.enqueue("会議資料作成", 2)

print("\n処理順序:")
while not tasks.is_empty():
    print(f"  処理: {tasks.dequeue()}")

キューを使うべき場面

キューは以下のような場面で特に有効です。

  • 待ち行列の管理 - プリンタスプール、顧客サービスなど
  • 非同期処理 - タスクキュー、メッセージキュー
  • 幅優先探索(BFS) - グラフやツリーの探索アルゴリズム
  • バッファリング - データストリームの一時保存

キューは、順序を保ちながら公平に処理を行う必要がある場面で欠かせないデータ構造です。

データ構造の選び方と使い分け

リンドくん

リンドくん

いろんなデータ構造があって混乱してきました...結局どれを使えばいいんですか?

たなべ

たなべ

確かに最初は迷うよね。でも、それぞれの特徴を理解すれば、自然と適切なものを選べるようになるんだ。
具体的な場面ごとに、どのデータ構造が適しているか見ていこう。

各データ構造の比較表

データ構造追加削除検索アクセス主な用途
配列O(n)O(n)O(n)O(1)固定サイズ、頻繁なアクセス
リストO(1) ※1O(1) ※1O(n)O(n)動的サイズ、頻繁な挿入削除
スタックO(1)O(1)-O(1) ※2LIFO処理、履歴管理
キューO(1)O(1)-O(1) ※2FIFO処理、待ち行列

※1 先頭への操作の場合。末尾への操作は異なる場合があります。
※2 一番上/先頭の要素のみ

場面別の選択ガイド

ランダムアクセスが必要な場合

  • 選択: 配列
  • 理由: インデックスによる高速アクセス(O(1))
  • : ゲームのスコアボード、座席予約システム

頻繁な挿入・削除が必要な場合

  • 選択: リスト
  • 理由: 要素の挿入・削除が効率的
  • : ToDo管理、プレイリスト

最新のデータから処理したい場合

  • 選択: スタック
  • 理由: LIFO構造で最新データに素早くアクセス
  • : アンドゥ機能、ブラウザ履歴

到着順に処理したい場合

  • 選択: キュー
  • 理由: FIFO構造で公平な順序処理
  • : プリンタキュー、タスク管理

パフォーマンスを考慮した選択

データ量が大きくなると、パフォーマンスの違いが顕著になります。

  • 10個程度のデータ - どのデータ構造でも問題なし
  • 100~1000個のデータ - データ構造の選択がパフォーマンスに影響し始める
  • 10000個以上のデータ - 適切なデータ構造の選択が必須

実際の開発では、まず動くものを作り、パフォーマンスに問題が出てから最適化するのが良いアプローチです。
早すぎる最適化は避け、必要になったタイミングで改善することを心がけましょう。

まとめ

リンドくん

リンドくん

データ構造って、それぞれに得意なことがあるんですね!これからは場面に応じて使い分けられそうです。

たなべ

たなべ

その通り!最初は難しく感じるかもしれないけど、実際に使ってみることが一番の学習方法だよ。
まずは小さなプログラムから始めて、徐々に慣れていってね。困ったことがあったら、いつでも相談してね!

今回は、プログラミングの基礎となる4つのデータ構造について学びました。

重要なポイントをおさらいしましょう。

  • 配列 - インデックスによる高速アクセスが特徴。固定サイズでランダムアクセスが必要な場面に最適
  • リスト - 動的なサイズ変更と効率的な挿入・削除が可能。柔軟なデータ管理に適している
  • スタック - LIFO(後入れ先出し)構造。履歴管理やアンドゥ機能に活用
  • キュー - FIFO(先入れ先出し)構造。待ち行列や順次処理に最適

データ構造は、プログラミングスキルを次のレベルに引き上げる重要な要素です。
エンジニアとしてのキャリアを築く上で、これらの基礎知識は必ず役に立ちます。

この記事をシェア

関連するコンテンツ