Javaの抽象クラスを完全理解!基本の書き方から実用的な使い方まで徹底解説

2025年8月20日水曜日

Java

Javaの抽象クラスは、オブジェクト指向プログラミングを学ぶ上で多くの人がつまずきやすい概念の一つです。

interfaceと何が違うのか、どんな時に使うべきなのか、疑問に感じている人も少なくないでしょう。

この記事では、Javaの抽象クラスの基本から、interfaceとの明確な違い、そして実用的な使い方まで、サンプルコードを豊富に用いて丁寧に解説を進めます。

この記事で学べること

  • Javaにおける抽象クラスの基本的な役割と概念
  • 抽象クラスと抽象メソッドの具体的な書き方
  • interfaceとの機能・思想面での明確な違い
  • 開発現場で役立つ抽象クラスの活用パターン
  • 初心者が陥りがちな注意点と解決策

Javaの抽象クラスとは?

この章では、Javaの抽象クラスがどのようなものか、その根本的な概念から解説します。「抽象」という言葉の意味を紐解きながら、なぜプログラミングの世界でこの仕組みが必要とされるのかを明らかにします。

  • そもそも「抽象」ってどういう意味?
  • 抽象クラスを一言でいうと「中途半端な設計図」
  • なぜJavaには抽象クラスが必要なの?

そもそも「抽象」ってどういう意味?

プログラミングの話の前に、「抽象」という言葉について考えてみましょう。
抽象とは、個別の具体的な事柄から、共通する性質や要素を抜き出してまとめることを指します。

例えば、「犬」「猫」「鳥」は具体的な動物ですが、これらに共通する「鳴く」「食べる」「動く」といった性質を抜き出すと、「動物」という抽象的な概念になります。

Javaの抽象クラスも、この考え方に基づいています。

抽象クラスを一言でいうと「中途半端な設計図」

Javaの抽象クラスは、具体的な処理が決まっている部分(具象メソッド)と、まだ決まっていない部分(抽象メソッド)の両方を持つことができるクラスです。

例えるなら、「どの車種にも共通する部品(ハンドルやタイヤ)は設計済みだけど、エンジンの種類や座席の数は車種ごとに決めてね」というような、一部が未完成の設計図のようなものです。

この「未完成」な部分があるため、抽象クラスそのものから直接インスタンス(実体)を作ることはできません。

なぜJavaには抽象クラスが必要なの?

抽象クラスは、クラス間の共通点をまとめ、プログラムの骨格を作るために役立ちます。

複数のクラスに共通する機能やデータを一箇所に定義しておくことで、コードの重複を防ぎ、修正や管理がしやすくなります。

また、サブクラス(設計図を元に作られる具体的なクラス)に対して、特定メソッドの実装を強制するという重要な役割も担っています。これにより、プログラム全体の品質と一貫性を保つことができるのです。

まずは覚えよう!Java抽象クラスの基本的な書き方

ここでは、Javaで抽象クラスを定義するための具体的な構文を、サンプルコードを見ながら学習します。`abstract`キーワードの使い方や、抽象メソッド・具象メソッドの記述方法をマスターしましょう。

  • `abstract`キーワードを付けてクラスを定義する
  • 処理の中身がない「抽象メソッド」の作り方
  • 処理の中身がある「具象メソッド」も書ける

`abstract`キーワードを付けてクラスを定義する

Javaで抽象クラスを定義するには、`class`キーワードの前に`abstract`修飾子を付けます。たったこれだけで、そのクラスは抽象クラスとして扱われます。

// `abstract` を付けて抽象クラスを宣言
public abstract class Animal {
    // フィールド(変数)も普通に定義できる
    String name;

    // ... この後にメソッドなどを定義 ...
}

`abstract`を付けることで、このクラスはインスタンス化できなくなり、継承して使うことが前提となります。

処理の中身がない「抽象メソッド」の作り方

抽象メソッドは、メソッド名、戻り値の型、引数だけを定義し、具体的な処理(`{}`ブロック)を記述しないメソッドです。
こちらも`abstract`修飾子を付け、文末はセミコロン`;`で終えます。

public abstract class Animal {
    // ... フィールドなど ...

    // 鳴き声は動物ごとに違うので、処理は書かない
    // サブクラスでの実装を強制する
    public abstract void cry(); 
}

