2017.11.17

N値化

はじめに

N値化とは、カラーの画像をN個のグレー色(白/灰色/黒)で表現した画像に変換することです。

変換には、2個以上の色を0~255の値で指定します。変換前の画素の色を加重平均法を使って計算したグレー値に最も近い指定値を求めます。この求められた値が変換後のグレー値です。

Javaソースコード

次に、フルカラー(24bit)画像グレースケール画像に変換するソースコードを解説します。以下が、そのソースコード例です。

NColor.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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
import java.awt.image.BufferedImage;
import java.io.File;
import javax.imageio.ImageIO;
import java.io.IOException;

public class NColor {
	// 最も近い値を格納した配列の添え字を求める
	public static int getNearestValue( int[] list, int v )
	{
		// 変数の宣言
		int i;		// ループ用
		int num;	// 配列の添え字
		int minv;	// 配列値-指定値vの絶対値

		// 配列の個数が1未満の処理
		if ( 1 >list.length ) return -1;

		// 指定値と全ての配列値の差を比較
		num = 0;
		minv = Math.abs( list[ 0 ] - v );
		for ( i = 1; i < list.length; ++ i ) {
			if ( Math.abs( list[ i ] - v ) < minv ) {
				num = i;
				minv = Math.abs( list[ i ] - v );
			}
		}

		return num;
	}


	// メイン
	public static void main( String[] args ) {
		// 結果格納フラグ
		boolean result;
		// ファイル名
		String inname, outname;
		// 画像格納クラス
		BufferedImage img = null;
		// 色リスト
		int[] colorlist;
		// 256色→N色変換テーブル
		int[] converttbl;

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

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

		// 色リストの作成
		colorlist = new int[ args.length - 2 ];
		for ( int i = 2; i < args.length; ++ i ) {
			// 値をcolorlistに代入
			try {
				// 文字列をintに変換して値を代入
				colorlist[ i - 2 ] = Integer.valueOf( args[ i ] );
			}
			catch( NumberFormatException ne )
			{
				System.out.println( "引数が不正です" );
				return;
			}
		}

		// JPEG画像の読み込み
		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;
		}

		// N値に変換
		int x, y;
		int width, height;
		int color, r, g, b;
		int p;
		int newcolor;

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

		// 256色→N色変換テーブル作成
		converttbl = new int[ 256 ];
		for ( int i = 0; i < 256; ++ i ) {
			// iに一番近い値を取得
			int num = getNearestValue( colorlist, i );
			if ( 0 <= num ) {
				// 配列の添え字numから色を取得
				p = colorlist[ num ];
				// pを0~255の範囲に変換
				if ( 0 > p )
					p = 0;
				else {
					if ( 255 < p ) p = 255;
				}
				converttbl[ i ] = p;
			}
		}

