javaarraysnullpointerexceptionbufferedimageawt-eventqueue

NullPointerException with BufferedImage Array


I am trying to take an image and turn it into tiles that are laid out in GridLayout, but receive a runtime error.

Here is the code:

public class TileList extends JPanel {

private static final int width = 16;            //width of a tile
private static final int height = width;
private int col = 1;
private int row = 1;

private BufferedImage image;
File tilesetImage = new File("image.png");
BufferedImage tileset[];

public void loadAndSplitImage (File loadImage) {
    try{
        image = ImageIO.read(loadImage);
    }catch(Exception error) {
        System.out.println("Error: cannot read tileset image.");
    }// end try/catch
    col = image.getWidth()/width;
    row = image.getHeight()/height;
    BufferedImage tileset[] = new BufferedImage[col*row];
}// end loadAndSplitImage

public TileList() {
    loadAndSplitImage(tilesetImage);
    setLayout(new GridLayout(row,col,1,1));
    setBackground(Color.black);

    int x=0;
    int y=0;
    for (int i = 0; i < (col*row); i++) {
        JPanel panel = new JPanel();
        tileset[i] = new BufferedImage(width, height, image.getType());  //first error
        tileset[i] = image.getSubimage(x,y,x+width,y+height);
        panel.add(new JLabel(new ImageIcon(tileset[i])));
        add(panel);
        x+=width;
        y+=height;
    }// end for loop
}// end constructor

public static void createAndShowGui() {
    JFrame frame = new JFrame("TileList");
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().add(new TileList());      //second error
    frame.pack();
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
}//end createAndShowGui

public static void main (String [] args) {
    SwingUtilities.invokeLater(new Runnable() {
        public void run () {
            createAndShowGui();           //third error
        }// end run
    });// end invokeLater
  }// end main
}// end class

This is the error I receive:

Exception in thread "AWT-EventQueue-0" java.lang.NullPointerException
    at TileList.<init>(TileList.java:55)
    at TileList.createAndShowGui(TileList.java:73)
    at TileList$1.run(TileList.java:82)
    at java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:251)
    at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:721)
    at java.awt.EventQueue.access$200(EventQueue.java:103)
    at java.awt.EventQueue$3.run(EventQueue.java:682)
    at java.awt.EventQueue$3.run(EventQueue.java:680)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDo
main.java:76)
    at java.awt.EventQueue.dispatchEvent(EventQueue.java:691)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThre
ad.java:244)
    at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.
java:163)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThre
ad.java:151)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:147)

    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:139)

    at java.awt.EventDispatchThread.run(EventDispatchThread.java:97)

What does this error mean and how did I write the code wrong?

Edit: I changed my code as Hovercraft suggested, however now I have new errors:

tileset[] = new BufferedImage[col*row];
       ^                           error: not a statement

tileset[] = new BufferedImage[col*row];
         ^                         error: ';' expected

tileset[] = new BufferedImage[col*row];
                             ^     error: not a statement

Solution

  • Your error is that you are shadowing a variable -- you're re-declaring the variable tileSet inside constructor when it was already declared in the class. Thus the tileSet variable declared in the class is never initialized since only the local variable, the one declared in the constructor is initialized.

    The solution: Only declare tileSet once, in the class.

    Thus, don't do this:

    public class TileList extends JPanel {
    
      //.... deleted for brevity
    
      BufferedImage tileset[];
    
      public void loadAndSplitImage (File loadImage) {
        try{
            image = ImageIO.read(loadImage);
        }catch(Exception error) {
            System.out.println("Error: cannot read tileset image.");
        }// end try/catch
        col = image.getWidth()/width;
        row = image.getHeight()/height;
        BufferedImage tileset[] = new BufferedImage[col*row]; // *** re-declaring variable!
      }
    

    but instead do this:

    public class TileList extends JPanel {
    
      //.... deleted for brevity
    
      BufferedImage tileset[];
    
      public void loadAndSplitImage (File loadImage) {
        try{
            image = ImageIO.read(loadImage);
        }catch(Exception error) {
            System.out.println("Error: cannot read tileset image.");
        }// end try/catch
        col = image.getWidth()/width;
        row = image.getHeight()/height;
    
        // ************
        // BufferedImage tileset[] = new BufferedImage[col*row]; // *****
        tileset[] = new BufferedImage[col*row]; // **** note the difference? ****
      }
    

    Note that even more important than the solution for this isolated problem is to gain an understanding on how to solve NPE (NullPointerExceptions). The key is to check all the variables on the line throwing the exception, check to see which one(s) is null, and then look back to see where one of them is not being initialized properly before you try using it.