この抽象メソッドは、このクラスを継承するサブクラスで必ず実装(オーバーライド)しなければならない、というルールを生み出します。

処理の中身がある「具象メソッド」も書ける

抽象クラスの中には、通常のクラスと同じように、具体的な処理が書かれたメソッド(具象メソッド)も定義できます。
これは、どのサブクラスにも共通する処理をまとめるのに便利です。

public abstract class Animal {
    String name;

    // 抽象メソッド
    public abstract void cry();

    // 具象メソッド(共通の処理)
    public void eat(String food) {
        System.out.println(name + "は" + food + "を食べます。");
    }
}

このように、抽象クラスは「実装を強制する部分」と「共通処理をまとめる部分」を両立できる点が特徴です。

実践!Java抽象クラスを継承して使ってみよう

定義した抽象クラスを実際にどう使うのか、継承からメソッドの実装までの一連の流れを解説します。`extends`キーワードの使い方と、オーバーライドのルールをコードで確認していきましょう。

  • `extends`キーワードでサブクラスを作成する
  • 抽象メソッドの実装を強制する(オーバーライド)
  • サンプルコードで学ぶ一連の流れ

`extends`キーワードでサブクラスを作成する

抽象クラスを利用するには、`extends`キーワードを使ってそのクラスを継承したサブクラス(具象クラス)を作成します。
「Animal」という抽象クラスを継承して、具体的な「Dog」クラスを作ってみましょう。

// Animalクラスを継承してDogクラスを定義
public class Dog extends Animal {
    // この時点では、抽象メソッドcry()を実装していないためコンパイルエラーになる
}

この段階では、親クラスであるAnimalの抽象メソッド`cry()`を実装していないため、エディタやコンパイラがエラーを通知します。

抽象メソッドの実装を強制する(オーバーライド)

エラーを解消するには、サブクラス内で親クラスの抽象メソッドを実装(オーバーライド)する必要があります。
オーバーライドするメソッドには`@Override`アノテーションを付けるのが一般的です。

public class Dog extends Animal {

    // 抽象メソッドcry()を具体的に実装する
    @Override
    public void cry() {
        System.out.println(name + ":ワン!");
    }
}

`@Override`を付けることで、親クラスのメソッドを正しくオーバーライドしているかをコンパイラがチェックしてくれるため、記述ミスを防げます。

サンプルコードで学ぶ一連の流れ

それでは、全体のコードと実行結果を見てみましょう。抽象クラス`Animal`を継承した`Dog`クラスと`Cat`クラスを作成し、それぞれのインスタンスを動かします。

抽象クラス:Animal.java

public abstract class Animal {
    public String name;

    public Animal(String name) {
        this.name = name;
    }

    // 抽象メソッド(鳴き声)
    public abstract void cry();

    // 具象メソッド(食事)
    public void eat(String food) {
        System.out.println(name + "は" + food + "を食べます。");
    }
}

具象クラス:Dog.java

public class Dog extends Animal {
    public Dog(String name) {
        super(name); // 親クラスのコンストラクタを呼び出す
    }

    @Override
    public void cry() {
        System.out.println(name + ":ワン!");
    }
}

具象クラス:Cat.java

public class Cat extends Animal {
    public Cat(String name) {
        super(name);
    }

    @Override
    public void cry() {
        System.out.println(name + ":ニャー!");
    }
}

実行クラス:Main.java

public class Main {
    public static void main(String[] args) {
        Animal dog = new Dog("ポチ");
        dog.cry(); // Dogクラスで実装したcry()が呼ばれる
        dog.eat("ドッグフード"); // Animalクラスの共通メソッド

        Animal cat = new Cat("タマ");
        cat.cry(); // Catクラスで実装したcry()が呼ばれる
        cat.eat("キャットフード"); // Animalクラスの共通メソッド
    }
}

実行結果

ポチ:ワン!
ポチはドッグフードを食べます。
タマ:ニャー!
タマはキャットフードを食べます。

`dog.cry()`と`cat.cry()`でそれぞれ異なる結果になっているのが分かります。共通の処理は`Animal`クラスに任せつつ、個別の処理は各サブクラスに実装を委ねる、という抽象クラスの役割がよく分かる例です。

ここが一番の悩みどころ!interfaceとの違いを徹底比較

