Java AWT Robot:为什么较小的屏幕捕获不等于整个屏幕捕获的子图像?

6uxekuva  于 2024-01-05  发布在  Java
关注(0)|答案(1)|浏览(159)

我在使用java.awt.Robot时遇到了一些奇怪的行为。如果我用它来拍摄两个屏幕截图,一个是整个屏幕,另一个只是屏幕的一个部分,屏幕的一个部分与原始屏幕截图的一个子图像具有不同的像素,与子图像具有相同的坐标。下面是我的代码:

  1. import java.awt.AWTException;
  2. import java.awt.Dimension;
  3. import java.awt.Rectangle;
  4. import java.awt.Robot;
  5. import java.awt.Toolkit;
  6. import java.awt.image.BufferedImage;
  7. import java.io.File;
  8. import java.io.IOException;
  9. import javax.imageio.ImageIO;
  10. public class ScreenCapTest {
  11. public static void main(String[] args) throws AWTException, IOException {
  12. // Construct Robot
  13. Robot r = new Robot();
  14. // Get dimensions of screen
  15. Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
  16. // Take screenshot
  17. BufferedImage screen = r.createScreenCapture(new Rectangle(0,0, (int)screenSize.getWidth(), (int)screenSize.getHeight()));
  18. // Take screenshot of small section of screen
  19. int x = 5;
  20. int y = 5;
  21. int w = 50;
  22. int h = 50;
  23. BufferedImage subscreen = r.createScreenCapture(new Rectangle(x,y,w,h));
  24. // Create a subimage of the same section of the screen from the original screenshot
  25. BufferedImage subimageOfScreen = screen.getSubimage(x,y,w,h);
  26. // Are they equal?
  27. System.out.println(imgEqual(subimageOfScreen, subscreen));
  28. // Output images for comparison
  29. ImageIO.write(subimageOfScreen, "png", new File("subimage.png"));
  30. ImageIO.write(subscreen, "png", new File("subscreen.png"));
  31. }
  32. public static boolean imgEqual(BufferedImage image1, BufferedImage image2) {
  33. int width;
  34. int height;
  35. boolean imagesEqual = true;
  36. if( image1.getWidth() == ( width = image2.getWidth() ) &&
  37. image1.getHeight() == ( height = image2.getHeight() ) ){
  38. for(int x = 0;imagesEqual == true && x < width; x++){
  39. for(int y = 0;imagesEqual == true && y < height; y++){
  40. if( image1.getRGB(x, y) != image2.getRGB(x, y) ){
  41. imagesEqual = false;
  42. }
  43. }
  44. }
  45. } else {
  46. imagesEqual = false;
  47. }
  48. return imagesEqual;
  49. }
  50. }

字符串
大多数情况下,它会报告false,这意味着从(5,5)到(5,5,5,5)的完整屏幕截图的子图像与从(5,5)到(5,5,5,5)的屏幕截图不同。奇怪的是,对于xywh的某些值,它会打印true
下面是一对BufferedImages的示例,代码将其打印为false。我可以看到它们略有不同,但我不明白为什么会存在这种行为。这是怎么回事?
完整屏幕截图的子图像:x1c 0d1x
屏幕同一部分的屏幕截图:

j7dteeu8

j7dteeu81#

