2016.08.09

画像をぼかす

はじめに

ぼかし処理とは、画像全体ををぼやけた感じにする処理です。

この処理は、ノイズを目立たせなくしたり、画像を合成したときの接続部の違和感をなくしたりする処理の1つです。

元の画像
元の画像
 
右矢印
ぼかし処理後の画像
ぼかし処理後の画像

画像ぼかし方は、注目する画素と周りの8ピクセルの色の平均値を計算し、その色を注目画素の位置に設定することで画像ぼかします。下図が具体的な例です。

ぼかし処理例

3x3の真ん中を注目画素としたときに、その周りのRGBの値の平均値をそれぞれ求めています。

上の例では、Rの平均は(255+255+255+255+255+255+255+255+255)÷9=255、Gの平均は(255+0+0+255+255+0+0+0+0)÷9=85、Bの平均は(255+0+0+255+255+0+0+0+0)÷9=85となります。

この方法で計算したRGBの値を注目画素座標に設定します。

注目画像の位置(x,y)の色を、上記の方法で計算したRGBに変更していくと、意図した「ぼかし処理」になりません。

ぼかし処理をしたRGBの値を、次のぼかし処理の計算で使用するためです。

これを避けるために、元の画像RGBは変更せずに、元の画像と同じサイズの画像を新しく作成します。

新しく作成した画像の全ての画素の位置(x,y)に、元の画像の位置(x,y)で計算したRGBの値を格納していきます。

こうすることで、意図した「ぼかし処理」を作成することができます。

Javaソースコード

これを、Javaソースコードにすると以下のようになります。

SoftFocus.java

001
002
003
004
005
006
007
008
009
010
011
012
013
014
015
016
017
018
019
020
021
022
023
024
025
026
027
028
029
030
031
032
033
034
035
036
037
038
039
040
041
042
043
044
045
046
047
048
049
050
051
052
053
054
055
056
057
058
059
060
061
062
063
064
065
066
067
068
069
070
071
072
073
074
075
076
077
078
079
080
081
082
083
084
085
086
087
088
089
090
091
092
093
094
095
096
097
098
099
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
import java.awt.image.BufferedImage;
import java.io.File;
import javax.imageio.ImageIO;
import java.io.IOException;

public class SoftFocus {
	public static void main( String[] args ) {
		// 結果格納フラグ
		boolean result;
		// ファイル名
		String inname, outname;
		// 画像格納クラス
		BufferedImage img = null;

		// 入力した引数が2つ以上かを調べる
		if ( 2 > args.length ) {
			// 入力した引数が2つ未満の場合、使用方法を表示する
			System.out.println( "SoftFocus [入力JPEG名]  [出力JPEG名]" );
			return;
		}

		// 入力JPEG名をinnameに代入(拡張子".jpg"省略なし)
		inname  = args[ 0 ];
		// 出力JPEG名をoutnameに代入(拡張子".jpg"省略なし)
		outname = args[ 1 ];

		try {
			// inname(入力JPEG)を読み込んでimgにセット
			img = ImageIO.read( new File( inname ) );
		} catch (Exception e) {
			// inname(入力JPEG)の読み込みに失敗したときの処理
			 e.printStackTrace();
			return;
		}

		// 画像の色の持ち方をチェック
		if ( BufferedImage.TYPE_3BYTE_BGR != img.getType() )
		{
			System.out.println( "対応していないカラーモデルです!("
									 + inname +")" );
			return;
		}

		// ぼかし処理
		int x, y, dx, dy;
		int width, height;
		int color, r, g, b;
		int sumr, sumg, sumb;
		int p;
		int newcolor;
		BufferedImage newimg = null;

		// 画像サイズの取得
		width = img.getWidth();
		height= img.getHeight();

		// 新しい画像を作成
		// 元の画像と同じ状態の24ビットカラー画像を作成
		try {
			newimg = new BufferedImage( width, height, img.getType() );
		} catch ( Exception e ) {
			// 画像作成に失敗したときの処理
			e.printStackTrace();
			return;
		}

		// ぼかし処理
		for ( y = 0; y < height; ++ y ) {
			for ( x = 0; x < width; ++ x ) {
				if ( ( 0 == x ) || ( 0 == y ) ||
					( x == ( width - 1 ) ) || ( y == ( height - 1 ) ) ) {
					// (x,y)が画像の外枠の場合、元の画像の色を取得
					newcolor = img.getRGB( x, y );
				}
				else {
					// 9画素の平均値を計算
					sumr = 0;		// R値の合計
					sumg = 0;	// G値の合計
					sumb = 0;	// B値の合計
					for ( dy = -1; dy <= 1; ++ dy ) {
						for ( dx = -1; dx <= 1; ++ dx ) {
							// (x,y)の色を取得
							color = img.getRGB( x + dx, y + dy );

							// 色をr,g,bに分解
							r = ( color >> 16 ) & 0xff;
							g = ( color >> 8 ) & 0xff;
							b = color & 0xff;
						
//							
							sumr += r;
							sumg += g;
							sumb += b;
						}
					}
					sumr /= 9;	// R値の平均
					sumg /= 9;	// G値の平均
					sumb /= 9;	// B値の平均

					// r,g,bの色を合成
					newcolor = (sumr << 16 ) + ( sumg << 8 ) + sumb;
				}

				// 新しい色を(x,y)に設定
				newimg.setRGB( x, y, newcolor );
			}
		}

		try {
			// imgをoutname(出力JPEG)に保存
			result = ImageIO.write( newimg, "jpeg", new File( outname ) );
		} catch ( Exception e ) {
			// outname(出力JPEG)の保存に失敗したときの処理
			e.printStackTrace();
			return;
		}

		// 正常に終了
		System.out.println( "正常に終了しました" );
	}
}

