December 2, 2025 (4mo ago)

関数型プログラミング vs オブジェクト指向: クイックガイド

関数型プログラミング vs オブジェクト指向:パラダイムを比較し、実践的なコード例を示し、プロジェクトに最適なアプローチの選び方を学ぶ。

← Back to blog
Cover Image for 関数型プログラミング vs オブジェクト指向: クイックガイド

関数型プログラミング vs オブジェクト指向:パラダイムを比較し、実践的なコード例を示し、プロジェクトに最適なアプローチの選び方を学ぶ。

関数型プログラミング vs オブジェクト指向: クイックガイド

関数型プログラミング vs オブジェクト指向:パラダイムを比較し、実践的なコード例を示し、プロジェクトに最適なアプローチの選び方を学ぶ。

手描きの図で関数型プログラミングとオブジェクト指向の原則を比較し、概念を示したイラスト。

はじめに

関数型プログラミングとオブジェクト指向プログラミングの本質的な違いは、注目する対象の違いに帰着します:コードを何に基づいて構成するのか、それとも誰がそれを実行するのか?関数型プログラミング(FP)は「何をするか」を重視し、ソフトウェアを状態を持たない数学的関数呼び出しの連鎖として扱います。オブジェクト指向プログラミング(OOP)は「誰がそれをするか」を重視し、状態を持つオブジェクト同士の相互作用を中心にロジックを組織します。本ガイドでは両方のパラダイムを比較し、TypeScript と React の実用的な例を示し、プロジェクトに最適なアプローチの選び方を支援します。

関数型 vs オブジェクト指向の簡単な比較

関数型スタイルとオブジェクト指向スタイルのどちらを選ぶかは単なる技術的な決定ではなく、ソフトウェアを構造化しデータフローを扱うためのマインドセットへのコミットメントです。このハイレベルな比較は、具体的な例や実践的なトレードオフの前提を設定します。

基本的な哲学的相違

OOP はデータとそれを操作する関数をオブジェクトにまとめることで世界をモデル化します。名前やメールなどのデータを保持し、updateEmail()sendPasswordReset() のようなメソッドを公開する User オブジェクトを想像してください。各オブジェクトは自分自身の状態を管理します。

FP は逆のアプローチを取ります。データと振る舞いを分離します。データは不変であることが多く、純粋関数は入力を受け取り副作用を伴わずに新しい出力を返します。これにより、大きな OOP システムでしばしば現れる共有状態の絡み合いが減少します。

“後でコードについて推論することが本当の課題だ。FP は移動する部品を最小化し、OOP はそれらの部品を理解しやすいコンポーネントとして整理する。”

多くの現代的なチームは両方のパラダイムからアイデアを借用して両者の利点を得ています。まずは基本的な違いの要約です。

関数型とオブジェクト指向の主な違い

ConceptFunctional Programming (FP)Object-Oriented Programming (OOP)
Primary Unit関数オブジェクト
State Management不変状態可変状態
Data & Operations分離される一緒にカプセル化される
Concurrency取り扱いが容易、ステートレス明示的なロックや同期が必要
Flow Control関数呼び出しと合成メソッド、ループ、条件分岐
Key Principles純粋関数、不変性カプセル化、継承、多態性

コア哲学の理解

これらのパラダイムは単なる構文の選択以上のものであり、データや振る舞い、システム構築の考え方です。どちらがプロジェクトに適しているかは、変化と複雑さをどのように管理したいかによります。

コア哲学を示す概念図、数学的ルーツ、純粋スタック、不変な関数、投資されたオブジェクト、多態性を示す図。

関数型プログラミングのパラダイム

FP はラムダ計算に遡り、計算を数学的関数の評価として扱います。純粋関数は同じ入力に対して常に同じ出力を返し、副作用を生じさせないことが中心です。不変性はインプレースの変更を防ぎ、新しい値を返すことを奨励します。

この予測可能性により、特に共有可変状態がバグの原因となりやすい並行システムにおいて、FP のコードはテストや推論が容易になります。FP の単方向データフローはデータ変換のための明確なパイプラインを生み出す傾向があります。

オブジェクト指向プログラミングのパラダイム

OOP は状態と振る舞いの両方をカプセル化する相互作用するオブジェクトとしてシステムをモデル化します。カプセル化は内部の詳細を隠し、オブジェクトがシステムの残りに対して単純なインターフェースを提供することを可能にします。継承と多態性はコードの再利用と柔軟な抽象化をサポートします。

このアプローチは複雑なドメインエンティティとその相互作用をモデル化するのに適しています。ドメインの概念が安定しており、オブジェクトの関係がビジネスルールに自然に対応する場合、OOP は直感的で保守しやすい設計を生み出せます。

保守性とスケーラビリティの比較

パラダイムの選択は長期的な保守性とシステムのスケール方法に影響します。FP と OOP はどちらも有用な戦略を提供しますが、複雑さへのアプローチは異なります。

