2016.05.26

ノイズ画像の作成

はじめに

乱数を使って指定サイズのノイズ画像を作る方法を解説します。ノイズが発生する割合をパーセントで指定することで、ノイズの密度を調整することができるようにします。

ノイズ画像

上の画像の例は、ノイズの発生率が30%のノイズ画像です。黒色がノイズピクセルです。

Javaソースコード

Noise1.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
import java.awt.image.BufferedImage;
import java.io.File;
import javax.imageio.ImageIO;
import java.io.IOException;

public class Noise1 {
	public static void main( String[] args ) {
		// 変数宣言
		int w, h;		// 画像の幅・高さ
		double noiserate;	// ノイズの割合
		String outname;	// 出力ファイル名
		BufferedImage img = null;	// 画像格納クラス

		// 入力した引数が4以上かを調べる
		if ( 4 > args.length ) {
			// 入力した引数が4未満の場合、使用方法を表示する
			System.out.println( 
				"Noise1 [出力PNG名][幅][高][ノイズの割合%]" );
			return;
		}

		try {
			// 引数を変換し、画像サイズに代入
			w  =  Integer.valueOf( args[ 1 ] );
			h  =  Integer.valueOf( args[ 2 ] );
			// 引数を変換し、ノイズの割合に代入
			noiserate = Double.valueOf( args[ 3 ] );
		}
		catch( NumberFormatException ne )
		{
			System.out.println( "引数が不正です" );
			return;
		}

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

		// 新しい画像を作成
		// 24ビットカラーの画像を作成
		try {
			img = new BufferedImage( w, h,
						 BufferedImage.TYPE_INT_RGB );
		} catch ( Exception e ) {
			// 画像作成に失敗したときの処理
			e.printStackTrace();
			return;
		}

		// ノイズ画像の作成
		int        x, y;
		int        r, g, b, color;
		double rate;

		for ( y = 0; y < h; ++ y ) {
			for ( x = 0; x < w; ++ x ) {
				// 乱数を発生
				rate = Math.random() * 100.0;

				// ノイズ色を決定
				if ( noiserate > rate ) {
					// ノイズとして黒色を設定
					r = 0;
					g =0;
					b =0;
				}
				else {
					// ノイズでない白色を設定
					r = 255;
					g = 255;
					b = 255;
				}

				// r,g,bの色を合成
				color = ( r << 16 ) + ( g << 8 ) + b;

				// 合成した色を(x,y)に設定
				img.setRGB( x, y, color );
			}
		}

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

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

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

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

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

C:\talavax\javasample>javac Noise1.java

実行

C:\talavax\javasample>java Noise1 noise05.png 256 256 5.0

2つ目と3つ目の引数ピクセル座標、4つ目の引数ノイズピクセルの発生率をパーセントで指定します。これらの引数からノイズ画像を作成し、1つ目の引数で指定したPNGファイル名で保存します。

この例では、ノイズピクセル発生率5%で、サイズ256x256のPNGファイル(noise05.png)を作成しています。

実行結果 ノイズ画像(5%)

ノイズ画像(5%)

ノイズ発生率5%、幅256ピクセル、高さ256ピクセルノイズ画像が作成されました。黒色がノイズピクセルです。

以下は、ノイズ発生率20%、50%、70%、90%の画像です。

ノイズ画像(20%)

ノイズ画像(20%)

ノイズ画像(50%)

ノイズ画像(50%)

ノイズ画像(70%)

ノイズ画像(70%)

ノイズ画像(90%)

ノイズ画像(90%)

Javaソースコードの解説

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

001
002
003
004
005
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 Noise1 {

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

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

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

008
009
010
011
012
		// 変数宣言
		int w, h;		// 画像の幅・高さ
		double noiserate;	// ノイズの割合
		String outname;	// 出力ファイル名
		BufferedImage img = null;	// 画像格納クラス

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

014
015
016
017
018
019
020
		// 入力した引数が4以上かを調べる
		if ( 4 > args.length ) {
			// 入力した引数が4未満の場合、使用方法を表示する
			System.out.println( 
				"Noise1 [出力PNG名][幅][高][ノイズの割合%]" );
			return;
		}

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

022
023
024
025
026
027
028
029
030
031
032
033
		try {
			// 引数を変換し、画像サイズに代入
			w  =  Integer.valueOf( args[ 1 ] );
			h  =  Integer.valueOf( args[ 2 ] );
			// 引数を変換し、ノイズの割合に代入
			noiserate = Double.valueOf( args[ 3 ] );
		}
		catch( NumberFormatException ne )
		{
			System.out.println( "引数が不正です" );
			return;
		}

与えられた引数をそれぞれ、ピクセル幅、、ピクセルの高さ、ノイズの割合を格納する変数に代入しています。

035
036
		// 出力PNG名をoutnameに代入(拡張子".png"省略なし)
		outname = args[ 0 ];

与えられた引数を、出力PNG名の変数(outname)代入しています。

038
039
040
041
042
043
044
045
046
047
		// 新しい画像を作成
		// 24ビットカラーの画像を作成
		try {
			img = new BufferedImage( w, h,
						 BufferedImage.TYPE_INT_RGB );
		} catch ( Exception e ) {
			// 画像作成に失敗したときの処理
			e.printStackTrace();
			return;
		}

BufferedImageクラスコンストラクタで、新しいBufferedImageを構築しています。

BufferedImageコンストラクタ

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

imgTypeで指定しているTYPE_INT_RGBは、整数ピクセルにパックされた 8 ビット RGB 色成分によるイメージを表します。これを指定することで24ビット画像を作成できます。

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

049
050
051
052
		// ノイズ画像の作成
		int        x, y;
		int        r, g, b, color;
		double rate;

ノイズ画像作成で使う変数を宣言しています。

054
055
		for ( y = 0; y < h; ++ y ) {
			for ( x = 0; x < w; ++ x ) {

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

056
057
				// 乱数を発生
				rate = Math.random() * 100.0;

乱数からパーセントを計算した値(0.0~99.999..)を変数rateに代入しています。Math.randomメソッドは、0.0~0.99999..の乱数を発生するので、その値に100.0を掛けることでパーセントを計算しています。

Math.randomメソッド

public static double Math.random()
・乱数を返します。

  パラメータ なし

  戻り値     0.0以上、1.0未満の乱数
059
060
061
062
063
064
065
066
067
068
069
070
071
				// ノイズ色を決定
				if ( noiserate > rate ) {
					// ノイズとして黒色を設定
					r = 0;
					g =0;
					b =0;
				}
				else {
					// ノイズでない白色を設定
					r = 255;
					g = 255;
					b = 255;
				}

引数で指定したノイズの割合(noiserate)より、乱数から作成した値(rate)が小さければノイズと判定し、ピクセルの色を黒にします。大きければ色を白にします。この処理で、指定したノイズ割合を大きくすれば、ノイズの発生率が高くなるようにしています。

073
074
				// r,g,bの色を合成
				color = ( r << 16 ) + ( g << 8 ) + b;

計算したRGB合成して、色(変数color)を作成しています。

076
077
				// 合成した色を(x,y)に設定
				img.setRGB( x, y, color );

色(変数color)を(x,y)に代入しています。

081
082
083
084
085
086
087
088
089
		try {
			// imgをoutname(出力PNG)に保存
			boolean result;
			result = ImageIO.write( img, "PNG", new File( outname ) );
		} catch ( Exception e ) {
			// outname(出力PNG)の保存に失敗したときの処理
			e.printStackTrace();
			return;
		}

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

関連コンテンツ

乱数の意味と、Math.randomメソッドの使い方をソースコードを使って詳しく解説しています。

2015.12.27

画像って何?

2022.07.25

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

2015.12.22

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

2015.11.29

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

2020.03.23

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

2020.03.23

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

2022.08.03

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

2020.03.23

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

2020.03.23

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

2022.12.13

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

2020.03.23

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

2020.03.20

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

2020.03.23

数値を2進数で表したときの各桁の「0」と「1」に対して演算を行えます。4種類の演算、AND(論理積)、OR(論理和)、XOR(排他的論理和)、NOT(否定)を詳しく説明しています。

2016.03.26

自然数と整数って何が違う?

2020.03.23

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

2016.03.02

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

2016.11.23

コンピュータは、いくつかの装置から構成されています。その主な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

乱数を使って指定の画像にノイズ画像を加える方法を解説します。

2016.06.14

乱数を使って指定サイズのノイズ画像を作る方法を解説します。ノイズが発生する割合はパーセントで指定します。

2020.11.11

画像全体をぼやけた感じにする処理のJavaソースコードを紹介しています。、

2016.08.09

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

2024.07.13

広告