キーワード (Java)
この項目ではJavaにおけるキーワード(予約語の記事も参照)[注釈 1]に関して説明する。全部で50個ある。
この項目は、プログラムの細かい説明には立ち入らず、他の言語と比較できるような説明を目的としている。
特徴
[編集]- プリミティブ型名や分岐処理 (if, switch)、反復処理 (for, while) などの基本的な構文とキーワードはC言語およびC++と共通するものが多い。クラスや例外処理に関してはC++類似の構文とキーワードが多い。
プリミティブ型
[編集]型名
[編集]プリミティブ型の名前である。C言語およびC++と共通するものが多い。ただし内部表現は処理系依存ではなく、言語仕様によって厳密に定められている。
- byte, short, int, long
- float, double
- boolean
- char
- void
- これはプリミティブ型ではないが、型と同じ文脈で使うキーワードなのでここに分類する。メソッドに戻り値がないことを宣言する。void (コンピュータ)も参照のこと。
値
[編集]以下はキーワードと同様に扱われるが、定義では論理値リテラル、またはnullリテラルというリテラルである。
制御構造
[編集]C言語およびC++と共通のキーワードを使用している。
- 分岐
- if, else
- switch, case, default
- 繰り返し
- for, while, do
- ジャンプ
クラス、パッケージ
[編集]- package
- import
- 自クラスの所属するパッケージとは異なるパッケージにあるクラスを使用するときにこの宣言でインポートする。
- class
- クラスの定義。例えば、以下のように宣言する。それぞれについてはこの後、説明する。
- 例
-
public class クラス名 { ... }
public class クラス名 extends 親クラス名 { ... }
public class クラス名 implements インタフェース名1, インタフェース名2, ... { ... }
public class クラス名 extends 親クラス名 implements インタフェース名1, ... { ... }
- interface
- extends
- implements
- class の宣言で、インタフェースを実装するときに使用する。カンマ区切りで並べた複数のインタフェースを実装することができる。
- this
- super
- new
- コンストラクタの呼び出し。インスタンスを生成するときに使用する。
- 例
-
SampleClass s = new SampleClass();
int[] numbers = new int[5]; // int型の配列を生成
List<String> list = new ArrayList<>(); // ジェネリッククラスのインスタンス化
- instanceof
- 指定されたクラスのオブジェクトかどうか、あるいは指定されたインタフェースを実装しているオブジェクトかどうかを判定する。
a instanceof ClassName
とした場合、a
がClassName
のオブジェクトか、その派生クラスのオブジェクトであれば true になる。ClassName
の箇所にオブジェクトの変数名を記述することはできない。
- 指定されたクラスのオブジェクトかどうか、あるいは指定されたインタフェースを実装しているオブジェクトかどうかを判定する。
- record
- record キーワードは、レコード型と呼ばれる新しい種類の型を定義するために使用される。レコード型は、従来のクラスと似ているが、いくつかの重要な違いがある。
- レコード型は、コンストラクタ、フィールド、アクセサメソッドを自動的に生成する。
- レコード型は、イミュータブル(変更不可)である。
- レコード型は、簡潔で読みやすいコードを作成するために使用できる。
- 例
record Person(String name, int age) { /* コンストラクタ、フィールド、アクセサメソッドは自動的に生成されます */ }
- record キーワードは、レコード型と呼ばれる新しい種類の型を定義するために使用される。レコード型は、従来のクラスと似ているが、いくつかの重要な違いがある。
- sealed
- sealed キーワードはクラスやインターフェースに対して使用され、指定されたサブクラスのみがクラスやインターフェースを拡張または実装できるようにする。これにより、クラス階層の管理が厳密に行えるようになる。
- 例
public sealed class Shape permits Circle, Square { // クラスの内容 }
- permits
- sealed クラスの許可されたサブクラスを指定するために使用される。
- 例
public sealed class Shape permits Circle, Square { // クラスの内容 }
- non-sealed
- sealed クラスのサブクラスで、このサブクラスがさらに他のクラスによって拡張可能であることを示す。
- non-sealed は単一の識別子ではなく、トークンシーケンスでコンテキスト・キーワードと呼ばれる。
- 例
public non-sealed class Circle extends Shape { // クラスの内容 }
修飾子
[編集]クラス、メンバの宣言に付けられる修飾子である。
- public, protected, private
- クラス、メンバのアクセス制御を指示する修飾子(アクセス修飾子)である。外部からアクセス可能な範囲を決める。Javaでは4つの段階が設けられ、以下の番号が大きくなるほど外部からのアクセスが制約される。
- publicでは、すべてのクラスからアクセスできる。
- protectedでは、同一パッケージにあるクラスかサブクラスからのみアクセスできる。異なるパッケージでサブクラスでないクラスからはアクセスできない。クラスを修飾することはできない (内部クラスは例外)。
- アクセス修飾子がない場合は、同一のパッケージ内のクラスからのみアクセスできる。サブクラスであってもパッケージが異なる場合はアクセスできない。
- privateでは、他のクラスからはアクセスできず、自クラスからのみアクセスできる。クラスに修飾することはできない (内部クラスは例外)。
- Java SE 1.0ではprivate protectedという組み合わせを使用して、自分自身とサブクラスからのアクセスができるが同じパッケージ内の他のクラスからはアクセスできないというアクセス権を使用することができたが、これはJava SE 1.1以降から廃止され利用することができなくなった。
- クラス、メンバのアクセス制御を指示する修飾子(アクセス修飾子)である。外部からアクセス可能な範囲を決める。Javaでは4つの段階が設けられ、以下の番号が大きくなるほど外部からのアクセスが制約される。
- final
例1:
public class SampleClass {
final int constantValue = 5;
}
例2:
public class SampleClass {
final int constantValue; // 初期化なし
// コンストラクタ
public SampleClass() {
constantValue = 5; // 1回だけ代入できる。
}
}
- static
- フィールドに付けられた場合は、クラスに属する変数(クラス変数、静的フィールド、クラスフィールド)が確保されることを表す。この修飾子のついたフィールドは、インスタンスがいくつ生成されてもフィールドの実体は1つで、すべてのインスタンスの間で共有される。
記述例:
class SampleClass {
static String shared = "shared";
static final String STR = "constant";
}
例:
class Sample1 {
// static なメソッド
public static void method() { ... }
}
class Sample2 {
void proc() {
...
Sample1.method(); // インスタンスを生成せず直接実行できる。
...
}
}
- クラス内にstatic {...}というブロックが現れた場合は、そのクラスが最初に参照されたときにそのブロック内のコードを実行する。これを静的初期化子と呼び、主にstatic finalな配列や
Collection
オブジェクトの初期化などに利用される。データベースにアクセスするためにJDBCドライバを呼び出すClass.forName("ドライバ名")
もこの静的初期化子を呼び出している。
- クラス内にstatic {...}というブロックが現れた場合は、そのクラスが最初に参照されたときにそのブロック内のコードを実行する。これを静的初期化子と呼び、主にstatic finalな配列や
例:
class StaticInitSampleClass {
private static final List<String> list;
static {
final List<String> tempList = new ArrayList<>();
tempList.add("Hello,");
tempList.add("World.");
StaticInitSampleClass.list = Collections.unmodifiableList(tempList);
}
}
内部クラス
- 入れ子にされたクラス(ネストされたクラス: nested class)のうち、
static
修飾されていないものは内部クラス (inner class) と呼ばれ、暗黙的に外側のクラスのインスタンスをキャプチャして参照することができるが、このネストクラスをstatic
修飾すると静的ネストクラス (static nested class) となり、暗黙的に外側のクラスのインスタンスを参照できなくなる[4]。
- 入れ子にされたクラス(ネストされたクラス: nested class)のうち、
例:
class OuterClass {
static int outerStaticField = 0;
int outerField = 0;
class InnerClass {
void setOuterField(int value) {
outerField = value; // 暗黙参照可能。
outerStaticField = value; // OK。
}
}
static class StaticClass {
void setOuterField(int value) {
//outerField = value; // コンパイルエラー。
outerStaticField = value; // OK。
}
}
public OuterClass() {
new InnerClass().setOuterField(100);
new StaticClass().setOuterField(-100);
}
public static void test() {
final OuterClass outer = new OuterClass();
System.out.println("OuterField = " + outer.outerField);
System.out.println("OuterStaticField = " + outerStaticField);
}
}
- abstract
クラスまたはメソッドに付けられる修飾子である。finalと併用することはできない。Javaでは、実装されないメソッドがあるクラスを作成することができる。抽象クラス自身のインスタンスを作成することはできない。似たものとしてインタフェースがあるが、abstract class は以下の点においてインタフェースと異なる。
- コンストラクタを持つことができる
- メソッドなどの実装を記述できる
- インスタンス変数を持つことができる
abstract class はそのままではインスタンス化できず、不足しているメソッドを実装した派生クラスを用意して使用する。
記述例:
public abstract class Sample {
// 実装があるメソッド
public void method1() {
...
}
// 実装がないメソッド
public abstract void method2();
}
- native
- メソッドがJava言語以外で実装されていることを宣言する(Java Native Interfaceを参照)。
- synchronized
記述例1:
public synchronized void method() { ... }
記述例2:
public void method() {
...
synchronized(this) { // 自クラスのインスタンスに対して排他制御
// 排他制御されるブロック
}
...
}
記述例3:
public void method(Object obj) {
...
synchronized(obj) { // Object obj に対して排他制御
// 排他制御されるブロック
}
...
}
- volatile
- フィールドの宣言に付けられ、キャッシュを見に行かずに常に最新の値を見に行くようになる。マルチスレッド下における軽量かつ限定的な同期処理に使われる[5]。
例:
public class SampleClass {
volatile int syncSample;
...
}
- transient
例外処理
[編集]例外処理はC言語にはない。Javaでは、C++およびC#とほぼ共通の構文およびキーワードが使用されている。ただしfinallyはC++には存在せず、throwsはC#には存在しない。
- try, catch, finally
- 例外が発生しうる箇所で使用する。tryブロック (
try{
と}
との間) で例外が発生しうるコードを囲み、catchブロック (catch(
Throwable
の派生クラス例外オブジェクト){
と}
との間) に例外発生後の処理を書く。finallyブロック (finally{
と}
との間) には、ファイルのクローズ、データベースセッションの切断、ログアウトなど、例外発生の有無にかかわらずtry/catchブロック内の処理を終了する前に必ず実行しておきたいコードを書く。catchを記述した場合finallyは省略可能であり、finallyを記述した場合catchは省略可能である。try/catchブロックとfinallyブロックの両方にreturn文がある場合、後者が優先される。catchブロックにはif-else-if文のように連続して捕捉したい例外を記述することができる。
- 例外が発生しうる箇所で使用する。tryブロック (
- throw
- 例外を発生させる。
- throws
- メソッドの宣言で使用される。そのメソッドがどのような例外をスローするかを列挙する。Javaにおける例外の種類には大きく分けて throws の明示が必要な例外と必要でない例外がある。前者は主にファイルエラーなどプログラムの動作中にその都度対応すべき例外で、後者は主に開発・デバッグの段階で対処すべき例外あるいは深刻なエラーである。
以下はIOException
を使った記述例である。
public void method() throws Exception { // メソッドが例外をスローすることを宣言
try {
...
if (エラー条件)
throw new IOException(); // 例外を発生させる。
// (1) 正常終了
// このtryブロックにはreturn文を記述することもできる。
// ただし、finallyブロックにすでにreturn文があるときは、ここにreturn文を書いてもfinallyブロックのreturnが優先される。
}
catch (IOException e) { // IOException は Java の例外クラスの1つ
// (2) 例外処理
// このcatchブロックにはreturn文を記述することもできる。
// ただし、finallyブロックにすでにreturn文があるときは、ここにreturn文を書いてもfinallyブロックのreturnが優先される。
}
catch (Exception e) { // IOException よりも上位(親)の例外クラス
// (3) 例外をさらに外のブロックにスローする。
throw e;
}
finally {
// (1), (2), (3) どの場合でもこのブロックは必ず実行される。
// ここにreturn文があるときはtry/catchブロックにreturn文があろうともこちらのreturn文が優先される。
...
}
}
追加されたキーワード
[編集]J2SE 1.4でassertが追加され、J2SE 5.0でenumが追加された。Java 9では制限されたキーワード (restricted keyword) として、open
, module
, requires
, transitive
, exports
, opens
, to
, uses
, provides
, with
が追加された[6]。また、Java 9では単一のアンダースコア_
が予約済みのキーワードとなり、識別子として使用できなくなった。
Java 10ではvar
による型推論が追加されたが、これはキーワードではなく、予約された型名 (reserved type name) によって実現されている[7]。
Java 13で追加されたyield
キーワードは、switch式で使用され、結果を返すために使用される。
Java 16で追加されたrecord
キーワードは、データキャリアとして使用される不変のデータオブジェクトを簡単に定義するために導入された。
Java 17で追加されたsealed
キーワードはクラスやインターフェースに対して使用され、指定されたサブクラスのみがクラスやインターフェースを拡張または実装できるようにする。permits
キーワードはsealed
クラスの許可されたサブクラスを指定するために使用される。non-sealed
キーワードはsealed
クラスのサブクラスで、このサブクラスがさらに他のクラスによって拡張可能であることを示す。
その他
[編集]浮動小数点演算
[編集]その他のキーワードとして、浮動小数点演算の移植性を高めるstrictfpがある。浮動小数点の演算では一般に全く同じ結果は期待できないものではあるが、x87などハードウェアによっては、途中の値においてより高い精度や、より広い値の範囲で演算が行われるために、最終結果が他のプラットフォームと一致しない場合があり、特にIEEE 754を前提として予想される結果から外れることが問題である。
strictfpを指定すると、途中結果についても全てIEEE 754の倍精度を強制し、相異なるプラットフォーム同士の間でも演算結果が一致することを保証するとされている(仕様では FP-strict という用語を使っている[8])。Java VMの実装で、これを正しく実現するには、x86系CPUを使った計算機の場合、x87を避けてSSE2で計算するか、それができない場合は特殊なテクニック[9]が必要である。
なお、同様に移植性のある結果を保証するStrictMath
というクラスがあるが、そちらはFDLIBMというSun製のライブラリと同じ結果を保証するとされている。
未使用
[編集]const、gotoがキーワードとして予約されているが、Javaでは使用できない。使用するとコンパイル時に構文エラーになる。
脚注
[編集]注釈
[編集]出典
[編集]- ^ “Chapter 3. Lexical Structure - §3.9. Keywords”. Java SE 7 Specifications > Java Language Specification. Oracle. 2024年3月5日閲覧。
- ^ Primitive Data Types (The Java™ Tutorials > Learning the Java Language > Language Basics)
- ^ Branching Statements (The Java™ Tutorials > Learning the Java Language > Language Basics)
- ^ Nested Classes (The Java™ Tutorials > Learning the Java Language > Classes and Objects)
- ^ Javaの理論と実践: volatile を扱う, Internet Archive
- ^ The Java Language Specification, Java SE 9 Edition
- ^ Java 10リリース | InfoQ
- ^ “Chapter 15. Expressions - §15.4. FP-strict Expressions”. Java SE 7 Specifications > Java Language Specification. Oracle. 2024年3月5日閲覧。
- ^ オライリー『Binary Hacks』Hack #98 を参照。
参考文献
[編集]- “The Java™ Language Specification Java SE 7 Edition”. 2013年2月4日閲覧。
- “Java言語規定 第2版”. 2013年2月4日閲覧。