Javaの文法

Javaの文法(Javaのぶんぽう)の記事では、プログラミング言語Java構文(シンタックス、: syntax)について解説する。また、それ以外についても解説している。

キーワード一覧

Javaの基本言語仕様は54種類程度に抑えたキーワードによって比較的コンパクトにまとめられている。Javaの構文はC++言語によく似たものであり、それよりも比較的平易化されている。従って以下のキーワードを眺めるだけでもJavaプログラミングの大まかなスタイルを掴むことができる。

類別 キーワード 説明
アクセス修飾子 import パッケージ名前空間の解決用
private 同クラス内でアクセス可
package 同クラス内と同パッケージ内でアクセス可、パッケージスコープの宣言
protected 同クラス内と同パッケージ内と派生クラス内でアクセス可
public 全範囲でアクセス可
クラス定義系 abstract 抽象クラス、抽象メソッド
class クラス
enum 列挙型
extends スーパークラスの指定
final クラスの継承不可、メソッドのオーバーライド不可、フィールドの定数化
implements 実装するインターフェース
interface インターフェース
native そのメソッドは他言語コンパイルコードで実装される
static 静的フィールド、静的メソッド、静的内部クラスはトップレベル扱いされる
strictfp 指定クラス、指定メソッドで厳密な浮動小数点計算が行われる
synchronized 同期用クラス、同期用メソッド、同期用ブロック
transient 直列化から除外されるフィールド
void 返り値無しのメソッド
volatile そのフィールドは各スレッドからキャッシュ参照されない
制御構文系 break ループ脱出
case パターンマッチ項目
continue ループ起点回帰
default パターンマッチ外
do 無条件ループ起点
else 条件式の偽側フロー
for 初期値定義と周回毎値変化で修飾された条件式ループ
if 条件式の真側フロー
instanceof インスタンスの実行時型チェック
return メソッドの評価終了
switch パターンマッチ起点
while 条件式ループ
例外処理系 assert 条件式が偽ならプログラム中断
catch 例外パターン捕捉
finally 例外デフォルト捕捉
throw 例外発生
throws そのメソッドで発生する例外候補
try 例外発生ブロック
基本値系 boolean 真偽値
byte 8ビット値
char 文字表現用16ビット値
double 64ビット浮動小数点
float 32ビット浮動小数点
false 偽値
int 符号付き32ビット値
long 符号付き64ビット値
short 符号付き16ビット値
true 真値
null ヌルポインタ
参照値系 super スーパークラスのインスタンス
this カレントクラスのインスタンス
new インスタンスの生成
使用できない goto
const

データ型

Javaのデータ型には大別して、プリミティブ型と参照型がある。

プリミティブ型

primitive data type[1]もしくはprimitive type[2]とも。 プリミティブ型はオブジェクトではなく、スーパークラスを持たない。

整数型
byte 8ビット符号付き
short 16ビット符号付き
int 32ビット符号付き
long 64ビット符号付き
浮動小数点数
float 32ビット符号付き
double 64ビット符号付き
float f = 2147483647f;
double d = 9223372036854775807;

float型は数字を符号付き32Bitの最大値(2,147,483,647)まで代入できるが[3]、double型は符号付き64Bitの最大値(9,223,372,036,854,775,807)まで代入できる。[4]

また、float型は代入する数字の末尾に「F」または「f」をつけなければdouble型とみなされ、コンパイルエラーになってしまう。[5]

  • 浮動小数点数は決して例外をスローしない
  • 0でない値を0(ゼロ)で割った値はInf(無限大)と等値である
  • 無限大でない値をInfで割った値は0(ゼロ)と等値である。
