Ошибка переполнения стека с ArrayLists - PullRequest
2 голосов
/ 15 апреля 2011

Я пытаюсь написать код для просмотра файла изображения для групп пикселей одного цвета.

Способ, которым я делаю это, состоит в том, чтобы перебирать изображение (в виде 1-мерного целочисленного массива с хэш-кодом цвета) по пикселям, чтобы найти пиксель цвета, который я ищу. Как только один из них найден, я делаю dfs, чтобы найти смежные пиксели того же цвета и добавить их к новому объекту, который я назвал Blob. Я использую логический массив, чтобы отслеживать, какие пиксели уже были добавлены, поэтому я не добавляю идентичные BLOB-объекты.

Я использую ArrayList для каждого объекта Blob, чтобы отслеживать количество пикселей. А потом я использую другой ArrayList Blobs для хранения разных групп.

Когда я пытаюсь выполнить это на простом примере, изображении с верхней половиной белого и нижней половиной снизу, я получаю ошибку переполнения стека, когда пытаюсь использовать слишком большое изображение. В частности, когда я пытаюсь сделать это с изображением 320x240, я получаю переполнение стека, как только 2752 пикселя добавляются в BLOB-объект.

Я просто не использую правильную структуру данных для того, что я хочу сделать? Я читал, что ArrayLists может хранить объекты Integer.maxValue в них.

Мой код вставлен ниже. Любая помощь с благодарностью.

//blobfind tests code to find similar pixels of a minimum size and groups them together for analysis later  
//purpose is to identify color coded objects through the webcam  

//util for ArrayList  
import java.util.*;  
import java.awt.Color;  
import java.io.*;  

public class Blobfind2 {  

  //width and height of image in pixels  
  private int width;  
  private int height;  
  //hash code for the color being searched for  
  private int colorCode;  
  //minimum blob size to be added  
  private int minPixels;  
  //image in form of array of hashcodes for each pixel  
  private int[] img;  
  //keeping track of which pixels have been added to a blob  
  private boolean[] added;  
  //keeping track of which pixels have been visited when looking for a new blob  
  private boolean[] visited;  

  //debugging variable  
  private int count;  

  public Blobfind2(int inwidth, int inheight, int inCCode, int inminPixels, int[] inimage)   {
    width = inwidth;  
    height = inheight;  
    colorCode = inCCode;  
    minPixels = inminPixels;  
    img = inimage;  
    count = 0;  
  }    

  //takes hashCodeof color, minimum pixel number, and an image in the form of integer array  
  public ArrayList findColor() {  
    //makes an arraylist of "blobs"  
    ArrayList bloblist = new ArrayList();  
    //keep track of which pixels have been added to a blob  
    boolean[] added = new boolean[width * height];  
    //checks through each pixel  
    for (int i = 0; i < img.length; i++) {  
      //if it matches and is not part of a blob, we run dfs to collect all the pixels in   that blob
      if ((img[i] == colorCode) && (added[i] == false)) {  
        //visited keeps track of which pixels in the blob have been visited  
        //refreshed each time a new blob is made  
        boolean[] visited = new boolean[width*height];  
        Blob currBlob = new Blob();  
        dfs(img, currBlob, i, Color.white.hashCode(), added, visited);  
        //adds the blob to the bloblist if it is of a certain size  
        if (currBlob.mass() >= minPixels) {  
          bloblist.add(currBlob);                          
        }  
      }  
    }  
    return bloblist;  
  }  

