[英]Java Image Scaling Quality Lost in JAR File
我正在编写一个程序,该程序具有将整个JPanel
放大并以JScrollPane
作为地图的功能,但我也想为用户提供JScrollPane
角落中整个地图的缩放视图和最小化视图。 我正在使用BlueJ,&(至少在BlueJ中),我基本上已经得到了想要的东西。 为了缩小地图,我创建了一个BufferedImage
bi
,即迷你地图的大小,然后获取其Graphics
对象g
,将其转换为Graphics2D
对象g2
,然后为g2
一组RenderingHints
,通过:
g2.setRenderingHints(new RenderingHints(RenderingHints.KEY_ANTIALIASING,
RenderingHints.VALUE_ANTIALIAS_ON));
从那里,我在g2
上使用Graphics.scale(double sx, double sy)
方法来设置比例,以便在调用以g2
为参数的大地图的paint(Graphics g)
方法时,它将绘制缩放比例的版本bi
上的大地图的,即小地图的确切大小。 每秒发生20次(某种程度的随意性,以该速度,我认为它看起来最好。迷你地图反映了用户实时执行的操作,如果每秒执行少于20次,我认为它看起来很懒。)
在我的Macbook Pro上,此代码的最终外观为:
当我运行程序时,此代码没有明显的滞后,这导致我以每秒20次的速度保持图片重新采样,因为这样做似乎有很多CPU。 但是,将我的项目导出到JAR文件后,结果如下:
程序运行几乎没有延迟,但是到底发生了什么? 只要我在BlueJ中运行代码,它就可以工作-但是在JAR文件中,迷你地图看起来很可怕。
为了澄清起见,这是要缩放的地图(具有讽刺意味的是,在我将其放置在本文中时,它也在缩放): 这是两个并排的BlueJ和JAR文件迷你地图:
这可能是什么原因? 当我运行“ java -version”时,输出为:
Java版本“ 1.8.0_31”
Java(TM)SE运行时环境(内部版本1.8.0_31-b13)
Java HotSpot(TM)64位服务器VM(内部版本25.31-b07,混合模式)
TL;博士:
当我在BlueJ中运行代码时,小图看起来不错。 导出到JAR文件后,它看起来很糟。 是什么原因造成的?
编辑:我现在包括一个SSCCE,以便更好地理解我的意思。
import javax.swing.*;
import java.awt.*;
import java.awt.image.*;
import javax.swing.border.*;
public class ScaleTest
{
int scrollSpeed = 16, imageSize = 200;
int rows = 40, columns = 40;
JFrame f1 = new JFrame("ScaleTest: Full Size");
JScrollPane scrollPane = new JScrollPane();
JPanel f1Panel = new JPanel(new GridLayout(rows,columns,1,1));
JFrame f2 = new JFrame("ScaleTest: Scaled");
JPanel f2Panel = new JPanel()
{
public void paintComponent(Graphics g)
{
Graphics2D g2 = (Graphics2D)g;
g2.drawImage(bi,0,0,null);
}
};
BufferedImage bi = new BufferedImage(imageSize,imageSize,BufferedImage.TYPE_INT_ARGB);
public ScaleTest()
{
f1.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f1.add(scrollPane);
scrollPane.setPreferredSize(new Dimension(600,600));
scrollPane.getHorizontalScrollBar().setUnitIncrement(scrollSpeed);
scrollPane.getVerticalScrollBar().setUnitIncrement(scrollSpeed);
scrollPane.setViewportView(f1Panel);
for (int i = 0; i < rows*columns; i++)
{
JPanel p = new JPanel();
p.setBackground(Color.WHITE);
p.setBorder(BorderFactory.createLineBorder(Color.BLACK,1));
p.setPreferredSize(new Dimension(100,100));
f1Panel.add(p);
}
f1.pack();
f2.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f2.setResizable(false);
f2.add(f2Panel);
f2Panel.setPreferredSize(new Dimension(imageSize,imageSize));
f2.pack();
Graphics2D g2 = (Graphics2D)bi.getGraphics();
g2.setRenderingHints(new RenderingHints(RenderingHints.KEY_ANTIALIASING,
RenderingHints.VALUE_ANTIALIAS_ON));
g2.scale(((double)imageSize)/f1Panel.getWidth(),((double)imageSize)/f1Panel.getHeight());
f1Panel.paint(g2);
}
public static void main(String[] args)
{
EventQueue.invokeLater(new Runnable()
{
public void run()
{
createAndShowGUI();
}
});
}
public static void createAndShowGUI()
{
ScaleTest s = new ScaleTest();
s.f1.setVisible(true);
s.f2.setVisible(true);
}
}
在BlueJ中,生成的缩放图像是这样的:
但是,在从BlueJ生成的JAR文件中,运行JAR会生成以下图像:
可能会发生什么?
我猜您的BlueJ使用的是JDK1.6.0。
我在SE上找到了这个答案: https : //stackoverflow.com/a/24746194/1695856 。 其结果是,提高按比例缩小的图像质量的一种方法是对其进行模糊处理并分阶段按比例缩小。 在下面的示例中,我将图像按比例缩小了50%三倍,然后缩小了80%,以获得最终比例为原始比例的10%。
引用的文件(map.png)是您先前发布的大图像。 据我所知,设置抗锯齿渲染提示在绘制缩放的图像时没有用,因为它仅对线条,椭圆,矩形等有效,而对图像没有影响。 插值渲染提示会有所帮助,但是依赖于任何渲染提示并不总是可行,因为它们可能无法在不同平台上实现相同(或根本没有实现)。
import java.awt.Dimension; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.awt.image.ConvolveOp; import java.awt.image.Kernel; import java.io.File; import java.io.IOException; import javax.imageio.ImageIO; import javax.swing.ImageIcon; import javax.swing.JFrame; import javax.swing.JLabel; import javax.swing.SwingUtilities; public class Rescale { public static void main(String[] args) { SwingUtilities.invokeLater(new Runnable() { private ConvolveOp cop; @Override public void run() { JFrame frame = new JFrame("Map"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); try { BufferedImage im = ImageIO.read(new File("map.png")); cop = new ConvolveOp(new Kernel(3, 3, new float[] { 0f, 1f / 5f, 0f, 1f / 5f, 1f / 5f, 1f / 5f, 0f, 1f / 5f, 0f }), ConvolveOp.EDGE_NO_OP, null); for (int i = 0; i < 3; i++) { im = getScaled(im, 0.5); } im = getScaled(im, 0.8f); JLabel lab = new JLabel(new ImageIcon(im)); frame.setContentPane(lab); frame.pack(); frame.setLocationRelativeTo(null); frame.setVisible(true); } catch (IOException e) { e.printStackTrace(); } } private BufferedImage getScaled(BufferedImage im, double scale) { BufferedImage nim; Dimension dim = new Dimension((int) (im.getWidth() * scale), (int) (im.getHeight() * scale)); nim = new BufferedImage(dim.width, dim.height, BufferedImage.TYPE_INT_RGB); Graphics2D g = (Graphics2D) nim.getGraphics(); g.scale(scale, scale); g.drawImage(im, cop, 0, 0); g.dispose(); return nim; } }); } }
我希望这有帮助。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.