コンパイル ソースコードが「ANSI」の場合

C:\talavax\javasample>javac -encoding sjis SoftFocus.java

コンパイル ソースコードが「UTF-8」の場合

C:\talavax\javasample>javac SoftFocus.java

実行

C:\talavax\javasample>java SoftFocus noise2.jpg softFocus.jpg

1つ目の引数で元のJPEGファイル名、2つ目の引数ぼかし処理をしたJPEG画像ファイル名を指定します。

・元の画像(noise2.jpg)

元画像

・ぼかし処理をした画像(SoftFocus.jpg)

ぼかし処理をした画像

画像に中にある白いノイズが、周りの色に少し溶け込んでいることが分かります。

Javaソースコードの解説

ここからは、このJavaソースコードを上から順番に解説していきます。

001
002
003
004
import java.awt.image.BufferedImage;
import java.io.File;
import javax.imageio.ImageIO;
import java.io.IOException;

Javaクラスライブラリの中から「java.awt.image.BufferedImage」と「java.io.File」と「javax.imageio.ImageIO」と「java.io.IOException」と「java.awt.image.RasterFormatException」というパッケージにあるクラスを、このプログラム内で使うために記述します。 この記述により、ImageIOクラスBufferedImageクラスRasterFormatExceptionが利用できるようになります。

