Конвертировать любой файл в PNG на Java
Я хочу преобразовать любой файл в PNG, а также полностью изменить процесс, все на Java.
Я хочу использовать форму int-RGB для изображения, и чтобы байты из файла были байтом в целом числе RGB. Это должно произвести изображение.
Я заставил это работать, только сохраняя байты в красном цвете, но я не могу понять, как также использовать зеленый и синий.
Это код, который я использую в данный момент, который использует только красный, и он отлично работает:
public static void fileToImage(String sourceFile, String imageFile) throws IOException {
DataInputStream dis = new DataInputStream(new FileInputStream(sourceFile));
int size = ((int) Math.sqrt(dis.available())) + 2;
BufferedImage image = new BufferedImage(size,size, BufferedImage.TYPE_INT_RGB);
for (int y = 0; y < size; y++) {
for (int x = 0; x < size; x++) {
int red = dis.read(); // I'm using only red
int green = 0; // default
int blue = 0; // default
int rgb = (0xFF << 24) | ((red & 0xFF) << 16) | ((green & 0xFF) << 8) | (blue & 0xFF);
image.setRGB(x, y, rgb);
}
}
dis.close();
ImageIO.write(image, "png", new File(imageFile));
}
public static void imageToFile(String imageFile, String outputFile) throws IOException {
BufferedImage image = ImageIO.read(new File(imageFile));
DataOutputStream dos = new DataOutputStream(new FileOutputStream(outputFile));
for (int y = 0; y < image.getHeight(); y++) {
for (int x = 0; x < image.getWidth(); x++) {
int rgb = image.getRGB(x, y);
int red = (rgb >> 16) & 0xFF;
int green = (rgb >> 8) & 0xFF;
int blue = rgb & 0xFF;
dos.write(red); // I'm using only red
}
}
dos.close();
}
РЕДАКТИРОВАТЬ: Хорошо, поэтому я изменил код, вот он:
public static void fileToImage(String sourceFile, String imageFile) throws IOException {
DataInputStream dis = new DataInputStream(new FileInputStream(sourceFile));
int size = ((int) Math.sqrt(dis.available())) + 2;
BufferedImage image = new BufferedImage(size,size, BufferedImage.TYPE_INT_RGB);
for (int y = 0; y < size; y++) {
for (int x = 0; x < size; x++) {
int red = dis.read();
int green = dis.read();
int blue = dis.read();
int rgb = (0xFF << 24) | ((red & 0xFF) << 16) | ((green & 0xFF) << 8) | (blue & 0xFF);
image.setRGB(x, y, rgb);
}
}
dis.close();
ImageIO.write(image, "png", new File(imageFile));
}
public static void imageToFile(String imageFile, String outputFile) throws IOException {
BufferedImage image = ImageIO.read(new File(imageFile));
DataOutputStream dos = new DataOutputStream(new FileOutputStream(outputFile));
for (int y = 0; y < image.getHeight(); y++) {
for (int x = 0; x < image.getWidth(); x++) {
int rgb = image.getRGB(x, y);
int red = (rgb >> 16) & 0xFF;
int green = (rgb >> 8) & 0xFF;
int blue = rgb & 0xFF;
dos.write(red);
dos.write(green);
dos.write(blue);
}
}
dos.close();
}
Это работает, но не совсем так, как ожидалось. В произведенном PNG много черного пространства, потому что я считаю, что "размер" изображения неправильный. Из-за этого при переводе PNG обратно в исходный файл он становится намного больше, чем изначально.
РЕДАКТИРОВАТЬ: проблема, с которой я сейчас сталкиваюсь, заключается в следующем: например, если я преобразую текстовый файл со следующим содержимым в PNG, используя метод fileToImage: hello world! Затем я использую imageToFile, чтобы преобразовать его обратно, вывод: hello world!SSSSSSSSSSSSSSS (S обозначает "пробел", их 15)
РЕДАКТИРОВАТЬ: до сих пор не могу понять это. Вот что я использую:
private static final int NAN = -1;
private static int readByte(DataInputStream dis) throws IOException {
int b;
try {
b = dis.readByte();
} catch (EOFException e) {
b = NAN;
}
return b;
}
public static void fileToImage(String sourceFile, String imageFile) throws IOException {
DataInputStream dis = new DataInputStream(new FileInputStream(sourceFile));
int size = ((int) Math.sqrt(dis.available())) + 2;
BufferedImage image = new BufferedImage(size, size, BufferedImage.TYPE_INT_RGB);
for (int y = 0; y < size; y++) {
boolean finished = false;
for (int x = 0; x < size; x++) {
int alpha = 3;
int red = readByte(dis);
int green = readByte(dis);
int blue = readByte(dis);
if (red == NAN) {
alpha--;
red = 0;
}
if (green == NAN) {
alpha--;
green = 0;
}
if (blue == NAN) {
alpha--;
blue = 0;
}
int rgb = ((alpha & 0xFF) << 24) | ((red & 0xFF) << 16) | ((green & 0xFF) << 8) | (blue & 0xFF);
image.setRGB(x, y, rgb);
if (alpha < 3) {
finished = true;
break;
}
}
if (finished) break;
}
dis.close();
ImageIO.write(image, "png", new File(imageFile));
}
public static void imageToFile(String imageFile, String outputFile) throws IOException {
BufferedImage image = ImageIO.read(new File(imageFile));
DataOutputStream dos = new DataOutputStream(new FileOutputStream(outputFile));
for (int y = 0; y < image.getHeight(); y++) {
boolean finished = false;
for (int x = 0; x < image.getWidth(); x++) {
int rgb = image.getRGB(x, y);
int alpha = (rgb >> 24) & 0xFF;
int red = (rgb >> 16) & 0xFF;
int green = (rgb >> 8) & 0xFF;
int blue = rgb & 0xFF;
if (alpha == 0) {
finished = true;
break;
}
if (alpha >= 1) dos.write(red);
if (alpha >= 2) dos.write(green);
if (alpha == 3) dos.write(blue);
}
if (finished) break;
}
dos.close();
}
2 ответа
Я думаю, вам просто нужно немного отрегулировать внутренние петли. Небольшой вспомогательный метод сделает эту вещь проще в использовании, хотя я уверен, что мой эскиз немного уродлив:
int myReadByte(DataInputStream dis) {
int b;
try {
b = dis.readByte():
} catch (EOFException e) {
b = 0;
}
return b;
}
Теперь с этим помощником...
for (int x = 0; x < size; x++) {
int red = myReadByte(dis);
int green = myReadByte(dis);
int blue = myReadByte(dis);
int rgb = (0xFF << 24) | ((red & 0xFF) << 16) | ((green & 0xFF) << 8) | (blue & 0xFF);
image.setRGB(x, y, rgb);
а также
for (int x = 0; x < image.getWidth(); x++) {
int rgb = image.getRGB(x, y);
int red = (rgb >> 16) & 0xFF;
int green = (rgb >> 8) & 0xFF;
int blue = rgb & 0xFF;
dos.write(red);
dos.write(green);
dos.write(blue);
}
Кажется, что ваш код преобразует 3 байта подряд в один пиксель. из-за этого вы получаете только 1/3 файла с цветом, а остальные 2/3 пусты.
Чтобы это исправить, просто разделите размер файла на 3.
int size = ((int) Math.sqrt(dis.available()/3)) + 2;