Javaの抽象クラスを学ぶ上で避けて通れないのが、interfaceとの違いです。機能面での違いだけでなく、どのような思想で使い分けるべきかまで踏み込んで解説します。ここを理解すれば、設計の幅が大きく広がります。

  • 機能面での違いを表でチェック
  • 思想・目的の違い:「is-a」と「can-do」
  • Java 8以降のinterfaceの変化と注意点

機能面での違いを表でチェック

まず、抽象クラスとinterfaceの機能的な違いを比較表で整理しましょう。

項目 抽象クラス (abstract class) interface
継承/実装の数 単一継承のみ (1つしか継承できない) 多重実装が可能 (複数実装できる)
フィールド 通常のフィールドを持てる 定数 (public static final) のみ持てる
メソッド 抽象メソッドと具象メソッドの両方 抽象メソッド (Java 8以降はdefaultメソッド、staticメソッドも可)
コンストラクタ 持てる (サブクラスから利用) 持てない
アクセス修飾子 public, protected, private など自由に指定可能 メソッドは public, フィールドは public static final のみ

機能面では、フィールドやコンストラクタを持てたり、アクセス修飾子を柔軟に設定できたりする抽象クラスの方が、通常のクラスに近い性質を持つと言えます。

思想・目的の違い:「is-a」と「can-do」

機能の違いよりも重要なのが、設計上の思想の違いです。これを理解することが、適切な使い分けに繋がります。

Javaの抽象クラスとinterfaceの使い分け

  • 抽象クラス:is-a 関係(AはBの一種である)
    クラス間に親子関係があり、共通の基盤や性質を持つ場合に用います。
    例:「犬(Dog) is a 動物(Animal)」

  • interface:can-do 関係(AはBができる)
    クラスの種別に関係なく、共通の機能(能力)を持たせたい場合に用います。
    例:「人(Human) can do 泳ぐ(Swimmable)」「鳥(Bird) can do 飛ぶ(Flyable)」

継承は「モノ」の本質的な関係性を示し、実装は「コト(機能)」の追加を示す、と考えると分かりやすいでしょう。


例えば、「車」と「飛行機」はどちらも「乗り物」という点では共通しているので、抽象クラス`Vehicle`を継承できます。

一方で、「飛行機」と「鳥」は全く別のモノですが、「飛ぶ」という共通の機能を持っています。この場合に、`Flyable`(飛べる)というinterfaceをそれぞれに実装する、といった使い分けができます。

Java 8以降のinterfaceの変化と注意点

もともとinterfaceは抽象メソッドしか持てませんでしたが、Java 8から`default`メソッドと`static`メソッドという、処理を記述できるメソッドが追加されました。

  • defaultメソッド
    • 実装クラスでオーバーライドしなくてもよい、デフォルトの処理を持つメソッド。
  • staticメソッド
    • interfaceに紐づく静的なメソッド。

この変更により、interfaceでも共通処理をある程度持てるようになり、抽象クラスとの境界が少し曖昧になりました。

しかし、interfaceは状態(インスタンス変数)を持つことができないという根本的な違いは残っています。共通のフィールドをサブクラスに持たせたい場合は、依然としてJavaの抽象クラスを選択するのが適切です。

Java抽象クラスを使いこなすための7つの活用パターン

抽象クラスの理論を学んだところで、次は実際の開発でどのように役立つのかを見ていきましょう。ここでは、実用的な7つの活用パターンを、より具体的なシナリオを交えて紹介します。これらを参考にすれば、より柔軟で保守性の高いコードを書けるようになります。

  • 共通の処理やフィールドをサブクラスで使い回したい時
  • 特定の処理の実装をサブクラスに強制したい時
  • 処理の大枠だけ決めて詳細はサブクラスに任せたい時(テンプレートメソッド)
  • is-a関係(親子関係)を明確に示したい時
  • 将来的な機能追加を見越した柔軟な設計にしたい時
  • フレームワークの基底クラスとして利用する時
  • 状態を持つ共通の振る舞いを定義したい時

共通の処理やフィールドをサブクラスで使い回したい時

これは最も基本的で分かりやすい使い方です。

例えば、`GeneralUser`(一般ユーザー)クラスと`AdminUser`(管理者ユーザー)クラスを作るとします。どちらのユーザーも`userId`や`userName`といった共通のフィールドと、`displayProfile()`のような共通のメソッドを持つはずです。

