如何用Java制作数组的数组


115

假设地,我有5个字符串数组对象:

String[] array1 = new String[];
String[] array2 = new String[];
String[] array3 = new String[];
String[] array4 = new String[];
String[] array5 = new String[];

我希望另一个数组对象包含这5个字符串数组对象。我该怎么做?我可以把它放在另一个数组中吗?


43
新手问题可能很严重。实际上,它们经常是。:-)
TJ Crowder

3
对于谁知道如何进行内存对齐,相关的问题和答案并不明显。+1
Benj 2012年

Answers:


153

像这样:

String[][] arrays = { array1, array2, array3, array4, array5 };

要么

String[][] arrays = new String[][] { array1, array2, array3, array4, array5 };

(后一种语法可以在变量声明点之外的赋值中使用,而较短的语法仅适用于声明。)


您能否进一步解释第二种语法的作用?对我来说还不清楚。
Terence Ponce

4
@Terence:它与第一个相同:创建一个字符串数组引用数组,初始化为值array1,array2,array3,array4和array5-每个值本身就是一个字符串数组引用。
乔恩·斯基特

1
快速提问:如果我不知道将创建多少个数组对象,该如何在运行时执行此操作?
Terence Ponce

1
@Terence:您能举一个更具体的例子吗?在编译时指定初始值时,您确实知道大小。你是说类似的东西new String[10][]吗?
乔恩·斯基特

是。类似于彼得的回答。
Terence Ponce

71

尝试

String[][] arrays = new String[5][];

1
这个更
灵活

您不应该在阵列上定义固定大小吗?
Filip

@Filip固定为5。设置下一个级别会预先分配它们,但是可以更改,因此设置它可能没有用。
彼得·劳瑞

8
如何将数据插入数组?是否有其动态数据?
Prakhar Mohan Srivastava 2015年

1
@PrakharMohanSrivastava可以单独设置元素:arrays[0] = new String[] {"a", "b", "c"}或使用临时列表:<pre> <code> List <String []> myList = new ArrayList <>(); myList.add(new String [] {“ a”,“ b”,“ c”})); myList.add(new String [] {“ d”,“ e”,“ f”})); myList.toArray(arrays); </ code> </ pre>
kntx

26

虽然有两个很好的答案告诉您如何做,但我感到缺少另一个答案:在大多数情况下,您根本不应该这样做。

数组很麻烦,在大多数情况下,最好使用Collection API

使用集合,您可以添加和删除元素,并且针对不同功能(基于索引的查找,排序,唯一性,FIFO访问,并发等)有专门的集合。

虽然了解数组及其用法当然是很重要且很重要的,但是在大多数情况下,使用Collections可以使API更加易于管理(这就是为什么像Google Guava这样的新库几乎根本不使用Arrays的原因)。

因此,对于您的情况,我希望使用“列表列表”,并使用Guava创建它:

List<List<String>> listOfLists = Lists.newArrayList();
listOfLists.add(Lists.newArrayList("abc","def","ghi"));
listOfLists.add(Lists.newArrayList("jkl","mno","pqr"));

比String [] []稍微复杂一点,但是允许更多操作,例如连接数据。但是,您的解决方案不能确保数据大小,这可能是一个问题。
Benj 2012年

1
@Benj,如有必要,总是可以编写仅接受一定数量项目的List装饰器。
肖恩·帕特里克·弗洛伊德

确切地说,装饰器/包装器是确保一致性的好方法。因此,我们所说的方式远比简单数组复杂。我所做的是一个小的实用程序类Array2D <T>,它封装了一些基本方法,例如exixts(...)等。
Benj 2012年

6

在与肖恩·帕特里克·弗洛伊德(Sean Patrick Floyd)一起发表的评论中,我提到了一个类:我用的是特殊的用法,它需要WeakReference,但是您可以轻松地用任何对象更改它。

希望有一天能对某人有所帮助:)

