我遇到内存不足错误:Java堆空间异常

时间:2014-12-16 16:59:01

标签: java out-of-memory

我目前正在尝试接收文本文件并将文件中的每个单词读入二叉树中,我得到的具体错误是:

Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

我正在阅读项目的文本文件是由教授为我赋予的,所以我知道这不应该遇到任何内存问题我以前从未感觉到这种类型的异常,也不知道在哪里开始请帮忙。这是我的代码:

public class Tester {
    public static void main(String[] args) throws FileNotFoundException {
        Tester run = new Tester();
        run.it();
    }

    public void it() throws FileNotFoundException { 
        BTree theTree = new BTree();
        String str = this.readInFile();
        String [] firstWords = this.breakIntoWords(str);
        String [] finalWords = this.removeNullValues(firstWords);

        for(int i = 0; i < finalWords.length; i++) {
            theTree.add(finalWords[i]);
        }
        theTree.print();
    }

    public String readInFile() throws FileNotFoundException {
        String myFile = "";
        int numWords = 0;
        Scanner myScan = new Scanner(new File("Dracula.txt"));

        while(myScan.hasNext() == true) {
            myFile += myScan.nextLine() + " ";
        }
        return myFile;
    }

    public String [] breakIntoWords(String myFile) {
        String[] words = new String[myFile.length()];
        String nextWord = "";
        int position = 0;
        int i = 0;

        while(myFile.length() > position) {
            char next = myFile.charAt(position);
            next = Character.toLowerCase(next);

            // First trim beginning
            while (((next < 'a') || (next > 'z')) && !Character.isDigit(next)) {
                position++;
                next = myFile.charAt(position);
                next = Character.toLowerCase(next);
            }

            // Now pull only letters or numbers until we hit a space
            while(!Character.isWhitespace(next)) {
                if (Character.isLetterOrDigit(next)) {
                    nextWord += myFile.charAt(position);
                }
                position++; 
                next = myFile.charAt(position);
            }
            words [i] = nextWord;
            i++;
        }
        return words;
    }

    public String[] removeNullValues(String[] myWords) {
        String[] justMyWords = new String[myWords.length];
        for (int i = 0; i < myWords.length; i++) {
            if (myWords[i] != null) {
                justMyWords[i] = myWords[i];
            }
        }
        return justMyWords;
    }
}

这是我的B树类:

public class BTree {
    private BTNode root;
    private int nodeCount;

    public boolean add(String word) {
        BTNode myNode = new BTNode(word);

        if(root == null) {
            root = myNode;
            nodeCount++;
            return true;
        }

        if(findNode(word)) {
            int tmp = myNode.getNumInstance();
            tmp++;
            myNode.setNumInstance(tmp);
            return false;
        }

        BTNode temp = root;
        while(temp != null) {
            if(word.compareTo(temp.getMyWord()) < 0) {
                if(temp.getRightChild() == null) {
                    temp.setLeftChild(myNode);
                    nodeCount++;
                    return true;
                } else {
                    temp = temp.getRightChild();
                }
            } else {
                if(temp.getLeftChild() == null) {
                    temp.setLeftChild(myNode);
                    nodeCount++;
                    return true;
                } else {
                    temp = temp.getLeftChild();
                }
            }
        }
        return false;
    }

    public boolean findNode(String word) {
        return mySearch(root, word);
    }

    private boolean mySearch(BTNode root, String word) {
        if (root == null) {
            return false;
        }
        if ((root.getMyWord().compareTo(word) < 0)) {
            return true;
        } else {
            if (word.compareTo(root.getMyWord()) > 0) {
                return mySearch(root.getLeftChild(), word);
            } else {
                return mySearch(root.getRightChild(), word);
            }
        }
    }

    public void print() {
        printTree(root);
    }

    private void printTree(BTNode root) {
        if (root == null) {
            System.out.print(".");
            return;
        }
        printTree(root.getLeftChild());
        System.out.print(root.getMyWord());
        printTree(root.getRightChild());

    }

    public int wordCount() {
        return nodeCount;
    }
}

我的B树节点类:

public class BTNode {
    private BTNode rightChild;
    private BTNode leftChild;
    private String myWord;
    private int numWords;
    private int numInstance;
    private boolean uniqueWord;
    private boolean isRoot;
    private boolean isDeepest;