修改你的代码一点。我不喜欢(原始)代码不断地创建一个屏幕图像的子图像来与实际的子图像进行比较。仍然没有帮助找到匹配。
我的下一步是从原始屏幕图像创建子图像。这是一种测试问题是否与逻辑或子图像有关的方法。在这种情况下,子图像匹配。
然后我回去测试两个机器人创建的图像,并注意到具有较小宽度/高度的子图像将匹配。
进一步测试和宽度似乎不会影响子图像的匹配。
然而,我注意到,高度确实造成了一个问题。在我的情况下,高度需要比屏幕的高度小31像素。巧合的是,31像素是我的Windows机器上的屏幕的高度。所以我的想法是,不知何故,当使用机器人的第二个图像没有正确创建。不知道为什么。
下面是我用来测试的代码:

  1. import java.awt.AWTException;
  2. import java.awt.Dimension;
  3. import java.awt.Point;
  4. import java.awt.Rectangle;
  5. import java.awt.Robot;
  6. import java.awt.Toolkit;
  7. import java.awt.image.BufferedImage;
  8. public class ScreenCapTest2 {
  9. public static void main(String[] args) throws AWTException {
  10. Robot r = new Robot();
  11. Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
  12. BufferedImage screen = r.createScreenCapture(new Rectangle(0,0, (int)screenSize.getWidth(), (int)screenSize.getHeight()));
  13. int xOffset = 20;
  14. int yOffset = 50;
  15. BufferedImage subscreen = screen.getSubimage(xOffset, yOffset, (int)screenSize.getWidth()-xOffset, (int)screenSize.getHeight()-yOffset);
  16. // BufferedImage subscreen = r.createScreenCapture(new Rectangle(xOffset, yOffset, (int)screenSize.getWidth()-xOffset, (int)screenSize.getHeight()-yOffset));
  17. // BufferedImage subscreen = r.createScreenCapture( new Rectangle(xOffset, 49, 1900, 1000) );
  18. System.out.println( screenSize );
  19. System.out.println(indexOfSubImage(screen, subscreen));
  20. }
  21. public static Point indexOfSubImage(BufferedImage img, BufferedImage subimg) {
  22. for (int x=0; x <= img.getWidth() - subimg.getWidth(); x++) {
  23. for (int y=0; y <= img.getHeight() - subimg.getHeight(); y++) {
  24. // if (imgEqual(img.getSubimage(x, y, subimg.getWidth(), subimg.getHeight()), subimg)) {
  25. if (imgEqual(x, y, img, subimg)) {
  26. return new Point(x,y);
  27. }
  28. }
  29. }
  30. return new Point(-1,-1);
  31. }
  32. public static boolean imgEqual(int offsetX, int offsetY, BufferedImage image1, BufferedImage image2) {
  33. for(int x = 0; x < image2.getWidth(); x++){
  34. for(int y = 0; y < image2.getHeight(); y++){
  35. if( image1.getRGB(x + offsetX, y + offsetY) != image2.getRGB(x, y) ){
  36. return false;
  37. }
  38. }
  39. }
  40. return true;
  41. }
  42. }

字符串
第一个子图像在不同的测试用例中对我有效。
第二个子图像仅在使用(0,0)作为偏移量时有效。
第三个子图像似乎适用于所有“y”偏移量高达49。
编辑:
使用你的新代码,我尝试自动化测试:

  1. import java.awt.AWTException;
  2. import java.awt.Dimension;
  3. import java.awt.Rectangle;
  4. import java.awt.Robot;
  5. import java.awt.Toolkit;
  6. import java.awt.image.BufferedImage;
  7. import java.io.File;
  8. import java.io.IOException;
  9. import javax.imageio.ImageIO;
  10. public class ScreenCapTest3{
  11. public static void main(String[] args) throws AWTException, IOException {
  12. for (int i = 0; i < 1920 - 50; i += 50)
  13. test( i );
  14. }
  15. public static void test(int xOffset) throws AWTException
  16. {
  17. // Construct Robot
  18. Robot r = new Robot();
  19. // Get dimensions of screen
  20. Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
  21. // Take screenshot
  22. BufferedImage screen = r.createScreenCapture(new Rectangle(0,0, (int)screenSize.getWidth(), (int)screenSize.getHeight()));
  23. // Take screenshot of small section of screen
  24. int x = xOffset;
  25. int y = 50;
  26. // int w = 1920 - xOffset;
  27. // int h = 1000;
  28. int w = 50;
  29. int h = 500;
  30. BufferedImage subscreen = r.createScreenCapture(new Rectangle(x,y,w,h));
  31. // Create a subimage of the same section of the screen from the original screenshot
  32. BufferedImage subimageOfScreen = screen.getSubimage(x,y,w,h);
  33. // Are they equal?
  34. System.out.println(xOffset + " : " + imgEqual(subimageOfScreen, subscreen));
  35. // Output images for comparison
  36. // ImageIO.write(subimageOfScreen, "png", new File("subimage.png"));
  37. // ImageIO.write(subscreen, "png", new File("subscreen.png"));
  38. }
  39. public static boolean imgEqual(BufferedImage image1, BufferedImage image2) {
  40. int width;
  41. int height;
  42. boolean imagesEqual = true;
  43. if( image1.getWidth() == ( width = image2.getWidth() ) &&
  44. image1.getHeight() == ( height = image2.getHeight() ) ){
  45. for(int x = 0;imagesEqual == true && x < width; x++){
  46. for(int y = 0;imagesEqual == true && y < height; y++){
  47. if( image1.getRGB(x, y) != image2.getRGB(x, y) ){
  48. // System.out.println(x + " : " + y);
  49. imagesEqual = false;
  50. }
  51. }
  52. }
  53. } else {
  54. imagesEqual = false;
  55. }
  56. return imagesEqual;
  57. }
  58. }