これらの共通部分を抽象クラス`AbstractUser`にまとめておくことで、各サブクラスではそれぞれのクラスに特化した機能(管理者向けの権限設定など)の実装に集中できます。

もし将来、全ユーザーに`lastLoginDate`というフィールドを追加したくなった場合も、抽象クラスを修正するだけで対応が完了します。

特定の処理の実装をサブクラスに強制したい時

抽象メソッドの主な役割です。

例えば、オンラインショップで`CreditCardPayment`(カード決済)や`BankTransferPayment`(銀行振込決済)といった複数の決済方法を扱うケースを考えます。これらをまとめる抽象クラス`Payment`を作り、その中に`executePayment()`という抽象メソッドを定義します。

こうすることで、新しい決済方法(例:`QrCodePayment`)を追加する開発者は、必ず`executePayment()`の具体的な処理を実装しなければならなくなります。

決済実行のロジックを実装し忘れるという致命的なミスを未然に防ぎ、システムの安全性を高めることができます。

処理の大枠だけ決めて詳細はサブクラスに任せたい時(テンプレートメソッド)

これはデザインパターンの一つで、処理の骨格(テンプレート)を親の抽象クラスで定義し、具体的な処理内容は子のサブクラスで実装させる手法です。

例えば、データ処理のバッチプログラムを考えます。処理の流れが「1. データソースに接続」「2. データを読み込み・加工」「3. 結果を書き出し」「4. 接続を閉じる」で固定されているとします。

この一連の流れを抽象クラスの`final`メソッドとして定義し、「2. データを読み込み・加工」の部分だけを抽象メソッドにしておきます。

CSVファイルを処理するサブクラスや、データベースを処理するサブクラスは、データ加工のロジックだけを考えればよくなり、定型的な接続処理などを毎回書く必要がなくなります。

【テンプレートメソッドパターンのイメージ】

  抽象クラス: AbstractBatch
  +-------------------------------------+
  | // 処理の骨格を定義 (変更させない)   |
  | final void execute() {             |
  |   connect();      // ①接続 (共通)   |
  |   processData();  // ②加工 (個別)   |
  |   write();        // ③書き出し(共通)|
  |   close();        // ④切断 (共通)   |
  | }                                   |
  |                                     |
  | // ②の具体的な処理をサブクラスに任せる |
  | abstract void processData();        |
  +-------------------------------------+
          ^
          | extends
  +----------------------+
  | サブクラス: CsvBatch |
  | (CSVの加工処理を実装)|
  +----------------------+

is-a関係(親子関係)を明確に示したい時

「interfaceとの違い」でも触れましたが、クラス間に明確な「is-a」(AはBの一種である)関係がある場合は、抽象クラスを使うのが自然です。

例えば、「正社員(RegularEmployee)」と「契約社員(ContractEmployee)」は、どちらも「従業員(Employee)」の一種です。この関係は、単に「給与計算ができる(Calculatable)」といった機能(interface)を追加するのとは異なり、もっと本質的な分類です。

`RegularEmployee extends Employee`と記述することで、オブジェクト間の関係性がコード上で明確に表現され、プログラム全体の構造が直感的で分かりやすくなります。

将来的な機能追加を見越した柔軟な設計にしたい時

開発の途中で、全てのサブクラスに共通の機能を追加したくなることはよくあります。
例えば、前述の`Employee`の例で、当初はなかった「全従業員の勤怠時間を記録する」という要件が後から追加されたとします。

この時、抽象クラス`Employee`に`recordWorkTime()`という具象メソッドを一つ追加するだけで、`RegularEmployee`も`ContractEmployee`も、その他の全従業員クラスが即座にその機能を使えるようになります。

もしinterfaceで設計していた場合、各クラスに個別にメソッドを実装するか、defaultメソッドを使うことになりますが、関連するフィールド(例:`totalWorkHours`)を追加したい場合は、やはり抽象クラスが有利です。

フレームワークの基底クラスとして利用する時

多くのJavaフレームワークでは、開発者が共通のルールに従ってクラスを作成できるよう、基盤となる抽象クラスが提供されています。

例えば、WebアプリケーションフレームワークであるJavaServer Faces (JSF)では、`Converter`という機能を作る際に、`javax.faces.convert.Converter`というinterfaceを実装します。しかし、より便利な基底クラスとして`javax.faces.convert.DateTimeConverter`のような抽象クラスが用意されていることもあります。