保守性:予測可能性 vs カプセル化

FP は純粋関数と不変性を通じて予測可能性を重視します。同じ入力に対して常に同じ結果を返す関数はテストと分離が容易です。OOP は関連するデータと振る舞いを一緒に整理することで、自己完結型コンポーネントを調べることでシステムを理解しやすくします。

トレードオフ:FP の明快さはデータと振る舞いを分離することから来ており、OOP の明快さはそれらをグルーピングすることから来ています。これによりデバッグのワークフローに影響が出ます—FP はしばしばバグを特定の関数に狭めますが、OOP では複数のオブジェクト間の相互作用を追う必要があるかもしれません。

スケーラビリティ:並行処理と並列処理

FP のステートレスなアプローチは、純粋関数が共有データを変更しないため、並行処理や並列処理を単純化します。これによりロックや同期の必要性が減ります。OOP を並行化することは可能ですが、しばしば競合状態を避けるために慎重な状態管理と同期メカニズムが必要になります。

FP は予測可能で並行性が重要なドメインで注目を集めていますが、主流の開発者採用はマルチパラダイムおよびオブジェクト指向言語に集中しています 1.2

実例:TypeScript と React

以下では同じユーザー設定フォームを 2 つのスタイルで構築します:クラシックな React のクラスコンポーネント(OOP)と Hooks を使ったモダンな関数コンポーネント(FP スタイル)。これによりアーキテクチャの選択が状態管理、ロジックの再利用、構造にどのように影響するかが浮き彫りになります。

クラスベースの React コンポーネントから関数型の React Hooks への移行を示す図。

OOP アプローチ:React クラスコンポーネント

クラスコンポーネントは状態とメソッドを単一のオブジェクトにまとめ、カプセル化の OOP モデルに適合します。

import React, { Component } from 'react';

interface UserSettings {
  name: string;
  email: string;
}

class UserSettingsForm extends Component<{}, UserSettings> {
  state = {
    name: 'Jane Doe',
    email: 'jane.doe@example.com',
  };

  handleChange = (event: React.ChangeEvent<HTMLInputElement>) => {
    const { name, value } = event.target;
    this.setState({ [name]: value } as Pick<UserSettings, keyof UserSettings>);
  };

  handleSubmit = (event: React.FormEvent) => {
    event.preventDefault();
    console.log('Submitting data:', this.state);
  };

  render() {
    return (
      <form onSubmit={this.handleSubmit}>
        <input name="name" value={this.state.name} onChange={this.handleChange} />
        <input name="email" value={this.state.email} onChange={this.handleChange} />
        <button type="submit">Save Settings</button>
      </form>
    );
  }
}

このパターンは関連するデータと振る舞いをまとめて保持しますが、大きなコンポーネントは高階コンポーネントのような追加パターンがないとリファクタリングや再利用が難しくなることがあります。

関数型へのリファクタ:Hooks と純粋なヘルパー

関数型アプローチは状態と純粋関数を分離し、個々の部分をテストや再利用しやすくします。

import React, { useState } from 'react';

const formatUserDataForApi = (name: string, email: string) => ({
  userName: name,
  userEmail: email,
});

const UserSettingsFormFunctional = () => {
  const [name, setName] = useState('Jane Doe');
  const [email, setEmail] = useState('jane.doe@example.com');

  const handleChange = (event: React.ChangeEvent<HTMLInputElement>) => {
    const { name, value } = event.target;
    if (name === 'name') {
      setName(value);
    } else {
      setEmail(value);
    }
  };

  const handleSubmit = (event: React.FormEvent) => {
    event.preventDefault();
    const payload = formatUserDataForApi(name, email);
    console.log('Submitting data:', payload);
  };

  return (
    <form onSubmit={handleSubmit}>
      <input name="name" value={name} onChange={handleChange} />
      <input name="email" value={email} onChange={handleChange} />
      <button type="submit">Save Settings</button>
    </form>
  );
};

React の Hooks は関数内で状態と副作用を可能にすることでこのスタイルを一般化し、モダンな React 開発で広く採用されるようになりました 3

正しいパラダイムの選び方

これは勝ち負けの議論ではありません。正しい選択はチーム、問題領域、長期的な目標によります。実用的な判断の指針として以下を参考にしてください。

OOP を選ぶべきとき

リッチなドメインエンティティと永続的な状態と振る舞いをモデル化する場合は OOP を選びましょう。例えば:

  • 多くの関連モジュールを持つ大規模なエンタープライズシステム
  • コンポーネントレベルの状態がオブジェクトに自然に対応するリッチで状態フルな UI コンポーネント
  • 概念が安定して明確に定義されたドメイン

FP を選ぶべきとき