  //recursive algorithm to find other members of a blob  
  public void dfs (int[] img, Blob blob, int currPixel, int colorCode, boolean[] added,    boolean[] visited) {  
    //System.out.print(currPixel + " - " + count + " ");   
    count++;  
    //check current pixel, this only happens on the first pixel  
    if (visited[currPixel] == false) {  
      blob.add(img[currPixel]);  
      added[currPixel] = true;  
      visited[currPixel] = true;  
    }  
    //checks down pixel  
    if ((currPixel + width < height*width) && (visited[currPixel + width] == false)) {  
      if (img[currPixel + width] == colorCode) {  
        blob.add(img[currPixel + width]);  
        currPixel = currPixel + width;  
        added[currPixel] = true;  
        visited[currPixel] = true;  
        dfs(img, blob, currPixel, colorCode, added, visited);  
      }  
    }  
    //checks up pixel  
    if ((currPixel - width > 0) && (visited[currPixel - width] == false)) {  
      if (img[currPixel - width] == colorCode) {  
        blob.add(img[currPixel - width]);  
        currPixel = currPixel - width;  
        added[currPixel] = true;  
        visited[currPixel] = true;  
        dfs (img, blob, currPixel, colorCode, added, visited);  
      }  
    }  
    //checks right pixel  
    if ((currPixel + 1 < width * height) && (visited[currPixel + 1] == false) && (((currPixel + 1) % width) != 0)) {  
      if (img[currPixel + 1] == colorCode) {  
        blob.add(img[currPixel + 1]);  
        currPixel = currPixel + 1;  
        added[currPixel] = true;  
        visited[currPixel] = true;  
        dfs(img, blob, currPixel, colorCode, added, visited);  
      }  
    }  
    //checks left pixel  
    if ((currPixel - 1 > 0) && (visited[currPixel - 1] == false) && (((currPixel - 1) % width) != width - 1)) {  
      if (img[currPixel - 1] == colorCode) {  
        blob.add(img[currPixel - 1]);  
        currPixel = currPixel - 1;  
        added[currPixel] = true;  
        visited[currPixel] = true;  
        dfs(img, blob, currPixel, colorCode, added, visited);  
      }  
    }  
    return;  
  }  

  //test case, makes a new image thats half black and half white  
  //should only return one blob of size width*height/2  
  public static void main(String[] args) {  
   int width = 320;  
   int height = 240;  
   //builds the image  
   int[] img = new int[width * height];  
   for (int i = 0; i < img.length; i++) {  
     if (i < img.length/4) {  
       img[i] = Color.white.hashCode();  
     } else {  
       img[i] = Color.black.hashCode();  
     }  
   }  

   //runs blobfind  
   Blobfind2 bf = new Blobfind2(width, height, Color.white.hashCode(), 1, img);  
   ArrayList bloblist = bf.findColor();  
   System.out.println(bloblist.size());  
   //need to typecast things coming out of arraylists  
   Blob firstblob = (Blob)bloblist.get(0);  
   System.out.println(firstblob.mass());  
  }  

 private class Blob {  
   private ArrayList pixels = new ArrayList();  
   private Blob() {  
   }  
   private int mass() {  
     return pixels.size();  
   }  
   private void add(int i) {  
     pixels.add(i);  
   }  
   private ArrayList getBlob() {  
     return pixels;  
   }  
 }  

}   

Ответы [ 2 ]

4 голосов
/ 15 апреля 2011

Ошибка переполнения стека не имеет ничего общего с тем, используете ли вы List, Map или какую-либо другую конкретную структуру данных. Эти конструкции расположены в куче. Вы видите ошибку переполнения стека, потому что вы делаете рекурсивные вызовы функций. Каждый рекурсивный вызов функции выделяет память в стеке. Вы можете увеличить значение -Xss (e.g java -Xss8m HelloWorld) или переписать свой алгоритм, чтобы он не был рекурсивным (при условии, что ваш алгоритм правильный).

1 голос
/ 15 апреля 2011

Это выглядит очень похоже на алгоритм flood-fill . Рекурсивная реализация может разрушить стек (например, сделать слишком много рекурсивных вызовов) для больших двоичных объектов просто потому, что вам нужно исследовать 4 соседа для каждого пикселя. В худшем случае это изображение в одном и том же месте!

Я бы попытался сделать стек явным. Вы хотите избежать рекурсии и использовать простой подход, основанный на циклах.

public void dfs () {
     Stack<Pixel> pixels = new Stack<Pixel>();
     pixels.push(currentPixel);

     while (!pixels.isEmpty()) {
         Pixel x = pixels.pop();

         // Do whatever processing on this pixel
         Pixel upPixel = getUpPixel();
         if (upPixel == colorCode) {
             pixels.push(upPixel);
         }

         // And so on
     }

}
...