文字
char 16ビット符号無しUnicode
  • J2SE v 1.4.2までのcharは基本多言語面 (BMP) の範囲内のコードポイントを符号無し16ビットで表現する。
  • J2SE 5.0からは、補助文字をサポートするため、charは符号無し16ビットで表現可能でありBMPの範囲内に限ればコードポイントと同値となるUTF-16符号化形式のコード単位を表現するように変更され、21ビットが必要となるコードポイントの表現にはintを使用するように変更された(JSR#204)[6]。つまりchar型は互換性の問題からあくまで符号無し16ビットのままとされており、UTF-16符号化形式を採用したことから補助文字を扱う場合はコードポイント一つにコード単位を格納したchar値のペアが対応する[7]。これらを適切に取り扱う便宜として、StringクラスやプリミティブラッパークラスであるCharacterクラスなどの各種メソッドが利用できる。
ブーリアン型
boolean true または false
  • C/C++や類似の言語と異なり、Javaではfalseの代わりに0(ゼロ)またはnullと書くことはできない
  • 同様に、0でない値を書いてtrueの代わりとすることはできない
  • ブール型をブール型でない基本型へキャストすることとその逆はできない

参照型

すべての参照型はオブジェクト型を表すクラスObjectから派生する。オブジェクトは参照型のインスタンスである。クラスclassは既定でObjectから派生する参照型である。列挙型enumは抽象クラスEnumから暗黙的に派生する参照型である。

リフレクション機能のための「クラスを表現するクラス(メタクラス)」としてClassが存在するが、クラス自体はオブジェクトではない。

文字列

  • Stringオブジェクトは不変(変更不能)である
  • Stringオブジェクトは生成時に初期化されなければならない
  • コンパイラは文字列リテラル(ダブルクォーテーションで囲まれた文字列)を見つけると、Stringオブジェクトを生成する
  • 演算子 ++= は文字列を連結するためにオーバーロードされる
String str1 = "alpha"; 
String str2 = new String("alpha");
  • StringBufferStringBuilderオブジェクトは可変(変更可能)なので、オブジェクト生成オーバヘッド無しで柔軟に文字列を生成・変更できる。StringBufferとStringBuilderの違いは、StringBufferがマルチスレッドに対応している(=スレッド・セーフである)のに対し、StringBuilderは対応していないことである。
  • StringとStringBufferは互いに独立であり、一方から派生したものではない
StringBuffer str1 = new StringBuffer("alpha");
str1.append("-meta");
str1.setCharAt(str1.indexOf("m"), 'b');
System.out.println(str1);     // str1.toString() を呼び出す。
                              // 印字結果は"alpha-beta"となる

プリミティブラッパークラス

プリミティブラッパークラス

機能
  • ある型の値を他の型へ変換する静的メソッドを提供する
  • 基本型を参照型としてラッピングする目的(ボックス化)で使用できる(コレクションに格納するなど)
その他
  • ブール型以外の基本型の初期値は0。ブール型の初期値はfalse。ラッパークラス(およびObjectクラスに属する全クラス)の初期値はnull。

配列

  • 配列自体はオブジェクト型Objectのサブクラスつまり参照型である。
  • プリミティブ型の配列、オブジェクト型の配列、さらにそれらの配列の配列(ジャグ配列)、などの配列を扱える。
  • プリミティブ型の配列では、全ての要素はその型の値でなければならない。
  • オブジェクトの配列では、階層関係にある必要がある(例えば、インタフェースの配列では、あるオブジェクトがその要素になるには、そのインタフェースを実装したクラスのインスタンスでなければならない)。
  • 配列オブジェクトは配列要素の数を表す読み取り専用の属性「length」を持つ。
  • すべての配列は動的に領域確保される。配列を生成する際の要素数として、定数だけでなく変数を使用することができる。
  • 「配列の配列」ではない、(C#にあるような「本物の」)多次元配列(矩形配列)は無い。
// 配列を宣言 - 配列名は「myArray」、要素の型は "SomeClass" への参照
SomeClass[] myArray = null;

// 配列を生成
myArray = new SomeClass[10];
// または宣言と生成を同時に行う
SomeClass[] myArray = new SomeClass[10];
// 配列の要素を割り当てる (ただし、基本型の配列なら必要ない)
for (int i = 0; i < myArray.length; i++)
    myArray[i] = new SomeClass();

リテラル

リテラルの例を示す。

データ型
byteshortintlong
  • 0365(8進法表記、0で始まる)
  • 0xF5(16進法表記、0xで始まる)
  • 245(10進法表記)
float
  • 23.5F
  • 23.5f
  • 1.72E3F
  • 1.72E3f
  • 1.72e3F
  • 1.72e3f
double
  • 23.5
  • 23.5D
  • 23.5d
  • 1.72E3
  • 1.72E3D
char
String
  • "Hello, world"

国際化サポート

Javaは1バイト整数の型(byte)と文字の型(char)とを区別する。なお、charは、名前に反して必ずしも文字を表現していない場合があり得る。すなわち、実際にはUCS-2(UTF-16)による、いわゆるダブルバイトである。J2SE 5.0以降ではサロゲートAPIによって、サロゲートペアに対応した。

JavaプログラムのソースコードおよびソースファイルはUTF-8でエンコードするのが基本である。Unicodeに対応したエンコードであれば、Javaプログラムのソースコードに直接Unicode文字を記述することもできる。なお、-encodingコンパイルオプションを指定することで、対応する任意のエンコードを使用することもできる。ロケール依存のコードページ(日本語版Microsoft Windowsで使われているCP932等)を利用することもできる。

文字列リテラルやコメントの他、クラス名や変数名も国際化に対応している。例えば、次のソースコードはJavaのコードとして正しく解釈されコンパイルされ実行できる。ここではクラス名、変数名、および文字列リテラルとして日本語の文字を使っている。

public class こんにちは世界 {
    private String 文字列 = "こんにちは世界";
}

演算子

算術演算子

算術演算子

二項演算子
表記 意味
+ 加算
- 減算
* 乗算
/ 除算
% 剰余 (整数の余りを返す)
単項演算子
表記 意味
- 単項マイナス (符号反転)
++ インクリメント (変数の前か後につけることができる)
-- デクリメント (変数の前か後につけることができる)
! ブール補数演算
~ ビット単位反転
(型名) キャスト

代入演算子

代入演算子

表記 意味
= 代入
+= 加算と代入
-= 減算と代入
*= 乗算と代入
/= 除算と代入
%= 剰余と代入
&= ビット演算 ANDと代入
|= ビット演算 ORと代入
^= ビット演算 XORと代入
<<= 左シフト(ゼロ埋め)と代入
>>= 右シフト (符号拡張)と代入
>>>= 右シフト (ゼロ埋め) と代入

関係演算子

関係演算子

表記 意味
== 左辺の数値は右辺の数値と等しい。左辺のインスタンスは右辺のインスタンスと同一である。
!= 左辺の数値は右辺の数値と等しくない。左辺のインスタンスは右辺のインスタンスと同一ではない。
> 左辺の数値は右辺の数値より大きい。
>= 左辺の数値は右辺の数値と等しい、または、より大きい。
< 左辺の数値は右辺の数値より小さい。
<= 左辺の数値は右辺の数値と等しい、または、より小さい。
instanceof 左辺のインスタンスのクラスは右辺のクラスまたは右辺のクラスのサブクラスである。

関係演算子(==と!=)を参照型に対して用いた場合、そこで比較されるのは参照先のオブジェクトが同じかどうかであり、オブジェクトの中身の値が一致するか否かではない。オブジェクトの中身を比較したい場合はObject.equals(Object)メソッドを使用する。instanceof演算子は、オブジェクトが指定クラスのインスタンスであるか否かを判定するために用いる。

三項演算子

三項演算子は二つの記号?:を組み合わせて記述する。条件演算子とも呼ぶ。構文は以下である。

条件 ? 式1 : 式2

条件がtrueであるとき、式1の値をとる。そうでない場合は式2の値をとる。

例:

String answer = (p < 0.05) ? "reject": "keep";
// これは以下のコードと等価である:
String answer;
if (p < 0.05) {
  answer = "reject";
} else {
  answer = "keep";
}

論理演算子

論理演算子

  • 短絡評価論理演算 (結果が判定するまでオペランドを左から右へと評価する)
  • 必要最小限の式しか評価しない
  • 部分的な評価 (完全な評価ではない)
表記 意味
&& 論理積(左のオペランドが false のとき、式は false を返し、右のオペランドは評価されない)
|| 論理和(左のオペランドが true のとき、式は true を返し、右のオペランドは評価されない)
! 論理否定

ビット演算子

ビット演算子

二項演算子
文法 意味
& ビット積
| ビット和
^ ビット排他的和
<< 左シフト(ゼロ埋め)
>> 右シフト(符号拡張
>>> 右シフト(ゼロ埋め)
単項演算子
表記 意味
~ ビット反転

文字列演算子

表記 意味
+ 連結
+= 連結と代入

制御構造

if... else

if文

if (expr) {
    statements;
}
else if (expr) {
    statements;
}
else {
    statements;
}
  • exprは真偽値を与えなければならない。

switch文

switch文

switch (expr) {
  case VALUE1:
    statement11 ; … ; statement1L ;
    break ;
  case VALUE2:
    statement21 ; … ; statement2M ;
    break ;
  default:
    statementd1 ; … ; statementdN ;
}
  • expr 値の型はbyte/short/int/charあるいはそれらのプリミティブラッパークラスでなければならない。
    • J2SE 5.0以降では列挙型 (Enumのサブクラス) を、JavaSE 7以降ではString型を使用することもできる。caseラベルの値にnullは使用できない。
  • 各々のcaseラベルの値はユニークなリテラル、あるいはコンパイル時に値が定まる定数式 (constant expression) でなければならず、変数を書くことはできない。

forループ

for文

for (initial-expr; cond-expr; incr-expr) {
    statements;
}

for-each ループ

J2SE 5.0では、for-each文[8]と呼ばれる新機能が追加された。これは集合の中の全要素を順番に参照するような処理を大いに簡素化する。このような場合、従来は次の例に示すような反復子 (iterator) を書かねばならなかった:

// Java 5.0以前にはジェネリクスもないため、実際にはさらに煩雑となる。
public int sumLength(Set<String> stringSet) {
    int sum = 0;
    Iterator<String> itr = stringSet.iterator();
    while (itr.hasNext()) {
        sum += itr.next().length();
    }
    return sum;
}

for-each文はこのメソッドを大いに簡素化する:

public int sumLength(Set<String> stringSet) {
    int sum = 0;
    for (String s : stringSet) {
        sum += s.length();
    }
    return sum;
}

この例の動作としては、stringSetに含まれる全てのStringについて、長さを取得してsumに加算する。

whileループ

while文

while (expr) {
    statements;
}

do ... while

do-while文

do {
    statements;
} while (expr);

分岐命令

文法 意味
break; 最も深いループから直ちに脱出する。
continue; ループの現在の回を中断し、次の回の冒頭に移る。
break LABEL ラベル付き文の実行を中断し、ラベル付き文の直後の文に移る。
continue LABEL ラベル付きの文にジャンプする(ラベル付きの文またはラベル付きループを冒頭から再開する)

int sum = 0;
for (int i = 1; i < 10; i++) {
    if (i == 3) {
        continue;  // このループの残りをスキップしforに戻る。
    }
    sum += i;

    if (sum > 15) {
        break;  // ループを脱出する。
    }
}

ラベル

  • 末尾にコロンがついた識別子からなる。
  • 分岐命令が参照する文またはブロックを識別するために使われる。
  • 仮にラベルを外したとすると、分岐命令は最も内側のループについて機能する。

例:

LABEL1: statement;
LABEL2: { statements; }

goto

Javaのキーワードには「goto」という綴りも含まれているが、goto文は無い(他にもうひとつ、constも同様に、キーワードとして予約されているが、予約されているだけで使われていない)。

オブジェクト

クラス

Javaではクラスあるいはインタフェースの内部で別のクラスを宣言することができる。これは「ネストされたクラス」(nested class) と呼ばれる。ネストされていないクラスは「トップレベルクラス」(top-level class) と呼ばれる。ネストされたクラスがstatic修飾されていない場合、「内部クラス」(inner class) となる。内部クラスは外側のクラスのインスタンスを暗黙的にキャプチャすることで、静的メンバ、非静的メンバいずれにもアクセスすることができる。ネストされたクラスがstatic修飾されている場合、「静的クラス」(static class) となり、静的メンバのみにアクセスできる。メソッドの内部にクラスを定義することもでき、これは「ローカルクラス」(local class) と呼ばれる。ローカルクラスでは、外側のローカル変数には読み取りアクセスのみできる。また、型の名前を持たないローカルクラスとして「匿名クラス」(anonymous class) を定義し、同時にインスタンス生成をすることもできる。

クラスあるいはインタフェースの内部で別のインタフェースを宣言することもできる。これは「ネストされたインタフェース」と呼ばれる。

クラスを宣言する際は以下の修飾子を付けることができる:

  • abstractインスタンス化できない。インタフェースとabstract (抽象) クラスだけがabstract (抽象) メソッドを持つことができる。抽象クラスを継承する具象 (非abstract) サブクラスは、引き継がれた全ての抽象メソッドを、abstractでないメソッドでオーバーライドしなければならない。修飾子finalと併用することはできない。
  • final – サブクラスを作らせない。 finalクラスのすべてのメソッドは無条件にfinalになる。 修飾子abstractと併用することはできない。
  • strictfp – このクラスとそこに含まれる全てのネストクラスにおいて、全ての浮動小数点演算は厳密な浮動小数点演算動作を使用する。厳密な浮動小数点演算動作は、演算結果がプラットフォームに関わりなく同一となることを保証する。

Javaのクラス定義ブロックはセミコロン;で終わらせる必要はない点に注意。この点はC++の文法と異なる。

継承

// 子クラスは親クラスを継承する
class ChildClass extends ParentClass { ... }
  • 任意のクラスのデフォルトの親クラスはObjectクラスである。
  • クラスは単一の親クラスだけを継承できる。実装多重継承はできない。

スコープ

  • this – 現在のサブクラス (デフォルト) への参照 (例:this.someMethod())。
  • super – 親クラスへの参照 (例:super.someMethod())。サブクラスがオーバライドした親クラスのメソッドや、サブクラスが継承しつつも隠蔽した親クラスのフィールドにアクセスするために使うことができる。

インタフェース

インタフェースとは、実装の詳細がいっさいない抽象型である。その目的は複数のクラスを多態性によって統一的に扱うことである。共通のインタフェースを実装する複数のクラスは、そのインタフェース型のコンテキストにしたがって互いに交換可能とすることができる(リスコフの置換原則)。インタフェースはまた、抽象化 – クラスの実装方法を隠蔽すること – という考え方を強制するのにも役立つ。

インタフェースは抽象メソッドと定数フィールド (static finalフィールド) だけを含むことができる。インタフェースメソッドはデフォルトでpublicかつabstractであり(実装を持たない)、インタフェースフィールドはデフォルトで public static final である。インスタンスフィールドやクラスフィールドすなわち状態を持つことができない点が抽象クラスと異なる。

Javaは完全な直交多重継承はサポートしておらず、インタフェースによる「型の多重継承」のみをサポートする。C++における多重継承(実装の多重継承)には、複数の親クラスや型から複数回継承したフィールドやメソッドを識別するための複雑なルールが伴う。インタフェースを実装から分離することにより、インタフェースはより単純かつ明快に多重継承が持つ利点の多くを提供する。もっとも、実装の多重継承を避ける代価としてコードは若干冗長になる。というのは、インタフェースはクラスのシグネチャを定義するのみで実装を持てないため、インタフェースを継承する全てのクラスは定義されたメソッドをいちいち実装しなければならないからである。純粋な多重継承であれば実装自体も継承されるのでこのようなことはない。

なお、Java 8ではインタフェースのデフォルトメソッドにより実装の多重継承を限定的にサポートするようになった。また、インタフェースが静的メソッドを持つこともできるようになった。

Javaのインタフェースは Objective-C規約のコンセプトによく似た振る舞いをする。

インタフェースの実装

クラスは、一つのクラスを継承できるのに加えて、implementsキーワードを用いて一つ以上のインタフェースを実装することができる。

interface MyInterface {
    void foo();
}

interface Interface2 {
    void bar();
}

class MyClass implements MyInterface {
    void foo() {...}
    ...
}

class ChildClass extends ParentClass implements MyInterface, Interface2 {
    void foo() {...}
    void bar();
    ...
}

以下の例では、Deleteableインタフェースを実装する非abstractクラスは、引数無しで戻り型がvoidであるdeleteという名前の非抽象メソッドを定義しなければならない。そのメソッドの実装と機能は各々のクラスによって決定される。

public interface Deleteable {
    void delete();
}

このコンセプトにはさまざまな使い道がある。例えば:

public class Fred implements Deleteable {
    // このメソッドはDeleteableインタフェースを満足する
    public void delete() {
        // ここにコードを実装
    }
    public void someOtherMethod() {
    }
}

public void deleteAll(Deleteable[] list) {
    for (int i = 0; i < list.length; i++) {
        list[i].delete();
    }
}

上の配列に含まれる全てのオブジェクトはdelete()メソッドを持つことが保証されるので、deleteAll()メソッドはFredオブジェクトと他の如何なるDeleteableオブジェクトをも区別する必要がない。

インタフェースの継承

インタフェースはextendsキーワードを用いて一つ以上のインタフェースを継承することができる。

interface ChildInterface extends ParentInterface, AnotherInterface {
    ...
}

結果として生じるインタフェースを実装するクラスは、元のインタフェースに含まれたメソッドをも併せて定義しなければならない。

public interface MyInterface {
    foo();
}

public interface Interface2 extends MyInterface {
    bar();
}

public class MyClass implements Interface2 {
    void foo() {...}
    void bar() {...}
    ...
}

アクセス修飾子

アクセス修飾子は、そのクラスやクラスメンバにアクセス可能なクラスを決定する。

トップレベルクラスアクセス

デフォルトでは、Javaのクラスは、それら自身のJavaパッケージからのみアクセスできる。これは、クラスのパッケージが、裏に隠れて機能を実行するようなAPIを提供することを可能とする。外にアクセスを公開されたクラスの動作を、隠されたクラスが支える形になる。

  • デフォルト (修飾子を省略した場合) – 定義されたパッケージ内からのみアクセス可能。
  • public – 定義されたパッケージの外のクラスからもアクセス可能。

クラスメンバアクセス

クラスメンバとはフィールド、メソッド、コンストラクタ、クラス内で定義されたネストされたクラスのことである。アクセス制限が厳しいものから並べると、クラスメンバのアクセス修飾子は次の通り。

  1. private – そのクラスからのみアクセス可能 (内部クラスからのアクセスを含む)。private宣言されたメンバはサブクラスによって引き継ぐことができない。
  2. package-private (修飾子を省略した場合) – 同じパッケージ内の他クラスからもアクセス可能。
  3. protected – 上に加え、パッケージ外の継承クラスからアクセス可能。
  4. public – 任意のクラスからアクセス可能。

メソッドをオーバライドする際、そのメソッドのアクセス権を「より厳しく」することはできない。さもなくば親クラスのインタフェース契約を壊してしまうからである。したがってオーバライドされる場合、publicメソッドはpublicとして宣言されねばならず、protectedメソッドをデフォルトアクセス権(修飾子省略)とすることはできない。しかしながら、メソッドをオーバライドしてアクセス権を「より緩める」ことは許される。したがってオーバライドする際、デフォルト (パッケージ) アクセス権のメソッドはprotectedまたはpublicとして宣言することができ、protectedメソッドはpublicとして宣言することができる。

フィールド

アクセス修飾子に加えて、データフィールドは以下の修飾子によって宣言される:

  • final – このフィールドの中身は変更できない。ただ一度のみ値を設定(初期化)できる。イニシャライザ (初期化子) がないfinalフィールドを「ブランクfinal」フィールドと呼ぶ。staticなブランクfinalフィールドは最終的にスタティックイニシャライザによって初期化されなければならない。staticでないブランクfinalフィールドはコンストラクタの実行中に必ず初期化されなければならない。volatileにはなれない。
  • static – クラスのインスタンスではなくクラスに属する。
  • transient – オブジェクトの中でも永続的 (persistent) にできないフィールドである。このフィールドの中身を待避または復元してはならないことをコンパイラに知らせる。
  • volatile – そのフィールドが他スレッドによって非同期にアクセスされる可能性があることをコンパイラに知らせる。finalにはなれない。

定数

staticfinal両方を宣言されたフィールドは事実上、定数である。staticはそのフィールドがそのクラスにおいてただ一つのみ存在することを示し、finalはそのフィールドがただ一度のみ値を設定(初期化)可能であることを意味する。

初期化子

「イニシャライザ」(初期化子)はフィールドのイニシャライザと同時に実行されるコードのブロックである。

静的初期化子

「スタティックイニシャライザ」(静的初期化子)はstaticフィールドのイニシャライザと同時に実行されるコードのブロックである。静的フィールド初期化子と静的初期化子は宣言された順番に実行される。静的初期化はクラスがロードされた後で実行される。

static int count = 20;
static int[] squares;
static {  // スタティックイニシャライザ
    squares = new int[count];
    for (int i = 0; i < count; i++)
        squares[i] = i * i;
}
static int x = squares[5];  // x には値25が代入される。

インスタンス初期化子

「インスタンスイニシャライザ」(インスタンス初期化子)はインスタンスの (非staticな) フィールドの初期化子と同時に実行されるコードのブロックである。インスタンスフィールド初期化子とインスタンス初期化子は宣言された順番に実行される。

インスタンス初期化子とインスタンスフィールド初期化子はコンストラクタが呼び出された際に実行される。正確な実行順序としては、親クラスのコンストラクタが実行された後、かつ、自身のコンストラクタが実行される前、となる。

メソッド

アクセス修飾子に加えて、メソッドには以下の修飾子を付けて宣言できる:

  • abstract – 当該クラスでは定義されないメソッドであり、代わりに当該クラスの全ての具象 (非abstract) サブクラスによって定義されなければならない。staticfinalnativeのいずれとも併用できない。
  • final – サブクラスによって再定義できないメソッド。そのメソッドがインスタンス (非static) メソッドでありかつ十分小さいならば、コンパイラはそのメソッドをインライン関数のように各所に展開する場合がある(訳注:性能改善目的と思われる)。abstractと併用はできない。
  • native – このメソッドはネイティブなマシン依存コードにリンクする。メソッド本体無しで宣言される。abstractと併用はできない。Java Native Interfaceにて使用される。
  • static – クラスのインタンスではなく、クラスに属する。abstractと併用はできない。
  • strictfp – メソッドとそこに含まれる内部クラス全てにおける浮動小数点演算が、厳密な浮動小数点演算動作を使用する。厳密な浮動小数点演算動作は、演算結果がプラットフォームに関わりなく同一となることを保証する。
  • synchronized – メソッド本体を実行する前に、関連オブジェクトを排他する。関連オブジェクトが他スレッドにて排他済である場合、他スレッドが排他を解除し自スレッドが排他を獲得するまで実行を待たされる。ここで言う関連オブジェクトとは、そのメソッドがstaticならばClassオブジェクトを指し、非staticならばオブジェクトインスタンスを指す。abstractメソッドをsynchronizedとして宣言することは可能だが意味はない。何故なら排他とは宣言ではなく実装に伴う機能であり、抽象メソッドは実装を持たないからである。

privateメソッドは自ずからfinalであり、abstractにはできない点に注意。

可変引数

Java SE 5.0において、引数の個数が可変であるようなメソッドについての文法上の便宜 (varargs) [1]が追加された。これによって引数の個数が可変であるメソッドをタイプセーフに使用することが容易になる。最後のパラメタの後に「...」と書くと、Javaは全ての引数を配列に格納する:

public void drawPolygon(Point... points) { /* ... */ }

このメソッドを呼ぶ際、プログラマは個々のpointsを単にカンマで区切って書けばよく、Pointオブジェクトの配列をわざわざ用意する必要はない。このメソッドの内部でpointsを参照する際はpoints[0]、points[1]、などのように書ける。pointsが渡されていない場合、配列のlengthは0となる。可変個の引数と別に固定的に必要なパラメタがある場合は、それらのパラメタは可変引数に先立って指定すればよい。

// ポリゴンは少なくとも3つの点を必要とする。
public void drawPolygon(Point p1, Point p2, Point p3, Point... otherPoints) { /* ... */ }

コンストラクタ

コンストラクタはオブジェクトが割り当てられた後すぐに呼び出され、オブジェクトの初期処理を行う。コンストラクタは典型的にはnewキーワードを使用して呼び出されるが、リフレクションを使用して呼ぶこともできる。リフレクション機能はjava.lang.reflectパッケージより提供される。

コンストラクタを宣言する際に使える修飾子はアクセス修飾子のみである。

  • 可能ならば、オブジェクトはひとたびコンストラクタを呼ばれた以後は直ちに有効かつ有意味なオブジェクトとなるべきである。分割された複数の初期化用メソッドを使わなければ初期処理が完了しないというような設計は好ましくない。
  • 慣習として、引数としてそのオブジェクト自身の型を受け取ってデータメンバを複写するようなコンストラクタを「コピーコンストラクタ」と呼ぶ。
  • コンストラクタが明示的に定義されていない場合、コンパイラは暗黙のうちに内容が空で引数を取らないデフォルトのコンストラクタを生成する。
  • コンストラクタはオーバーロードできる。
  • コンストラクタ内の最初の文は親クラスのコンストラクタ:super(...);または同じクラス内の別のコンストラクタ:this(...);を呼び出せる。
  • もし、super(...) または this(...)に対する明示的な呼び出しがないならば、コンストラクタ本体が実行される前に、親クラスのデフォルトコンストラクタsuper();が呼ばれる。

Objectクラスのメソッド

Objectクラスのメソッドは継承されるので、全てのクラスにて使用できる。

cloneメソッド

Object.clone()メソッドは現在のオブジェクトのコピーである新しいオブジェクトを返す。クラスは、それがクローンできることを明示するためにマーカーインタフェースCloneableを実装しなければならない。

equalsメソッド

Object.equals(Object)メソッドはそのオブジェクトともう一つのオブジェクトを比較し、二つのオブジェクトが同一かどうかをboolean型の値で返す。意味的には、このメソッドはオブジェクトの内容を比較するのに対し、関係演算子"=="はオブジェクトの参照を比較する。equalsメソッドはjava.utilパッケージにあるデータ構造クラスの多くで使われる。これらのデータ構造クラスのいくつかはObject.hashCodeメソッドにも依存している - equalshashCodeとの間の契約の詳細について、hashCodeメソッドを参照のこと。

finalizeメソッド

Object.finalize()メソッドはガベージコレクタがオブジェクトのメモリを解放する前に必ず一度だけ呼び出される。オブジェクトが消滅する前に実行しなければならない何らかの後処理がある場合、各クラスはfinalizeをオーバーライドすることができる。とはいえほとんどのオブジェクトはfinalizeをわざわざオーバーライドする必要はない。

finalizeメソッドがいつ呼ばれるかは保証されない。複数のオブジェクトのfinalizeがどのような順番で呼ばれるかも不定である。もしJVMがガベージコレクションを実行せずに終了するならば、OSがオブジェクトを解放する可能性があり、その場合finalizeメソッドは呼ばれない。

finalizeメソッドは、他のクラスから呼ばれるのを防ぐために、常にprotectedとして宣言されるべきである。

protected void finalize() throws Throwable { ... }

getClassメソッド

Object.getClass()メソッドはオブジェクトをインスタンス化するために使われたクラスのClassオブジェクトを返す。このクラスオブジェクトはJavaにおけるリフレクションの基本となる。その他のリフレクション機能はjava.lang.reflectパッケージにて提供される。

hashCodeメソッド

Object.hashCode()メソッドは連想配列にオブジェクトを保存するための「ハッシュ値」として (int型の) 整数を返す。java.util.Mapインタフェースを実装するクラスは連想配列を提供しhashCodeメソッドに依存する。hashCodeの良い実装は安定 (不変) かつ均等に分布するハッシュ値を返す (異なるオブジェクトのハッシュ値は互いに異なる値となる傾向を持ち、かつハッシュ値は整数値の範囲内で均等に分布する)。

連想配列はequalshashCodeの両メソッドに依存するため、これら二つのメソッドの間では、オブジェクトがMapに挿入される場合に関する或る重要な契約[注 1]が維持されねばならない:

二つのオブジェクト a と b に関して
  • a.equals(b) == b.equals(a)でなければならない。
  • もしa.equals(b)trueならば、a.hashCode() == b.hashCode()でなければならない。

この契約を維持するために、equalsメソッドをオーバーライドしたクラスは同時にhashCodeメソッドもオーバーライドし、逆もまた同様として、hashCodeequalsが常に同じ性質(または同じ性質の一部)に基づくようにしなければならない。

マップがオブジェクトとの間に有する更なる契約は、ひとたびオブジェクトがマップに挿入されたなら、hashCodeequals両メソッドの結果は以後変わらないということである。したがって、一般にハッシュ関数はオブジェクトの不変(変更不能)な属性に基くように設計するのが良い。

toStringメソッド

Object.toString()メソッドはオブジェクトの文字列表現をStringで返すものである。toStringメソッドは、オブジェクトが文字列連結演算子(++=)のオペランドとして使われたとき、コンパイラによって暗黙のうちに呼び出される。

waitとnotifyスレッドシグナルメソッド

全てのオブジェクトは、そのオブジェクトに関連するスレッドについての二つの待ちリストを持つ。一つの待ちリストはsynchronizedキーワードに伴いオブジェクトをミューテックス排他するために使われる。もしミューテックスが他スレッドによって排他されているならば、自スレッドは排他を待っているスレッドのリストに追加される。もう一つの待ちリストはスレッド間でシグナルを送るためのもので、これはwaitnotifynotifyAllの各メソッドを通して使用される。

wait/notifyを用いるとスレッド間での能率的な連携が可能となる。あるスレッドが別スレッドでの処理が終わるのを待ち合わせる必要があるとき、または何らかのイベントが発生するまで待たねばならないとき、スレッドはその実行を一時停止してイベントが発生した際に通知を受け取ることができる。これはポーリングとは対照的である。ポーリングにおいては、スレッドは一定時間スリープしてはフラグや他の状態表示をチェックする処理を繰り返す。ポーリングはスレッドがチェックを繰り返さねばならないという点でより計算コストが掛かる上に、実際にチェックしてみるまでイベント発生を検知できないという意味で鈍感でもある。

waitメソッド

waitメソッドには三つのオーバーロード版があり、タイムアウト値の指定方法がそれぞれ異なる:wait()wait(long timeout)wait(long timeout, int nanos)の三つである。一つ目のメソッドはタイムアウト値が0であり、これはタイムアウトが発生しないことを意味する。二つ目のメソッドはミリ秒単位のタイムアウト値を取る。三つ目のメソッドはナノ秒単位のタイムアウト値を取り、これは1000000 * timeout + nanosとして計算される。

waitを呼んだスレッドは待機状態となり、そのオブジェクトの待ちリストに追加される。そのスレッドは以下の三つのイベントのいずれか一つが起きるまで、オブジェクトの待ちリスト上に留まる:

  1. 別のスレッドがそのオブジェクトのnotifyまたはnotifyAllメソッドを呼ぶ (詳細はnotifyメソッド参照)
  2. 別のスレッドがそのスレッドのinterrupt()メソッドを呼ぶ
  3. waitにて指定した0でないタイムアウト値が満了する

waitメソッドは、そのオブジェクトについての同期(synchronized)ブロックまたは同期(synchronized)メソッドの内部からのみ呼ばねばならない。これはwaitnotifyとの間で競合を起こさないためである。スレッドが待ちリストに入るとき、そのスレッドはそのオブジェクトのミューテックス排他を解除する[注 2]。そのスレッドが待ちリストから削除され実行可能スレッドとなった際に、そのスレッドは走行を再開するのに先立ってそのオブジェクトのミューテックスを改めて排他しなければならない。

notifynotifyAllメソッド

Object.notify()Object.notifyAll() メソッドはオブジェクトの待ちリストから一つ以上のスレッドを削除し、それらを実行可能スレッドとする。notifyは待ちリストから1スレッドのみ削除し、notifyAllは待ちリストから全てのスレッドを削除する。notifyがどのスレッドをリストから削除するかは規定されておらず、JVMの実装に依存する。

notifynotifyAllメソッドは、そのオブジェクトについての同期(synchronized)ブロックまたは同期(synchronized)メソッドの内部からのみ呼ばねばならない。これはwaitnotifyとの間で競合を起こさないためである。

アノテーション

Javaのアノテーションクラスインタフェースメソッドフィールドパッケージなどに対してメタデータとして付加情報を記入する機能で、Java SE 5 で追加された。

アノテーションはjava.lang.annotation.Annotationインタフェースを実装することで自作することもできる。

Javaのアノテーションは三つに分けることができる。

  • マーカー・アノテーション – データが無く名前だけを持つアノテーション。
  • 単一値アノテーション – データを一つだけ持つアノテーション。見かけはメソッド呼び出しに似ている。
  • フル・アノテーション – 複数のデータを持つアノテーション。

Java標準APIの主なアノテーション

JUnit4 から利用可能になったアノテーション

@Test
そのメソッドがテストメソッドであることを示す。このメソッドにテストを記述する。従来のJUnitでメソッド名がtestで始まるメソッドと同じ。
@Before
このアノテーションが付加されたメソッドは、@Test アノテーションが付いたメソッドを実行するたびに事前に実行されることを意味する。
@After
このアノテーションが付加されたメソッドは、@Test アノテーションが付いたメソッドを実行するたびに、必ず後から実行されることを意味する。
@BeforeClass
このアノテーションが付加されたメソッドは、そのテストクラスを呼び出す前に実行される。
@AfterClass
このアノテーションが付加されたメソッドは、そのテストクラスを呼び出した後に実行される。

アノテーションを使用する

マーカーアノテーション

クラスやメソッドにマーカーアノテーションを付加するには以下のようクラスやメソッドの接頭辞に最低一つ以上のスペースまたは改行コードを入れて修飾子のように記述する。 この例は、クラスに非推奨、メソッドに、スーパークラスからのメソッドをオーバーライドしていることを意味するマーカーアノテーションを付加している例である。

@Deprecated class DeprecatedClass {
    @Override public boolean equals(Object obj) {
        return this == obj;
    }
}

単一値アノテーション

単一値アノテーションを付加するには以下のようにする。 この例は、Serializableインタフェースを実装したクラスのフィールドにstatic finalなserialVersionUIDが宣言されていないという警告を無視するアノテーションを付加していることを意味する。

@SuppressWarnings(value = {"serial"})
class NoSerialVersionIDClass implements java.io.Serializable { }

これは、以下のような書き方もできる。これは単一値アノテーション@SuppressWarningsがvalue()メソッドを一つしか持たないことがわかっているためvalue = を省略できることを意味する。

@SuppressWarnings("serial")
class NoSerialVersionIDClass implements java.io.Serializable { }

このアノテーションは、戻り値の型がString[]になっているため同じvalue値であっても以下のように複数指定することができる。以下のように指定することで、シリアルバージョンIDが設定されていない警告と、コレクション総称型による型チェックを行われていないことによって生ずる警告を無視することができる。 "unchecked"はメソッドに対してのみ設定することもできる。

@SuppressWarnings("serial", "unchecked")
class NoSerialVersionIDClass implements java.io.Serializable {
    public void setupList() {
        List list = new ArrayList();
        list.add("abcdef");
    }
}

このアノテーションは正確に記述すると以下のようにString[]配列の初期化宣言のようになる。

@SuppressWarnings(value = {"serial", "unchecked"})

フルアノテーション

フルアノテーションは、複数のデータ型を持つアノテーションである。ここでは自作したアノテーション @MyAnnotation があるとき、以下のように、変数名 = 値の形をカンマで区切って記述する。各値の記法は、各アノテーションで定義されているメソッドの戻り値の型で決まる。たとえばこの場合valueという変数名はStringを戻り型にとる value() というメソッドと、int を戻り型にとる version() というメソッドを持つ。フルアノテーションの場合は、default によりデフォルト値が設定されているアノテーション以外は、value =version = を省略することはできない。

@MyAnnotation(value = "abc", version = 2)
class AnnotatedClass { }

アノテーションを定義する

アノテーションを定義するには、interfaceキーワードの接頭辞に@をつけて定義する。

マーカーアノテーション

マーカーアノテーションは以下のように定義する。メソッドやフィールドが一切ないマーカーインタフェースのアノテーション版ともいえる。@Override@Deprecatedがこれらのアノテーションに相当する。

public @interface MarkerAnnotation { }

単一値アノテーション

単一値アノテーションは以下のように定義する。このアノテーションには少なくともメソッドがひとつだけ定義されている。単一値アノテーションのメソッド名にはvalueという名前をつけるのが儀礼である。

@interface Single {
    String value();
}

フルアノテーション

フルアノテーションは以下のように定義する。以下のように二つ以上のメソッドを定義する。

@interface FullAnnotation {
    String value();
    int id();
}

メタアノテーション

メタアノテーションとは、定義しているアノテーションのみにつけられるアノテーションのことである。メタアノテーションの例としては@Target@Retention@Documented@Inheritedがあり、これらはクラスやメソッドなどには使うことができず、アノテーションのみに使うことができる。アノテーションを定義するために使われるアノテーションということから、メタアノテーションと呼ばれる。

メタアノテーションを使用する

メタアノテーションを使ってアノテーションを定義するには、以下のように記述する。

@Retention(RetentionPolicy.SOURCE)
@Target({ElementType.ANNOTATION_TYPE, ElementType.METHOD})
public @interface NewAnnotation {
}

このとき、@Retentionは、新たにアノテーション NewAnnotation を作るとき、このアノテーション情報はソースコードのみにしか保存されないことを意味する。@Targetはこのアノテーションをどの型に使うことができるかを指定している。この場合、ANNOTATION_TYPEMETHOD を指定しているのでこのアノテーションはアノテーション型とメソッドにしか使うことができない。つまり、この NewAnnotation もまた、メソッドだけでなくアノテーションにも保存できるため、メタアノテーションとしても使えることを示している。

@Retention

メタアノテーション@Retentionには以下のRetentionPolicy列挙型を設定することができる。

RetentionPolicy 説明
RetentionPolicy.CLASS アノテーション情報はコンパイル時に保存されるが実行時にはJava仮想マシンによって保持されない。
RetentionPolicy.RUNTIME アノテーション情報はコンパイル時に保存され、実行時にもJava仮想マシンによって保持される。
RetentionPolicy.SOURCE アノテーション情報はコンパイル時に破棄される。ソースコード内のみで有効。

@Target

メタアノテーション@Targetには以下のElementType列挙型を設定することができる。これは配列を使って

 @Target({ElementType.METHOD, ElementType.FIELD, ElementType.TYPE})

と複数指定することができる。ただし、同じ値を {} 内で複数使用するとエラーとなる。これによって型を指定することで、そのアノテーションが、どの型に対して使うことができるのかを指定できる。

ElementType 説明
ElementType.ANNOTATION_TYPE アノテーション型に指定できることを示す。
ElementType.CONSTRUCTOR コンストラクタに指定できることを示す。
ElementType.LOCAL_VARIABLE ローカル変数に指定できることを示す。
ElementType.FIELD フィールドに指定できることを示す。
ElementType.METHOD メソッドに指定できることを示す。
ElementType.PACKAGE パッケージに指定できることを示す。
ElementType.PARAMETER メソッド引数に指定できることを示す。
ElementType.TYPE クラス、またはインタフェース(アノテーション型を含む)、列挙型に指定できることを示す。

入出力

関連項目: Java Platform, Standard Edition#java.ioNew I/O

J2SE1.4よりも前のバージョンのJavaはストリーム・ベースブロッキングI/Oのみをサポートしていた。これは1ストリームにつき1スレッドを必要とした。何故ならストリームの入力または出力を行おうとすると、それが完了するまでそのスレッドは完全に待ちに入ってしまい、他の処理がいっさい行えなくなったからである。これは、Javaを用いたネットワークサービスを構築する上で、スケーラビリティと性能双方の面で大きな問題となっていた。J2SE1.4以降では非ブロッキングI/OフレームワークとしてNIO (New I/O) が導入され、このスケーラビリティ問題は修正された (ただし、サンによるNIO APIの実装にはまだ多くの問題点がある)。

非ブロッキングIOフレームワークは、以前のブロッキングIOフレームワークより遥かに複雑ではあるが、一つのスレッドで任意の数の"チャネル"を扱うことができる。このフレームワークはReactorパターンをベースとしている。

実行コード

アプリケーション

public class MyClass {
    public static void main(String[] args) {...}
    ...
}

アプレット

  • ウェブブラウザで明示された表示領域上で実行されるJavaコード。
  • initdestroyは一度だけ呼ばれるが、startstopはユーザがウェブページを訪問するたびに何回も呼ばれる。
// MyApplet.java

import java.applet.*;

public class MyApplet extends Applet {
    init() {...}     // ブラウザが最初にアプレットを読み込むときに呼ばれる。
    destroy() {...}  // ユーザがブラウザを終了するときに呼ばれる。
    start(){...}     // アプレットを実行し始めるときに呼ばれる。
    stop() {...}     // ユーザがウェブページを去るとき、再読込するとき、
                     // ブラウザを終了するときに呼ばれる。
}
<applet code="MyApplet" width="200" height="200">
</applet>

appletタグの埋め込み

  • HTMLのappletタグをアプレットのソースコードに埋め込むことができる。
  • appletタグを書くと、そのアプレットは.htmlファイル無しでも、簡易アプレットビューアによって直接実行可能となる。
  • 典型的には、appletタグはimport文の直後に書かれる。
  • appletタグは/* */コメントによって囲まれていなければならない。
// MyApplet.java
...
/*
<applet code="MyApplet" width="200" height="200">
</applet> 
*/
...

サーブレット

JSP (JavaServer Pages)

  • ウェブページにJavaコードを埋め込む形態。
  • JSPタグはウェブサーバで処理され、出力結果 (一般的にHTMLやXML) はクライアントに送信される。
  • JSPコードは実行される前にJava Servletにコンパイルされる。
  • JSPはJava Servletの拡張である。
  • JSPタグの使用法はPHPASPのタグの使用方法と類似する。

JSPタグ

文法 意味
<% Java構文 %> スクリプトレット
<%= 単一Java構文の出力 %> 構文
<%! Java宣言文 %> 宣言
<%@ [page, include, taglib] JSPディレクティブ %> ディレクティブ

その他

ケースセンシティビティ (大文字小文字区別)

Javaはケースセンシティブ (大文字小文字を区別する) である。

コメント

コメントの文法はC++と同じである。

// 一行コメント

/* 複数行
   コメント */

なお、ドキュメンテーションコメントとしてJavadoc方式をサポートする。

/**
 * この行はクラス、インタフェース、メソッド、フィールド宣言の直前に記述する。
 * このコメントはクラスのドキュメントを自動生成する
 * ユーティリティで使用することができる。
 */

関連項目

参考文献

脚注

注釈

  1. ^ ここで言う「契約」の意味については契約プログラミングを参照。
  2. ^ オブジェクトを排他していないスレッドがwait、notify、またはnotifyAllを呼ぶと、IllegalMonitorStateExceptionが生成され、current thread not ownerと表示される

出典

  1. ^ Primitive Data Types (The Java™ Tutorials > Learning the Java Language > Language Basics)
  2. ^ Chapter 4. Types, Values, and Variables
  3. ^ 符号付き整数型とは - IT用語辞典”. IT用語辞典 e-Words. 2023年5月17日閲覧。
  4. ^ 64ビット整数の最大値は計算の限界ではない”. Qiita (2020年8月4日). 2023年5月17日閲覧。
  5. ^ 「F」”. 2023年5月17日閲覧。
  6. ^ Java プラットフォームにおける補助文字のサポート
  7. ^ Sun Microsystems, Inc.. “Character (Java 2 Platform SE 5.0)”. Sun Microsystems, Inc.. 2009年6月11日閲覧。
  8. ^ The For-Each Loop

外部リンク

オラクル