    public BTNode(String myWord){

        this.numInstance = 1;
        this.myWord = myWord;
        this.rightChild = null;
        this.leftChild = null;

    }

    public String getMyWord() {
        return myWord;
    }

    public void setMyWord(String myWord) {
        this.myWord = myWord;
    }

    public BTNode getRightChild() {
        return rightChild;
    }

    public void setRightChild(BTNode rightChild) {
        this.rightChild = rightChild;
    }

    public BTNode getLeftChild() {
        return leftChild;
    }

    public void setLeftChild(BTNode leftChild) {
        this.leftChild = leftChild;
    }

    public int getnumWords() {
        return numWords;
    }

    public void setnumWords(int numWords) {
        this.numWords = numWords;
    }

    public boolean isUniqueWord() {
        return uniqueWord;
    }

    public void setUniqueWord(boolean uniqueWord) {
        this.uniqueWord = uniqueWord;
    }

    public boolean isRoot() {
        return isRoot;
    }

    public void setRoot(boolean isRoot) {
        this.isRoot = isRoot;
    }

    public boolean isDeepest() {
        return isDeepest;
    }

    public void setDeepest(boolean isDeepest) {
        this.isDeepest = isDeepest;
    }

    public int getNumInstance() {
        return numInstance;
    }

    public void setNumInstance(int numInstance) {
        this.numInstance = numInstance;
    }
}

5 个答案:

答案 0 :(得分:1)

在构建树时,如果要将元素插入右侧,则会在错误的一侧插入节点。

您应该在BTree类替换此代码:

        while(temp != null) {
        if(word.compareTo(temp.getMyWord()) < 0) {
            if(temp.getRightChild() == null) {
                temp.setRightChild(myNode); // <-- You were using setLeftChild()
                nodeCount++;
                return true;
            } else {
                temp = temp.getRightChild();
            }
        ....
    }

您可能正在创建一个包含左侧所有元素并获取OutOfMemoryError的巨大树

答案 1 :(得分:1)

这个小文件不应该是OutOfMemory错误的原因。

<强>性能 这不是错误,但如果你想读取内存中的整个文件
不要每行读取行并连接字符串。这会减慢您的程序。
您可以使用:

String myFile = new String(Files.readAllBytes(Paths.get("Dracula.txt")));
myFile = myFile.replaceAll("\r\n", " ");
return myFile;

这也不是超快,但更快。

现在错误

字数组太大

public String[] breakIntoWords(String myFile) {
    String[] words = new String[myFile.length()];

您将单词定义为文件 lenght 的lengh 长度数组。如果你这太大了 名称是助记符,表示您需要一个长度文件中的单词数

的数组

nextWord永远不会被重置(OutOfMemory的原因

        // Now pull only letters or numbers until we hit a space
        while (!Character.isWhitespace(next)) {
            if (Character.isLetterOrDigit(next)) {
                nextWord += myFile.charAt(position);
            }
            position++;
            next = myFile.charAt(position);
        }
        words[i] = nextWord;
        i++;

因为下一个字永远不会被设置为&#34;&#34;在将其分配给单词[i]之后。所以下一个词会增长 逐字逐句,你的数组内容如下:

words[0] = "Word1"
words[1] = "Word1Word2"
words[2] = "Word1Word2Word3"

可以想象,这将导致大量的已用空间。

答案 2 :(得分:0)

添加VM参数:

-Xms<size> set initial Java heap size 
-Xmx<size> set maximum Java heap size
-Xss<size> set java thread stack size

或使用以下命令运行:java -Xmx256m yourclass.java

答案 3 :(得分:0)

这取决于各种因素。

  1. 正在运行的Java堆的数量(默认值因32位和64位JDK而异)
  2. 您提供给java程序的文件大小

答案 4 :(得分:0)

您正在尝试将文件的整个内容(即流对象)加载到Java Memory中。在这种情况下,你的文件大小有限(即小)然后上面的代码将在你有限的内存中工作,但如果文件大小增加(即文件的内容增加)。然后你将面临问题。

您必须通过读取chuck中的文件内容来更好地解决此问题。否则你将面临同样的问题。

如果增加JVM参数也不适用于较大的文件。

我觉得你的教授也在测试你项目的实施。