import java.lang.ref.WeakReference;
import java.util.LinkedList;
import java.util.NoSuchElementException;
import java.util.Queue;


/**
 *
 * @author leBenj
 */
public class Array2DWeakRefsBuffered<T>
{
    private final WeakReference<T>[][] _array;
    private final Queue<T> _buffer;

    private final int _width;

    private final int _height;

    private final int _bufferSize;

    @SuppressWarnings( "unchecked" )
    public Array2DWeakRefsBuffered( int w , int h , int bufferSize )
    {
        _width = w;
        _height = h;
        _bufferSize = bufferSize;
        _array = new WeakReference[_width][_height];
        _buffer = new LinkedList<T>();
    }

    /**
     * Tests the existence of the encapsulated object
     * /!\ This DOES NOT ensure that the object will be available on next call !
     * @param x
     * @param y
     * @return
     * @throws IndexOutOfBoundsException
     */public boolean exists( int x , int y ) throws IndexOutOfBoundsException
    {
        if( x >= _width || x < 0 )
        {
            throw new IndexOutOfBoundsException( "Index out of bounds (get) : [ x = " + x + "]" );
        }
        if( y >= _height || y < 0 )
        {
            throw new IndexOutOfBoundsException( "Index out of bounds (get) : [ y = " + y + "]" );
        }
        if( _array[x][y] != null )
        {
            T elem = _array[x][y].get();
            if( elem != null )
            {
            return true;
            }
        }
        return false;
    }

    /**
     * Gets the encapsulated object
     * @param x
     * @param y
     * @return
     * @throws IndexOutOfBoundsException
     * @throws NoSuchElementException
     */
    public T get( int x , int y ) throws IndexOutOfBoundsException , NoSuchElementException
    {
        T retour = null;
        if( x >= _width || x < 0 )
        {
            throw new IndexOutOfBoundsException( "Index out of bounds (get) : [ x = " + x + "]" );
        }
        if( y >= _height || y < 0 )
        {
            throw new IndexOutOfBoundsException( "Index out of bounds (get) : [ y = " + y + "]" );
        }
        if( _array[x][y] != null )
        {
            retour = _array[x][y].get();
            if( retour == null )
            {
            throw new NoSuchElementException( "Dereferenced WeakReference element at [ " + x + " ; " + y + "]" );
            }
        }
        else
        {
            throw new NoSuchElementException( "No WeakReference element at [ " + x + " ; " + y + "]" );
        }
        return retour;
    }

    /**
     * Add/replace an object
     * @param o
     * @param x
     * @param y
     * @throws IndexOutOfBoundsException
     */
    public void set( T o , int x , int y ) throws IndexOutOfBoundsException
    {
        if( x >= _width || x < 0 )
        {
            throw new IndexOutOfBoundsException( "Index out of bounds (set) : [ x = " + x + "]" );
        }
        if( y >= _height || y < 0 )
        {
            throw new IndexOutOfBoundsException( "Index out of bounds (set) : [ y = " + y + "]" );
        }
        _array[x][y] = new WeakReference<T>( o );

        // store local "visible" references : avoids deletion, works in FIFO mode
        _buffer.add( o );
        if(_buffer.size() > _bufferSize)
        {
            _buffer.poll();
        }
    }

}

使用方法示例:

// a 5x5 array, with at most 10 elements "bufferized" -> the last 10 elements will not be taken by GC process
Array2DWeakRefsBuffered<Image> myArray = new Array2DWeakRefsBuffered<Image>(5,5,10);
Image img = myArray.set(anImage,0,0);
if(myArray.exists(3,3))
{
    System.out.println("Image at 3,3 is still in memory");
}

4
+1是您的努力,但是:与其将int字段初始化为-1并在构造函数中重新分配它们,不应该将它们定型并在构造函数中分配它们。
肖恩·帕特里克·弗洛伊德

1
@Sean:我修改了代码(使用“ no-GC buffer”发布了新代码,包括您的明智评论。)
Benj 2012年
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.