我觉得我像往常一样在这里缺少一些简单的东西。
我正在尝试使用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正如文档建议的那样,在读取标头并尝试使用DataInputStream读取原始字节后,仅结果为EOFException 。 仍在寻找解决方案...