006
public class SoftFocus {

クラス名を、SoftFocusとしています。

007
	public static void main( String[] args ) {

このmainメソッドからプログラムを実行します。

008
009
010
011
012
013
		// 結果格納フラグ
		boolean result;
		// ファイル名
		String inname, outname;
		// 画像格納クラス
		BufferedImage img = null;

このプログラムで使う変数を宣言しています。

015
016
017
018
019
020
		// 入力した引数が2つ以上かを調べる
		if ( 2 > args.length ) {
			// 入力した引数が2つ未満の場合、使用方法を表示する
			System.out.println( "SoftFocus [入力JPEG名]  [出力JPEG名]" );
			return;
		}

2つ以上の引数が与えられたかをチェックし、2つ未満の場合に、使い方のメッセージを表示し、returnによってmainメソッドを抜けています。

022
023
024
025
		// 入力JPEG名をinnameに代入(拡張子".jpg"省略なし)
		inname  = args[ 0 ];
		// 出力JPEG名をoutnameに代入(拡張子".jpg"省略なし)
		outname = args[ 1 ];

与えられた引数をそれぞれ、入力JPEGファイル名、出力JPEGファイル名、ノイズの割合を格納する変数に代入しています。

027
028
029
030
031
032
033
034
		try {
			// inname(入力JPEG)を読み込んでimgにセット
			img = ImageIO.read( new File( inname ) );
		} catch (Exception e) {
			// inname(入力JPEG)の読み込みに失敗したときの処理
			 e.printStackTrace();
			return;
		}

入力JPEG名の変数(inname)を読み込んで、BufferedImageクラスのimgに格納しています。この処理には、ImageIOクラスreadメソッドを使います。

ImageIO.readメソッド

public static BufferedImage read( File input ) throws IOException
・Fileオブジェクトを復元した結果をBufferedImageに格納します。

  パラメータ input : Fileオブジェクト

  戻り値     inputを復元したBufferedImageaを返します。

try { ~ } catchは、失敗する可能性がある処理を波括弧で囲み、その処理に失敗したときにcatch { ~ }の波括弧で囲まれた処理を実行するということです。この場合は、JPEGファイル名が不正であったり、存在していなかったり、フォーマットが違っているなどが原因で処理が失敗する可能性があります。処理が失敗するとreturnによってmainメソッドを抜けるようにしています。

036
037
038
039
040
041
042
		// 画像の色の持ち方をチェック
		if ( BufferedImage.TYPE_3BYTE_BGR != img.getType() )
		{
			System.out.println( "対応していないカラーモデルです!("
									 + inname +")" );
			return;
		}

BufferedImageクラスgetTypeメソッド画像のイメージ型を取得しています。

BufferedImage.getTypeメソッド

public static int getType()
・イメージ型を返します。

  パラメータ なし

   戻り値     BufferedImage のイメージ型を返します。
044
045
046
047
048
049
050
051
		// ぼかし処理
		int x, y, dx, dy;
		int width, height;
		int color, r, g, b;
		int sumr, sumg, sumb;
		int p;
		int newcolor;
		BufferedImage newimg = null;

ぼかし処理で使う変数を宣言しています。

053
054
055
		// 画像サイズの取得
		width = img.getWidth();
		height= img.getHeight();

widthに画像の幅(ピクセル)、heightに画像の高さ(ピクセル)を代入しています。

057
058
059
060
061
062
063
064
065
		// 新しい画像を作成
		// 元の画像と同じ状態の24ビットカラー画像を作成
		try {
			newimg = new BufferedImage( width, height, img.getType() );
		} catch ( Exception e ) {
			// 画像作成に失敗したときの処理
			e.printStackTrace();
			return;
		}

BufferedImageクラスコンストラクタで、もとの画像と同じ大きさのイメージ形式のBufferedImageを構築しています。

BufferedImageコンストラクタ

BufferedImage( int width, int height, int imageType )
・新しい BufferedImage を構築します。
  パラメータ width     : 構築する画像の横ピクセル
        height    : 構築する画像の縦ピクセル
        imageType : 構築する画像のイメージ形式

try { ~ } catchは、失敗する可能性がある処理を波括弧で囲み、その処理に失敗したときにcatch { ~ }の波括弧で囲まれた処理を実行するということです。この場合は、JPGファイル名が不正であったり、存在していなかったり、フォーマットが違っているなどが原因で処理が失敗する可能性があります。処理が失敗するとreturnによってmainメソッドを抜けるようにしています。

067
068
069
		// ぼかし処理
		for ( y = 0; y < height; ++ y ) {
			for ( x = 0; x < width; ++ x ) {

画像の中の全てのピクセルの座標を参照するループをつくっています。具体的には、変数yを0~height-1、変数xを0~width-1に変化させています。

070
071
072
073
074
				if ( ( 0 == x ) || ( 0 == y ) ||
					( x == ( width - 1 ) ) || ( y == ( height - 1 ) ) ) {
					// (x,y)が画像の外枠の場合、元の画像の色を取得
					newcolor = img.getRGB( x, y );
				}

(x,y)が画像の縁の場合、ぼかし処理をせずに元の画像の色をそのまま採用しています。x=0またはy=0またはx=width-1またはy=height-1の条件を満たしているとき画像の縁と判定しています。

この判定を入れている理由は、縁にある画像注目画素にしたときに、その周りの8画素の一部が元画像の外に存在するため、8個すべての色情報が取れないからです。ここで紹介する方法は処理を簡単にするために、このような手法を使っています。

076
077
078
079
080
081
082
083
084
085
086
087
088
089
090
091
092
093
094
095
096
097
098
					// 9画素の平均値を計算
					sumr = 0;		// R値の合計
					sumg = 0;	// G値の合計
					sumb = 0;	// B値の合計
					for ( dy = -1; dy <= 1; ++ dy ) {
						for ( dx = -1; dx <= 1; ++ dx ) {
							// (x,y)の色を取得
							color = img.getRGB( x + dx, y + dy );

							// 色をr,g,bに分解
							r = ( color >> 16 ) & 0xff;
							g = ( color >> 8 ) & 0xff;
							b = color & 0xff;
						
//							
							sumr += r;
							sumg += g;
							sumb += b;
						}
					}
					sumr /= 9;	// R値の平均
					sumg /= 9;	// G値の平均
					sumb /= 9;	// B値の平均

注目画素座標(x,y)とその周りの8画像RGBのそれぞれの輝度を合計し、RGBそれぞれの値を9で割り平均値を計算しています。

100
101
					// r,g,bの色を合成
					newcolor = (sumr << 16 ) + ( sumg << 8 ) + sumb;

RGB平均値を合成し、newcolorに格納しています。

104
105
				// 新しい色を(x,y)に設定
				newimg.setRGB( x, y, newcolor );

色(変数newcolor)を新しく作ったnewimgの(x,y)に代入しています。これにより、元画像(img)からRGB平均値を計算することができます。imgを変更すると平均値を求めた値で、さらに平均値を計算することになり狙いどおりの画像に変換できません。

109
110
111
112
113
114
115
116
		try {
			// imgをoutname(出力JPEG)に保存
			result = ImageIO.write( newimg, "jpeg", new File( outname ) );
		} catch ( Exception e ) {
			// outname(出力JPEG)の保存に失敗したときの処理
			e.printStackTrace();
			return;
		}

BufferedImageクラスのnewimgのメモリ内のデータを、出力JPEG名の変数(outname)に格納されているファイル名で保存します。この場合は、JPEGファイル名が不正であったり、保存先のHDDなどが存在していなかったり、空き容量が少ないなどが原因で処理が失敗する可能性があります。

ImageIO.wrireメソッド

public static boolean write( RenderedImage im, String formatName, File output ) throws IOException
・BufferedImageを画像ファイルに保存します。

  パラメータ RenderedImage : 保存するRenderedImage
                  formatName     : 画像ファイルのフォーマット(png/jpeg/bmp/gifなど)
                  output             : Fileオブジェクト

  戻り値     保存に成功するとtrue、失敗するとfalseを返します。
118
119
		// 正常に終了
		System.out.println( "正常に終了しました" );

全ての処理が正常終了すると、ここまで処理が実行されます。

以上です。

関連コンテンツ

一般に使われている画像の色の種類ってどんなものがありますか?。色の種類ごとの用途も説明しています。

2015.12.22

一般に使われている画像フォーマットには、いろいろな種類があります。画像フォーマットBMP、JPEG、PNG、GIF、TIFFの特徴を知ってますか?

2015.11.29

画像に対する空間フィルタリングについて解説しています。

2024.07.13

乱数を使って指定サイズのノイズ画像を作る方法を解説します。Javaのソースコード付きです。

2016.05.26

PCやスマートフォンのディスプレイに表示されている色、プリンターで印刷される色の仕組みについて解説しています。

2020.03.23

デジタル画像データを構成している要素について解説しています。

2015.12.22

処理を繰り返すために使用するfor文について解説しています。

2020.03.23

条件式を判断して処理を分岐する方法を詳しく説明しています。

2023.03.20

変数やクラスに格納されている値をコンソール出力する方法は?

2020.03.23

メソッドを抜けるときに使用するreturn文について説明しています。

2020.03.20

プログラムの最初に実行されるメソッドは?

2022.12.13

プログラミングで使う変数って何?

2020.03.23

Javaのプログラムを書いてみませんか?プログラムの書き方をくわしく説明しています。

2020.03.23

「Javaソースコード」から実行可能な「オブジェクトコード」に変換する方法をくわしく説明しています。

2020.03.23

画像って何?

2022.07.25

複数の数値の合計値と平均値を計算するプログラムをJavaのソースコードを使って解説しています。

2020.03.23

平面上の位置を表す座標系の1つXY座標系について詳しく解説。

2020.03.23

プログラミング言語とは?種類や特徴について説明しています。

2022.08.03

メソッドの定義方法を詳しく解説しています。Javaのサンプルソースコードを使った説明もあります。

2020.03.23

Javaプログラムの構成について解説しています。詳しくは、こちらをご覧ください。

2020.03.23

コンピュータに保存されたファイルを特定するための名前がファイル名です。その付け方は?

2016.11.23

画像の座標系はどのように定義されていますか?

2020.03.23

繰り返し処理の作り方を解説しています。

2016.03.02

for文で変数名iがよく使われる理由について説明しています。興味のある方は是非。

2022.08.29

コンピュータは、いくつかの装置から構成されています。その主な5つの装置(機能)って何?

2022.07.10

データを記憶する部品のことで、ハードディスク、ハードディスクドライブと呼ばれます。電源の供給がなくてもデータが消えない記憶装置です。

2022.07.14

「ゆるゆるプログラム」のコンテンツを紹介しています。興味のある方はこの記事をご覧ください。

2020.03.23

Javaの学習に役立つソースコードを多数紹介しています。是非、ご覧ください。

2022.09.10

Javaを使った簡単な応用プログラム(生年月日から年齢を計算プログラムなど)を紹介しています。

2022.07.07

Swingパッケージを使ってグラフィック表示を行う方法を解説しています。

2020.03.23

画像フォーマット形式・色・大きさ・傾きなどの変更、特定の図形(文字・記号など)を見つけたり、取り出したりする画像処理について詳しく解説。

2015.11.29

三角形、台形、円などいろいろな図形の面積を計算するプログラムを紹介しています。詳しくは、記事をご覧ください。

2021.05.18

配列を使うJavaソースコードを多数紹介しています。

2021.05.18

繰り返し処理を使ったJavaのソースコードサンプルを紹介しています。

2020.03.23

数学に関係するJavaのメソッドやソースコードなどを紹介しています。

2022.10.25

StringクラスとStringBuilderクラスを利用したプログラミングの仕方を紹介しています。

2016.12.16

日本で使われてきた伝統文様「和柄」について解説しています。

2022.07.27

プログラミング、ITに関する用語をまとめています。

2022.10.17

広告