java.lang.ArrayStoreException異常的解決方案

java.lang.ArrayStoreException異常

異常提示

java.lang.ArrayStoreException: java.lang.Boolean
at java.util.stream.Nodes$FixedNodeBuilder.accept(Nodes.java:1222)
at java.util.stream.ReferencePipeline$3$1.accept(ReferencePipeline.java:193)
at java.util.Spliterators$ArraySpliterator.forEachRemaining(Spliterators.java:948)
at java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:481)
at java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:471)
at java.util.stream.AbstractPipeline.evaluate(AbstractPipeline.java:545)
at java.util.stream.AbstractPipeline.evaluateToArrayNode(AbstractPipeline.java:260)
at java.util.stream.ReferencePipeline.toArray(ReferencePipeline.java:438)
at

查詢百度的解釋:試圖將錯誤類型的對象存儲到一個對象數組時拋出的異常。之後,在看看自己錯誤的代碼:

Field[] filterCopyFields = Stream.of(appendFields)
     .map(f -> !preFieldNames.contains(f.getName())).toArray(Field[]::new);

很容易看出問題的所在,這裡我是想過濾Field[]數組中的元素,!preFieldNames.contains(f.getName())這個是過濾條件,發現瞭這裡使用的居然是map,過濾應該是使用filter,map中的元素應該是返回結果並在toArray方法中轉換成數組,這裡map中返回的是Boolean佈爾類型的數據,也就是說不能將boolean類型的對象存儲到Field對象數組中。

這裡可以看一下JDK8源碼中對toArray(IntFunction<A[]> generator)方法的定義:

    /**
     * Returns an array containing the elements of this stream, using the
     * provided {@code generator} function to allocate the returned array, as
     * well as any additional arrays that might be required for a partitioned
     * execution or for resizing.
     *
     * <p>This is a <a href="package-summary.html#StreamOps" rel="external nofollow" >terminal
     * operation</a>.
     *
     * @apiNote
     * The generator function takes an integer, which is the size of the
     * desired array, and produces an array of the desired size.  This can be
     * concisely expressed with an array constructor reference:
     * <pre>{@code
     *     Person[] men = people.stream()
     *                          .filter(p -> p.getGender() == MALE)
     *                          .toArray(Person[]::new);
     * }</pre>
     *
     * @param <A> the element type of the resulting array
     * @param generator a function which produces a new array of the desired
     *                  type and the provided length
     * @return an array containing the elements in this stream
     * @throws ArrayStoreException if the runtime type of the array returned
     * from the array generator is not a supertype of the runtime type of every
     * element in this stream
     */
    <A> A[] toArray(IntFunction<A[]> generator);

可以看到toArray()的參數是IntFunction<A[]>類型,從@param A the element type of the resulting array這個註解中可以看到,A是表示返回數組的元素類型,在我的例子中返回類型是一個Field,而如果Stream中使用瞭map遍歷,返回的類型又是Boolean,類型不匹配而出現錯誤。

解決更改

Field[] filterCopyFields = Stream.of(appendFields)
     .filter(f -> !preFieldNames.contains(f.getName())).toArray(Field[]::new);

其實這種小問題應該很容易避免,在出現ArrayStoreException異常時應該對應著數組中的元素類型去查找錯誤,構造數組時應按照正確的類型來構造。

Java工具類List的toArray方法及java.lang.ArrayStoreException

1.List接口中有兩個方法

Object[] toArray(); 
T[] toArray(T[] a);

分析:不帶參數的方法默認是把數組轉換為Object類型,而帶參數的方法會將數組轉換為指定的類型;

指定目標數組數據類型:

        List<Integer> list = new ArrayList<Integer>();
        list.add(12);
        list.add(13);
        list.toArray(new Integer[list.size()]);

不指定目標數組數據類型獲得的數組類型是Object類型:

        List<Integer> list = new ArrayList<Integer>();
        list.add(12);
        list.add(13);
        list.toArray();

2.使用toArray方法是出現java.lang.ArrayStoreException異常

public class StingUtilsTest{
    public static void main(String[] args) {
        List<Integer> list = new ArrayList<Integer>();
        list.add(12);
        list.add(13);
        list.toArray(new Long[list.size()]);
    }
}

Exception in thread “main” java.lang.ArrayStoreException
at java.lang.System.arraycopy(Native Method)
at java.util.ArrayList.toArray(ArrayList.java:390)
at common.lang.StingUtilsTest.main(StingUtilsTest.java:23)

分析:出現這種異常是由於數組中存入的數據與要轉換的目標數組的類型不一致導致的;還有一點需要註意的是toArray參數數組的初始化大小如果list.size大於等於list的列表的長度那麼就默認使用當前的參數數組,如果小於list的長度就會重新創建一個數組,建議如果知道list的長度一定要初始化數組的長度,這樣可以節省內存空間,提高效率;

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: