質問
ここで(いつものように)簡単なものが足りないように感じます。
Javaを使用してPGM画像を読み込もうとしています。 Matlabはそれを正常に行います - Matlabの画像ピクセル(たとえば、小さな32x32画像)を出力すると、次のようなものが与えられます。
1 0 11 49 94 118 118 106 95 88 85 96 124 143 142 133
しかし、私のJavaリーダーはこれを出力します:
1 0 11 49 94 118 118 106 95 88 85 96 124 65533 65533 65533
127を超えるピクセル値は65533で満たされているように見えますが、いくつかのランダム値が正しくありません。また、ほぼ全列全体を-1の値に割り当てます。
これが私が使用しているコードです:
filePath = 'imagepath.pgm'; FileInputStream fileInputStream = new FileInputStream(filePath); BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(fileInputStream)); // read the header information ... int [][] data2D = new int [picWidth] [picHeight]; for (int row = 0; row < picHeight; row++) { for (int col = 0; col < picWidth; col++) { data2D[row][col] = bufferedReader.read(); System.out.print(data2D[row][col] + " "); } System.out.println(); } fileInputStream.close();
どんなアイデアも大歓迎です。
編集 署名されていないPGM値は次のとおりです。
1 0 11 49 94 118 118 106 95 88 85 96 124 143 142 133 30 26 29 57 96 122 125 114 102 94 91 101 127 146 145 136 96 85 70 75 101 128 136 126 111 106 106 112 131 149 153 147 163 147 114 93 99 120 132 123 110 113 124 129 137 154 166 168 215 195 149 105 88 99 114 111 106 123 148 158 160 174 191 197 245 224 173 115 81 82 100 109 117 144 179 194 194 205 222 230 235 217 170 115 78 78 113 117 100 83 80 212 214 226 244 253 178 167 135 93 68 78 123 129 106 77 69 202 204 222 244 255 114 110 92 64 54 81 107 105 83 59 56 182 184 201 222 231 79 80 71 52 55 97 67 55 41 33 42 184 179 181 185 183 62 66 65 52 63 115 29 16 12 17 30 209 197 174 150 132 40 47 52 44 55 109 171 196 188 186 208 229 218 179 136 107 31 38 44 37 43 89 145 167 158 159 191 223 219 179 133 105 48 52 56 51 57 91 128 133 117 120 157 196 200 168 128 105 64 67 70 73 87 114 127 107 79 81 118 159 173 154 123 104 63 67 73 83 107 132 129 91 54 54 88 130 153 146 123 106
ヘッダーは次のようになります:
P5 # MatLab PGMWRITE file, saved 27-Jun-2002 16 16 255
編集#2
以下の概念実証コードへの完全な出力は次のとおりです。
Skipping unknow token: "" Skipping unknow token: "1^vvj_XU`|���" Skipping unknow token: "" Skipping unknow token: "9`z}rf^[e���`UFKe��~ojjp������r]cx�{nq|������ÕiXcroj{��������sQRdmu��������٪sNNqudSP�����]DN{�jME�����rn\@6QkiS;8�����OPG47aC7)!*�����>BA4?s" Skipping unknow token: "" Skipping unknow token: "" Skipping unknow token: "�Ů��(/4,7m�ļ���ڳ�k" Skipping unknow token: "&,%+Y������۳�i04839[��ux��Ȩ�i@CFIWrkOQv���{h?CISk��[66X���{j" Exception in thread "main" java.util.NoSuchElementException at java.util.Scanner.throwFor(Scanner.java:838) at java.util.Scanner.next(Scanner.java:1347) at Test.main(Test.java:49)
スローされた例外で言及されている49行目は次のとおりです。
System.out.println(String.format("Skipping unknow token: \"%s\"", scan.next()));
問題は、これらの画像ファイルがASCIIテキスト/数字とバイナリ画像データの両方で構成されているという事実と関係があると確信しています。しかし、JavaにPNGを読むのに問題がない場合、なぜPGMのサポートがないのでしょうか?
編集3
わかりました、私は機能する実装を見つけました...残念ながら、それは非難されています:
filePath = "imagepath.pgm"
FileInputStream fileInputStream = new FileInputStream(filePath);
DataInputStream dis = new DataInputStream(fileInputStream);
StreamTokenizer streamTokenizer = new StreamTokenizer(dis);
// read header text using StreamTokenizer.nextToken()
data2D = new int [picWidth] [picHeight];
for (int row = 0; row < picHeight; row++) {
for (int col = 0; col < picWidth; col++) {
data2D[row][col] = dis.readUnsignedByte();
System.out.print(data2D[row][col] + " ");
}
System.out.println();
}
Javaのドキュメントによると、 StreamTokenizer(InputStream)
コンストラクターは非推奨です DataInputStream.readLine()
メソッドは、生のバイトを文字に正しく変換しません。ただし、ヘッダー上のこの特定のケースでは機能しているようであり、明らかにその後のバイナリ画像データで機能します。
残念ながら、それはまだ非難されており、 BufferedReader
ドキュメントが示唆しているように、結果のみが生じます EOFException
sヘッダーを読んで、使用しようとした後 DataInputStream
生のバイトを読む。まだ解決策を探しています...
解決
コードの問題は、ファイルから生データを読み取るために間違ったクラスを使用していることです。として BufferedReader
ドキュメントは言っています:
public int read() throws IOException
単一の文字を読みます。
返品:キャラクターは、0〜65535(0x00-0xffff)の範囲の整数として、またはストリームの終了に達した場合に読み取ります。
したがって、各呼び出し read()
の方法 BufferedReader
実際、入力ストリームから1つまたは2つのバイト(文字エンコードに基づいて)を消費しますが、これは必要なものではありません。これは、なぜあなたが多くの-1を得るのかを説明しています:あなたが思っていたよりもはるかに早くストリームが終了しました。
PGMにはASCII 10進数として値が含まれているため、使用して解析するのは簡単です スキャナー クラス。
これが次のとおりです ほとんどテストされていません PGM画像を読み取る方法を示すコード:
- マジック番号の後に1つのコメントが含まれています(つまり、2番目のコメントを除く#から始まる行がありません)
- PGMファイルの長さはちょうど4行です。
これがコードです:
String filePath = "image.pgm";
fileInputStream = new FileInputStream(filePath);
Scanner scan = new Scanner(fileInputStream);
// Discard the magic number
scan.nextLine();
// Discard the comment line
scan.nextLine();
// Read pic width, height and max value
int picWidth = scan.nextInt();
int picHeight = scan.nextInt();
int maxvalue = scan.nextInt();
fileInputStream.close();
// Now parse the file as binary data
fileInputStream = new FileInputStream(filePath);
DataInputStream dis = new DataInputStream(fileInputStream);
// look for 4 lines (i.e.: the header) and discard them
int numnewlines = 4;
while (numnewlines > 0) {
char c;
do {
c = (char)(dis.readUnsignedByte());
} while (c != '\n');
numnewlines--;
}
// read the image data
int[][] data2D = new int[picHeight][picWidth];
for (int row = 0; row < picHeight; row++) {
for (int col = 0; col < picWidth; col++) {
data2D[row][col] = dis.readUnsignedByte();
System.out.print(data2D[row][col] + " ");
}
System.out.println();
}
実装する必要があります:コメント行のサポート、各要素の値は、 maxvalue
, 、不正なファイルのエラーチェック、例外処理。 Unixの終わりのエンドオブラインを使用してPGMファイルでテストしましたが、Windowsでも動作するはずです。
それを強調させてください これは、PGMパーサーの堅牢性や完全な実装ではありません. 。このコードは、お客様のニーズに合わせて十分に達成する可能性がある概念実証として意図されています。
堅牢なPGMパーサーが本当に必要な場合は、によって提供されるツールを使用できます。 netpbm.