		// N色変換
		for ( y = 0; y < height; ++ y ) {
			for ( x = 0; x < width; ++ x ) {
				// (x,y)の色を取得
				color = img.getRGB( x, y );

				// 色をr,g,bに分解
				r = ( color >> 16 ) & 0xff;
				g = ( color >> 8 ) & 0xff;
				b = color & 0xff;

				// rgbに重みを付けてpを計算
				p =  (int)( 0.29891 *  (double)r
					 + 0.58661 *  (double)g
						 + 0.11448 *  (double)b );

				// 256色→N色変換テーブルから変換色を取得
				p = converttbl[ p ];

				// r,g,bにpを代入
				r = p;
				g = p;
				b = p;

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

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

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

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

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

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

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

C:\talavax\javasample>javac NColor.java

実行

C:\talavax\javasample>java NColor sampleimage001_400x320.jpg ncolor2.png 0 255

1つ目の引数で渡したJPEGファイルをN値変換し、2つ目の引数で指定したPNGファイル名で保存します。この場合、3つ目の引数から2つの値を指定しているので2値化変換になります。

・元の画像(sampleimage001_400x320.jpg)

元画像

・変換後の画像(ncolor2.png)

2値化画像

3つの値で実行した例は、以下のとおりです。

実行

C:\talavax\javasample>java NColor sampleimage001_400x320.jpg ncolor3.png 0 200 255

1つ目の引数で渡したJPEGファイルをN値変換し、2つ目の引数で指定したPNGファイル名で保存します。この場合、3つ目の引数から3つの値を指定しているので3値化変換になります。

・元の画像(sampleimage001_400x320.jpg)

元画像

・変換後の画像(ncolor3.png)

3値化画像

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」というパッケージにあるクラスを、このプログラム内で使うために記述します。 この記述により、ImageIOクラスBufferedImageクラスが利用できるようになります。

006
public class NColor {

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

007
008
	// 最も近い値を格納した配列の添え字を求める
	public static int getNearestValue( int[] list, int v )

指定値(v)に1番近い値を持つ配列(list[])の添え字を求めるメソッドです。

010
011
012
013
		// 変数の宣言
		int i;		// ループ用
		int num;	// 配列の添え字
		int minv;	// 配列値-指定値vの絶対値

メソッドで使う変数を宣言しています。

015
016
		// 配列の個数が1未満の処理
		if ( 1 >list.length ) return -1;

配列(list[])に格納されている値が1個未満の場合、配列添え字を求めることができないので、結果として-1を返しています。

018
019
020
		// 指定値と全ての配列値の差を比較
		num = 0;
		minv = Math.abs( list[ 0 ] - v );

求める値の候補(初期値)として配列の先頭(list[0])の情報を変数numと変数minvに代入しています。numには配列の先頭の添え字である0、minvには指定値vと配列の先頭の値(list[0])との差の絶対値を代入しています。

021
022
023
024
025
026
		for ( i = 1; i < list.length; ++ i ) {
			if ( Math.abs( list[ i ] - v ) < minv ) {
				num = i;
				minv = Math.abs( list[ i ] - v );
			}
		}

指定値と配列の2番目の値から最後の値の差を計算していき、その差がminvより小さくなるときにnumとminvの値を変更していきます。

028
		return num;

for文を抜けたときのnumの値が求めたい配列添え字になっています。その値をreturn文で返します。

032
033
	// メイン
	public static void main( String[] args ) {

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

034
035
036
037
038
039
040
041
042
043
		// 結果格納フラグ
		boolean result;
		// ファイル名
		String inname, outname;
		// 画像格納クラス
		BufferedImage img = null;
		// 色リスト
		int[] colorlist;
		// 256色→N色変換テーブル
		int[] converttbl;

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

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

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

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

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

057
058
059
060
061
062
063
064
065
066
067
068
069
070
		// 色リストの作成
		colorlist = new int[ args.length - 2 ];
		for ( int i = 2; i < args.length; ++ i ) {
			// 値をcolorlistに代入
			try {
				// 文字列をintに変換して値を代入
				colorlist[ i - 2 ] = Integer.valueOf( args[ i ] );
			}
			catch( NumberFormatException ne )
			{
				System.out.println( "引数が不正です" );
				return;
			}
		}

与えられた引数の3番目以降がN値化するための値です。与えられた値の個数は引数から2を引いたもので、この個数で配列colorlistを作成し、値を代入しています。

072
073
074
075
076
077
078
079
080
		// JPEG画像の読み込み
		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メソッドを抜けるようにしています。

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

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

BufferedImage.getTypeメソッド

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

  パラメータ なし

   戻り値     BufferedImage のイメージ型を返します。
090
091
092
093
094
095
		// N値に変換
		int x, y;
		int width, height;
		int color, r, g, b;
		int p;
		int newcolor;

N値変換で使う変数を宣言しています。

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

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

101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
		// 256色→N色変換テーブル作成
		converttbl = new int[ 256 ];
		for ( int i = 0; i < 256; ++ i ) {
			// iに一番近い値を取得
			int num = getNearestValue( colorlist, i );
			if ( 0 <= num ) {
				// 配列の添え字numから色を取得
				p = colorlist[ num ];
				// pを0~255の範囲に変換
				if ( 0 > p )
					p = 0;
				else {
					if ( 255 < p ) p = 255;
				}
				converttbl[ i ] = p;
			}
		}

0~255の値に最も近い値をgetNearestValueメソッドで求め、その値を配列converttblに格納しています。値が0未満になった場合には0を、255を超えた場合は255を格納しています。

119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
		// N色変換
		for ( y = 0; y < height; ++ y ) {
			for ( x = 0; x < width; ++ x ) {
				// (x,y)の色を取得
				color = img.getRGB( x, y );

				// 色をr,g,bに分解
				r = ( color >> 16 ) & 0xff;
				g = ( color >> 8 ) & 0xff;
				b = color & 0xff;

				// rgbに重みを付けてpを計算
				p =  (int)( 0.29891 *  (double)r
					 + 0.58661 *  (double)g
						 + 0.11448 *  (double)b );

				// 256色→N色変換テーブルから変換色を取得
				p = converttbl[ p ];

				// r,g,bにpを代入
				r = p;
				g = p;
				b = p;

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

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

画像の中の全てのピクセルの座標を参照するループをつくり、その座標色情報を取得しています。具体的には、変数yを0~height-1、変数xを0~width-1に変化させながら計算したグレー値をカラーテーブル(配列converttbl)で変換しています。

BufferedImage.getRGBメソッド

public static int getRGB( int x, int y )
・(x,y)で指定した画像座標の色情報を取得します。

  パラメータ x : 画像のx座標(単位ピクセル)
        y : 画像のy座標(単位ピクセル)

   戻り値      (x,y)の色情報
151
152
153
154
155
156
157
158
159
		// PNG画像の保存
		try {
			// imgをoutname(出力PNG)に保存
			result = ImageIO.write( img, "png", new File( outname ) );
		} catch ( Exception e ) {
			// outname(出力JPEG)の保存に失敗したときの処理
			e.printStackTrace();
			return;
		}

BufferedImageクラスのimgのメモリ内のデータを、出力PNG名の変数(outname)に格納されているファイル名で保存します。この場合は、PNGファイル名が不正であったり、保存先の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を返します。
161
162
		// 正常に終了
		System.out.println( "正常に終了しました" );

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

以上です。

関連コンテンツ

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

2015.11.29

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

2020.03.23

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

2015.12.22

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

2022.07.07

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

2022.09.10

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

2022.10.17

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

2023.03.20

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

2020.03.20

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

2020.03.23

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

2022.12.13

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

2020.03.23

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

2020.03.23

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

2020.03.23

画像って何?

2022.07.25

NTSC 加重平均法を使ったグレースケール変換を紹介しています。Javaのソースコードを使って詳しく解説しています。

2016.01.26

カラーの画像を白・灰色・黒だけで表現した画像に変換する方法を解説しています。Javaのソースコード付きです。

2020.03.23

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

2015.11.29

2値化は、画像処理の1つの方法で、カラー画像を2つ色だけで表現する画像に変換することです。この画像処理をJavaで作ってみませんか?

2016.01.21

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

2020.03.23

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

2022.08.03

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

2020.03.23

同じ型の変数(データ)を複数個まとめて管理するデータの持ちかたがあります。これが配列です。くわしくは、記事をご覧ください。

2016.01.14

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

2022.08.29

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

2020.03.23

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

2020.03.23

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

2020.03.23

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

2016.03.02

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

2020.03.23

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

2016.11.23

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

2022.07.10

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

2022.07.14

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

2020.03.23

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

2020.03.23

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

2021.05.18

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

2021.05.18

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

2020.03.23

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

2022.10.25

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

2016.12.16

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

2022.07.27

広告