予測可能性、並行性、データ変換が優先される場合は FP を選びましょう。例えば:

  • データ処理パイプラインや ETL タスク
  • 状態同期がコスト高となる並行または並列システム
  • 関数がアルゴリズムに自然に対応する数学的・科学的計算

実用的な意思決定チェックリスト

  1. データの性質は何か:状態を持つオブジェクトか、それとも変換可能なフローか?
  2. 性能と正しさのために並行性はどれほど重要か?
  3. チームの専門知識と新しいパターンを学ぶ意欲はどれほどか?
  4. 一方のパラダイムに無理に寄せることなくハイブリッドアプローチは可能か?

ハイブリッドアプローチの採用

多くの成功したシステムは両方のパラダイムを組み合わせています。TypeScript や Python のようなマルチパラダイム言語では、クラス内で不変データを使ったり、コレクションで mapfilter を使ったり、コアビジネスロジックのために純粋関数を分離したりできます。

実践におけるパラダイムの組み合わせ

一般的なハイブリッドパターン:

  • クラス内での不変状態:メソッドは内部状態を変更するのではなく新しいインスタンスを返す。
  • サービスのための純粋関数:入力を受け取り出力を返すステートレスな関数としてビジネスロジックを実装する。
  • 関数型コレクションメソッド:配列を変更するループの代わりに mapfilterreduce を使用する。

オブジェクトを「モノ」を表すために使い、純粋関数をそれらの間の振る舞いをオーケストレーションするために使いましょう。その分離により明快さとテスト容易性が向上します。

オブジェクト指向(OP)概念と map filter を経るフローで関数型(FP)を比較する図。

よくある質問

関数型はオブジェクト指向より高速ですか?

パフォーマンスは問題、言語、ランタイムによって異なります。FP の不変性は割り当てオーバーヘッドを増やすことがありますが、ステートレスな性質は並列システムでスループットを改善する可能性があります。インプレース更新に依存するシングルスレッドの処理は OOP スタイルのミューテーションの方が速い場合があります。

関数型とオブジェクト指向のコードを混在させてもいいですか?

はい。パラダイムの混在は一般的で、しばしば最も実用的な選択です。コアエンティティはオブジェクトでモデル化し、複雑なロジックは純粋関数で表現しましょう。

初心者はどちらのパラダイムを先に学ぶべきですか?

OOP はクラスとオブジェクトが現実世界の概念に直感的に対応するため、最初は理解しやすいことが多いです。ただし、純粋関数や不変性などの関数型の基本概念を早期に学ぶことはコード品質を高める習慣を築く助けになります。

結論と推奨事項

  1. 問題に応じてパラダイムを選ぶ。予測可能なデータ変換と並行性には FP を、リッチなドメインモデリングには OOP を使う。
  2. コアロジックには小さな純粋関数を優先する。コンポーネントやクラスは小さく焦点を絞る。
  3. 有用ならハイブリッドアプローチを採用する。コードベース全体で単一のパラダイムに固執する必要はない。

実用的な Q&A

Q: 既存のコードベースで FP と OOP のどちらを選ぶかはどう決める?

A: 最大の問題点を評価してください。バグが共有可変状態に起因しているなら、不変性と純粋関数を導入しましょう。ドメインが自然にオブジェクトベースであればオブジェクトを維持しつつステートレスなサービスを抽出してください。

Q: OOP プロジェクトに安全に FP 原則を導入するには?

A: 小さくテスト可能なサービスを純粋関数として実装することから始め、関数型の配列メソッドを追加し、状態を変更する代わりに新しいオブジェクトインスタンスを返すことを検討してください。

Q: 今日すぐに保守性を改善するためのクイックウィンは?

A: 小さい関数を強制し、純粋ロジックの自動テストを追加し、破壊的なループの代わりに map/filter を使用し、状態を持つオブジェクトの不変条件を文書化してください。


1.
https://insights.stackoverflow.com/survey/2023 — Stack Overflow Developer Survey: マルチパラダイム言語やスクリプト言語の優勢な使用状況を示し、ニッチな関数型言語と比較した調査結果。
2.
https://octoverse.github.com/ — GitHub Octoverse のレポートと言語使用傾向。JavaScript、TypeScript、Python の広範な採用と純粋な関数型言語のより小さいシェアを示す。
3.
https://reactjs.org/docs/hooks-intro.html — React の Hooks に関するドキュメント。Hooks は関数コンポーネントと状態ロジックの再利用を普及させた。
4.
https://www.cs.kent.ac.uk/people/staff/dat/miranda/whyfp.html — John Hughes, “Why Functional Programming Matters”: 関数型設計の利点に関する基礎的なエッセイ。
← Back to blog
🙋🏻‍♂️

AIがコードを書きます。
あなたがそれを長持ちさせます。

AI加速の時代において、クリーンコードは単なる良い実践ではありません—スケールするシステムと自らの重みで崩壊するコードベースの違いです。