我在命令提示符下使用:java ScreenCapTest3.java运行代码,得到了true/false的混合结果。
然后运行代码:java ScreenCapTest.java > out.txt
当我看out.txt时,一切都是真的。
所以在我看来,子图像中有一个微妙的变化,因为System.out.println(...)在捕获第二个图像之前显示在屏幕上?
另一个编辑:
创建了一个简单的框架来覆盖桌面。我用左边的按钮和中间的大图像来打破一个空框架的白色空间。
当你点击按钮时,我使用了一个计时器来延迟屏幕捕获过程,以允许按钮被绘制回原始状态。

  1. import java.awt.*;
  2. import java.awt.event.*;
  3. import java.awt.image.*;
  4. import javax.swing.*;
  5. import javax.swing.event.*;
  6. public class ScreenTest extends JPanel
  7. {
  8. JList<String> list = new JList<>();
  9. DefaultListModel<String> model = new DefaultListModel<>();
  10. public ScreenTest()
  11. {
  12. setLayout( new BorderLayout() );
  13. JButton create = new JButton("Test Sub Image");
  14. add(create, BorderLayout.PAGE_START);
  15. JPanel west = new JPanel( new GridLayout(50, 5) );
  16. for (int i = 0; i < 250; i++)
  17. west.add( new JButton("just for decoration") );
  18. add(west, BorderLayout.LINE_START);
  19. JLabel center = new JLabel( new ImageIcon("grass.jpg") );
  20. add(center);
  21. add(new JScrollPane(list), BorderLayout.LINE_END);
  22. create.addActionListener((e) -> startTest());
  23. }
  24. private void startTest()
  25. {
  26. ActionListener al = new ActionListener()
  27. {
  28. public void actionPerformed(ActionEvent e)
  29. {
  30. try
  31. {
  32. for (int i = 0; i < 1920 - 500; i += 50)
  33. test( i );
  34. list.setModel( model );
  35. }
  36. catch (Exception e2) { System.out.println(e2); }
  37. }
  38. };
  39. Timer timer = new Timer(1000, al);
  40. timer.setRepeats( false );
  41. timer.start();
  42. }
  43. private void test(int xOffset) throws AWTException
  44. {
  45. try
  46. {
  47. // Construct Robot
  48. Robot r = new Robot();
  49. // Get dimensions of screen
  50. Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
  51. // Take screenshot
  52. BufferedImage screen = r.createScreenCapture(new Rectangle(0,0, (int)screenSize.getWidth(), (int)screenSize.getHeight()));
  53. // Take screenshot of small section of screen
  54. int x = xOffset;
  55. int y = 50;
  56. int w = 500;
  57. int h = 500;
  58. BufferedImage subscreen = r.createScreenCapture(new Rectangle(x,y,w,h));
  59. // Create a subimage of the same section of the screen from the original screenshot
  60. BufferedImage subimageOfScreen = screen.getSubimage(x,y,w,h);
  61. // Are they equal?
  62. model.addElement(xOffset + " : " + imgEqual(subimageOfScreen, subscreen));
  63. }
  64. catch (Exception e) { System.out.println(e); }
  65. }
  66. private boolean imgEqual(BufferedImage image1, BufferedImage image2)
  67. {
  68. int width;
  69. int height;
  70. boolean imagesEqual = true;
  71. if( image1.getWidth() == ( width = image2.getWidth() ) &&
  72. image1.getHeight() == ( height = image2.getHeight() ) ){
  73. for(int x = 0;imagesEqual == true && x < width; x++){
  74. for(int y = 0;imagesEqual == true && y < height; y++){
  75. if( image1.getRGB(x, y) != image2.getRGB(x, y) ){
  76. // System.out.println(x + " : " + y);
  77. imagesEqual = false;
  78. }
  79. }
  80. }
  81. } else {
  82. imagesEqual = false;
  83. }
  84. return imagesEqual;
  85. }
  86. private static void createAndShowGUI()
  87. {
  88. JFrame frame = new JFrame("SSCCE");
  89. frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  90. frame.add(new ScreenTest());
  91. frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
  92. frame.setExtendedState(JFrame.MAXIMIZED_BOTH);
  93. frame.pack();
  94. frame.setLocationByPlatform( true );
  95. frame.setVisible( true );
  96. }
  97. public static void main(String[] args)
  98. {
  99. java.awt.EventQueue.invokeLater( () -> createAndShowGUI() );
  100. }
  101. }

展开查看全部

相关问题