シモセラ エドガー
public class CountTen {
public static void main(String[] args) {
CountTen ct = new CountTen();
ct.count();
for (int i = 0; i < 1000; i++) {
System.out.println("main:i = " + i);
}
}
public void count() {
for (int i = 0; i < 1000; i++) {
System.out.println("count:i = " + i);
}
} }
count:i = 0
count:i = 1
・・・
count:i = 999
main:i = 0
main:i = 1
・・・
main:i = 999
java.lang.Thread
クラスの拡張java.lang.Runnable
インタフェースの実装run()
メソッドに定義main
のようなもの。new
で作成start()
の中で実行主体が具体的に作られて run()
が自動的に実行public class CountTenA extends Thread {
public static void main(String[] args) {
CountTenA ct = new CountTenA();
ct.start(); // 新しいスレッド開始
for (int i = 0; i < 1000; i++) {
System.out.println("main:i = " + i);
}
}
public void run() {
for (int i = 0; i < 1000; i++) {
System.out.println("run:i = " + i);
}
}
}
main:i = 0
main:i = 1
・・・
main:i = 99
run:i = 0
run:i = 1
・・・
run:i = 88
main:i = 100
・・・
main:i = 999
・・・
run:i = 999
Runnable
のrun()
を実装し、処理内容を定義start()
メソッドを呼ぶpublic class CountTenB implements Runnable {
public static void main(String[] args) {
CountTenB ct = new CountTenB();
Thread th = new Thread(ct);
th.start();
for (int i = 0; i < 10; i++) {
System.out.println("main:i = " + i);
}
}
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println("run:i = " + i);
}
}
}
main:i = 0
main:i = 1
・・・
main:i = 99
run:i = 0
run:i = 1
・・・
run:i = 88
main:i = 100
・・・
main:i = 999
・・・
run:i = 999
(A) Thread拡張 | (B) Runnable実装 | |
---|---|---|
用途 | 簡単にスレッド作成。 | 手間がかかる。既存のクラス階層を生かせる。 |
スーパークラス | Thread | 任意 |
クラスの宣言 | class X extends Thread |
class Y implements Runnable |
スレッドの起動 | start() を呼ぶ。X x = new X(); x.start(); |
クラスを「包む」スレッドのstart()を呼ぶ。Y y = new Y(); Thread t = new Thread(y); t.start(); |
実行開始点 | run() |
run() |
Thread#run()
メソッド中に終了条件を組み込むべき。Thread#stop()
メソッド: 強制停止。非推奨。Thread.sleep
(x)java.lang.InterruptedException
スローclass Foo extends Thread {
boolean running = true;
public void stopRunning() {
running = false;
}
public void run() {
while (running) {
...
}
} }
public class Periodic {
public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
int tm = i * 1000;
System.out.println("Start sleep:tm = " + tm);
try {
Thread.sleep(tm);
} catch (InterruptedException e) {
}
}
} }
Periodic
との違いは?public class Periodic2 extends Thread {
public void run() {
for (int i = 0; i < 10; i++) {
int tm = i * 1000;
System.out.println("Start sleep:tm = " + tm);
try {
Thread.sleep(tm);
} catch (InterruptedException e) {
}
}
}
public static void main(String[] args) {
new Periodic2().start();
// for (int i = 0; i < 100000; i++) {
// System.out.println(i);
// }
}
}
Thread#join()
java.lang.InterruptedException
スローjoin(long m)
: mミリ秒でタイムアウトjoin(long m, int n)
: mミリ秒+nナノ秒でタイムアウトpublic class JoinTest extends Thread {
public static void main(String[] args){
JoinTest th = new JoinTest();
System.out.println("main:開始");
th.start();
try {
System.out.println("main:待ち中");
th.join();
} catch (InterruptedException e) {
System.out.println(e);
}
System.out.println("main:終了");
}
public void run() {
System.out.println("run:開始");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
System.out.println(e);
}
System.out.println("run:終了");
}
}
addMoney()
: フィールドvalueに引数の値を加算。矛盾の検知。Thread.currentThread()
: 実行中のスレッドインスタンスの取得System.exit(-1)
: Java仮想マシンの終了。引数-1で異常終了の明示。public class BadBank {
private int value = 0;
public void addMoney(int money) {
int currentValue = value;
System.out.println(Thread.currentThread() + "が addMoney に入った");
value += money;
if (currentValue + money != value) {
System.out.println(Thread.currentThread() + "で矛盾");
System.exit(-1);
}
System.out.println(Thread.currentThread() + "が addMoney から出た");
}
}
public class BadBankTest extends Thread {
BadBank bank;
public BadBankTest(BadBank bank) {
this.bank = bank;
}
public void run() {
while (true) {
bank.addMoney(100);
bank.addMoney(-100);
}
}
public static void main(String[] args) {
BadBank bank = new BadBank();
new BadBankTest(bank).start();
new BadBankTest(bank).start();
} }
public class BadBankTest extends Thread {
BadBank bank;
public BadBankTest(BadBank bank) {
this.bank = bank;
}
public void run() {
while (true) {
bank.addMoney(100);
bank.addMoney(-100);
}
}
public static void main(String[] args) {
BadBank bank = new BadBank();
new BadBankTest(bank).start();
new BadBankTest(bank).start();
} }
Thread[Thread-0,5,main]が addMoney に入った
Thread[Thread-0,5,main]が addMoney から出た
・・・
Thread[Thread-0,5,main]が addMoney に入った
Thread[Thread-1,5,main]が addMoney に入った
Thread[Thread-1,5,main]が addMoney から出た
Thread[Thread-1,5,main]が addMoney に入った
Thread[Thread-1,5,main]が addMoney から出た
Thread[Thread-1,5,main]が addMoney に入った
Thread[Thread-1,5,main]が addMoney から出た
Thread[Thread-1,5,main]が addMoney に入った
Thread[Thread-1,5,main]で矛盾
Thread[Thread-0,5,main]で矛盾
synchronized
と同期synchronized
メソッド(同期メソッド)synchronized
なメソッドpublic class BadBank {
private int value = 0;
public void addMoney(int money) {
int currentValue = value;
System.out.println(Thread.currentThread() + "が addMoney に入った");
value += money;
if (currentValue + money != value) {
System.out.println(Thread.currentThread() + "で矛盾");
System.exit(-1);
}
System.out.println(Thread.currentThread() + "が addMoney から出た");
}
}
synchronized
ブロック、クラスメソッドsynchronized
ブロックsynchronized
public class BadBank {
private int value = 0;
public void addMoney(int money) {
synchronized(this) {
int currentValue = value;
・・・
}
}
}
public class MyClass {
public synchronized void instanceMethod() {
...
}
public synchronized static void classM() {
...
}
}
synchronized
ブロックの利用BadBank
クラスそのものを修正せずに、BadBankTest
を以下のように修正すると?public class BadBankTest extends Thread {
BadBank bank;
public BadBankTest(BadBank bank) {
this.bank = bank;
}
public void run() {
while (true) {
synchronized(bank) {
bank.addMoney(100);
bank.addMoney(-100);
}
}
}
・・・
Thread#join()
java.lang.InterruptedException
スローjoin(long m)
: mミリ秒でタイムアウトjoin(long m, int n)
: mミリ秒+nナノ秒でタイムアウトpublic class JoinTest extends Thread {
public static void main(String[] args){
JoinTest th = new JoinTest();
System.out.println("main:開始");
th.start();
try {
System.out.println("main:待ち中");
th.join();
} catch (InterruptedException e) {
System.out.println(e);
}
System.out.println("main:終了");
}
public void run() {
System.out.println("run:開始");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
System.out.println(e);
}
System.out.println("run:終了");
}
}
join()
ではスレッドの終了しか待てないObject#wait()
Object#notify()
・notifyAll()
class Stack {
int[] array = new int[3];
int index = 0;
public synchronized void put(int n) throws InterruptedException {
while (index == array.length) {
System.out.println(Thread.currentThread().getName()+" 空き待ち");
wait();
}
array[index] = n;
index++;
notifyAll();
}
public synchronized int take() throws InterruptedException {
while(index == 0) {
System.out.println(Thread.currentThread().getName() + " データ待ち");
wait();
}
int n = array[index - 1];
index--;
notifyAll();
return n;
}
}
class Producer extends Thread {
Stack stack = null;
Producer(Stack s) {
stack = s;
}
public void run() {
try {
for (int i = 0; i < 30; i++) {
stack.put(i);
System.out.println(getName() + " " + i + " 追加");
sleepRandomly();
}
} catch (InterruptedException e) {
}
}
void sleepRandomly() {
try {
int n = (int)(Math.random() * 1000);
Thread.sleep(n);
} catch (InterruptedException e) {
}
}
}
class Consumer extends Thread {
Stack stack = null;
Consumer(Stack s) {
stack = s;
}
public void run() {
try {
while (true) {
int x = stack.take();
System.out.println(getName() + " " + x + " を消費");
sleepRandomly();
}
} catch (InterruptedException e) {
}
}
void sleepRandomly() {
try {
int n = (int)(Math.random() * 1000);
Thread.sleep(n);
} catch (InterruptedException e) {
}
}
}
public class ProducerConsumer {
public static void main(String[] args) {
Stack stack = new Stack();
Producer producer = new Producer(stack);
Consumer consumer = new Consumer(stack);
producer.start();
consumer.start();
}
}
Thread-0 0 追加
Thread-1 0 を消費
Thread-0 1 追加
Thread-0 2 追加
Thread-0 3 追加
Thread-1 3 を消費
Thread-1 2 を消費
Thread-0 4 追加
Thread-1 4 を消費
Thread-1 1 を消費
Thread-0 5 追加
Thread-1 5 を消費
Thread-0 6 追加
Thread-0 7 追加
Thread-0 8 追加
Thread-1 8 を消費
Thread-0 9 追加
Thread-1 9 を消費
Thread-1 7 を消費
Thread-0 10 追加
Thread-0 11 追加
Thread-1 11 を消費
Thread-0 12 追加
Thread-1 12 を消費
Thread-0 13 追加
Thread-0 空き待ち
Thread-0 14 追加
Thread-1 13 を消費
Thread-1 14 を消費
Thread-1 10 を消費
Thread-1 6 を消費
Thread-0 15 追加
Thread-1 15 を消費
Thread-1 データ待ち
Thread-0 16 追加
Thread-1 16 を消費
...
synchronized
メソッドにおいて、while文で条件が満たされるまでwait()
して、満たされたら処理を再開し最後にnotifyAll()
if
では不適切。他のスレッドにnotifyAll()
されたからといって、条件が満たされているとは限らない。notifyAll()
すれば直ちに他のスレッドが動き出すわけではない。synchronized
メソッドを抜けてロックを解除してから。wait()
, notifyAll()
できない。synchronized
メソッドの実行でロックを獲得できる。public synchronized void doSomething() throws InterruptedException {
while( !求めている条件 ) {
wait();
}
// 処理
notifyAll();
}
Thread.currentThread()
Thread#getPriority()
: 優先順位取得Thread#setPriority(int newPriority)
: 設定Thread.MIN_PRIORITY
: 最低の優先順位Thread.NORM_PRIORITY
: 通常の優先順位Thread.MAX_PRIORITY
: 最高の優先順位Thread
のサブクラス化と、Runnable
インタフェースの実装の2種類があるrun()
に書き、start()
で開始する。スレッドの一時停止sleep()
、他スレッドの終了待ち合わせjoin()
も可能。synchronized
によるメソッドの同期化でロックを獲得できる。wait()
, notifyAll()
, notify()
によりロックの解放・待機、待機中スレッドの「起こし」を行えるpackage A;
public class XXX {
}
class YYY {
}
package B;
public class XXX {
}
package
パッケージ名;package A;
public class XXX {
}
class YYY {
}
package B;
public class XXX {
}
java.util.Random r = new java.util.Random();
(java.util
パッケージ中のRandom
クラス)import
を書いておけば以降は省略できるimport
宣言シングル・タイプ・インポート宣言
import java.util.Random;
import java.util.Calendar;
class Foo {
Random r = new Random();
Calendar c = new Calendar();
... }
タイプ・インポート・オン・デマンド宣言
import java.util.*;
class Foo {
Random r = new Random();
Calendar c = new Calendar();
... }
java.lang
パッケージimport
されているjava.lang.String s = "文字列";
import java.lang.*;
が暗黙にされているのでString s = "文字列";
と書けてしまうjava.lang.String
, java.util.Random
public
クラスに「アクセス不可」package mytool;
import mytool.security.Trust;
public class TextParser {
Trust t = ...;
}
package mytool.security;
import mytool.TextParser;
public class User {
TextParser p = ...; }
package mytool.security;
class Trust {
}
import
における「*」は、パッケージを「引っ掛けない」。unnamed package
)java.lang.Object
をRectangle
クラスで拡張していた!C:¥WORK> javac HelloWorld.java
C:¥WORK> javac mytool¥TextParser.java
C:¥WORK> javac mytool¥security¥User.java
C:¥WORK> javac mytool¥security¥Trust.java
C:¥WORK> java mytool.TextParser
// 組織の場合:waseda.jp
package jp.waseda;
package jp.waseda.testpackage;
// 個人の場合:ess@waseda.jp
package jp.waseda.ess;
package jp.waseda.ess.mypackage;
HelloWorld.java
を移動させて、ソースファイルを修正せよpublic class HelloWorld {
public static void main(String[] args) {
System.out.println("Hello World.");
}
}
-verbose:class
オプションを付けることで、仮想マシンがロードするクラスを順に確認できるjava –verbose:class jp.waseda.ess.HelloWorld
[Loaded java.lang.Object from shared objects file]
...
[Loaded java.lang.String from shared objects file]
...
[Loaded java.lang.ClassLoader from shared objects file]
[Loaded java.lang.System from shared objects file]
...
[Loaded java.io.FileReader from shared objects file]
...
[Loaded jp.waseda.ess.HelloWorld from
file:/C:/WORK/jp/waseda/ess/]
Hello World.
...
[Loaded java.util.LinkedHashMap$KeyIterator from shared objects file]
import static java.lang.System.out;
import static java.lang.Math.sin;
import static java.lang.Math.*;
public class Foo {
public static void main(String[] args){
out.println("text");
out.println(sin(1.0));
out.println(cos(1.0));
}
}
// 左と同じ意味
public class Foo {
public static void main(String[] args) {
System.out.println("text");
System.out.println(Math.sin(1.0));
System.out.println(Math.cos(1.0));
}
}
public
修飾子 | アクセスの可否 |
---|---|
無指定 | 同一パッケージからのみ使える |
public |
あらゆるところから使える |
package A;
public class XXX {
}
class YYY { }
package B;
public class XXX {
}
private
、無指定、protected
、public
修飾子 | publicクラスのメソッド/コンストラクタ/フィールド | 無指定クラスのメソッド/コンストラクタ/フィールド |
---|---|---|
private |
クラス内でのみ使える。メソッド/フィールドはサブクラスに継承されない(コンストラクタはそもそも継承されない)。 | |
無指定 | 同一パッケージからのみ | |
protected |
同一パッケージとサブクラスからのみ | |
public |
あらゆるところから | 同一パッケージからのみ |
private
、無指定、protected
、public
修飾子 | publicインタフェースのメソッド/コンストラクタ/フィールド | 無指定インタフェースのメソッド/コンストラクタ/フィールド |
---|---|---|
private |
<指定不可> | |
protected |
<指定不可> | |
無指定=public |
あらゆるところから | 同一パッケージからのみ |
private
メソッドやprivate
フィールドはサブクラスに「継承されない」class Rectangle {
private int width = 0;
private void setSize(
int w, int h){
...
}
public int getArea() {
return width * height;
}
}
// 外部からのアクセス
Rectangle r = new Rectangle();
r.width = 50; // NG!
r.setSize(100,99); // NG!
System.out.println(r.getArea()); // OK
encapsulation
)private
にして外部に非公開public
にして公開class NamedRectangle extends Rectangle {
private String name = null;
// nameのアクセスメソッド (getter)
String getName() {
return name;
}
// nameのアクセスメソッド (setter)
void setName(String n) {
name = n;
} }
public class MyClass {
private MyClass() {
... //共通の初期化処理
}
public MyClass(String s) {
this();
... //付加的な初期化処理
}
public MyClass(int x, int y) {
this();
... //付加的な初期化処理
} }
package java.lang;
public class Math {
public static double PI
= 3.14159265358979323846;
private Math() {}
public static double sin(double a){
...
}
}
getInstance()
を追加し、同メソッドを Client に呼ばせればよい。その唯一のインスタンスを最初にどのように作っておけばよいだろう?public class Repository {
public int size = 0;
public Repository() {
}
}
public class Client {
public static void main(String[] args) {
Repository r1 = new Repository();
Repository r2 = new Repository();
r1.size = 100;
r2.size = 500;
System.out.println(r1.size);
System.out.println(r2.size);
} }
Singleton
パターン (1)private
)にし、Singleton
パターン (2)abstract
修飾子abstract method
)abstract class
)abstract class Player {
void loop(int n) {
for(int i = 0; i < n; i++) {
play();
}
}
// 抽象メソッド
abstract void play();
}
class TextPlayer extends Player {
// 抽象メソッドの実装
void play() {
System.out.println("TEXT");
}
public static void main(String[] args) {
Player p = new TextPlayer();
p.loop(2);
}
}
static
: インスタンス生成不要でクラス固有のフィールド/メソッドfinal
: 継承できず上書きできない、値を(宣言時の初期化後に)代入できないクラス/フィールド/メソッドpublic
, protected
, private
: アクセス制限abstract
: 抽象クラス/メソッド。継承されて実装されて始めてインスタンス生成/アクセス可import
節によるインポート宣言で、クラスやインタフェース利用時のパッケージ名を省略できるprivate
は、情報隠蔽のためよく用いられる