開発者はこれらの抽象クラスを継承し、特定の部分(例えば、日時のフォーマット方法)だけをオーバーライドすることで、面倒な定型処理をフレームワークに任せ、本来のビジネスロジックに集中できるのです。

状態を持つ共通の振る舞いを定義したい時

サブクラス間で共通のフィールド(状態)を持ち、そのフィールドを使った共通の処理を行いたい場合は、抽象クラスが最適です。
例えば、RPGのキャラクターを考えてみましょう。`Hero`(勇者)も`Wizard`(魔法使い)も、共通して`hp`(ヒットポイント)というフィールド(状態)を持ちます。そして、「ダメージを受ける」という振る舞いは、どちらのキャラクターも「`hp`からダメージ値を引く」という共通の処理になります。

この時、抽象クラス`GameCharacter`に`hp`フィールドと、`hp`を操作する`takeDamage()`メソッドを具象メソッドとして定義します。このように「状態」と、その「状態を操作する振る舞い」をセットでサブクラスに継承させられるのは、フィールドを持てないinterfaceにはできない、抽象クラスならではの強力な機能です。

初心者がハマりがちなJava抽象クラスの注意点

Javaの抽象クラスは便利ですが、いくつか守るべきルールがあります。ここでは、特に初心者が間違いやすいポイントを3つに絞って解説します。これらの注意点を押さえて、不要なコンパイルエラーを避けましょう。

  • 注意点1:抽象クラスはインスタンス化できない
  • 注意点2:`final`修飾子とは一緒に使えない
  • 注意点3:`private`な抽象メソッドは作れない

注意点1:抽象クラスはインスタンス化できない

最も基本的なルールです。抽象クラスは未完成な設計図なので、直接`new`キーワードを使ってインスタンスを生成することはできません。

// Animalは抽象クラスなので、以下のコードはコンパイルエラーになる
Animal animal = new Animal("ななし"); // NG!

抽象クラスは、必ずそれを継承した具象クラスのインスタンスとして利用します。

注意点2:`final`修飾子とは一緒に使えない

`final`修飾子には「これ以上変更できない」という意味があります。

クラスに`final`を付けると「継承できない」という意味になり、メソッドに`final`を付けると「オーバーライドできない」という意味になります。

一方で、抽象クラスや抽象メソッドは「継承・オーバーライドされること」を前提としています。両者の目的は正反対であるため、`abstract`と`final`を同時に一つのクラスやメソッドに付けることはできません。

注意点3:`private`な抽象メソッドは作れない

`private`修飾子を付けたメソッドは、そのクラスの内部からしかアクセスできません。

抽象メソッドはサブクラスでオーバーライドされるために存在しますが、`private`にしてしまうとサブクラスから見えなくなってしまい、オーバーライドできなくなります。

目的が矛盾するため、抽象メソッドに`private`を付けることは文法的に許可されていません。サブクラスに実装を強制するメソッドは、通常`public`または`protected`にします。

まとめ:Java抽象クラスを理解して設計力をアップしよう!

この記事では、Javaの抽象クラスについて、基本的な概念からinterfaceとの違い、実用的な活用法まで幅広く解説しました。最後に、重要なポイントを振り返っておきましょう。

  • 抽象クラスのポイントおさらい
  • 次のステップ:実際にコードを書いてみよう!

抽象クラスのポイントおさらい

  • 抽象クラスは`abstract`を付けて定義し、インスタンス化はできない。
  • 具象メソッド(共通処理)と抽象メソッド(実装の強制)を両方持てる。
  • 利用するには`extends`キーワードで継承し、抽象メソッドをオーバーライドする必要がある。
  • interfaceとの使い分けは「is-a」(モノの関係)なら抽象クラス、「can-do」(コトの機能)ならinterfaceが基本。
  • 状態(フィールド)を持つ共通の振る舞いを定義したい場合に特に有効。

このブログを検索

  • ()

自己紹介

自分の写真
リモートワークでエンジニア兼Webディレクターとして活動しています。プログラミングやAIなど、日々の業務や学びの中で得た知識や気づきをわかりやすく発信し、これからITスキルを身につけたい人にも役立つ情報をお届けします。 note → https://note.com/yurufuri X → https://x.com/mnao111

QooQ