большие виртуальные списки в Swing с использованием AbstractListModel - лимит в 119 000 000 элементов? - PullRequest
1 голос
/ 13 марта 2009

Мой "виртуальный список" на качелях работает хорошо, но, похоже, он не работает, когда я превышаю определенное количество элементов. Под «неудачей» я подразумеваю, что полоса прокрутки волшебным образом исчезает, когда количество элементов> Nmax, и возвращается, когда количество элементов <= Nmax; Кажется, что Nmax где-то около 119,304,000 в моей системе. </p>

С чем я сталкиваюсь?!?!

(Вот тестовая программа: на моем компьютере, если я наберу 119,304, она работает нормально, но я нажимаю стрелку вверх и полоса прокрутки исчезает)

package com.example.test;

import java.awt.BorderLayout;
import java.awt.Dimension;
import javax.swing.AbstractListModel;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JSpinner;
import javax.swing.SpinnerModel;
import javax.swing.SpinnerNumberModel;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

// based on:
// http://www.java2s.com/Tutorial/Java/0240__Swing/extendsAbstractListModel.htm
// http://www.java2s.com/Tutorial/Java/0240__Swing/SpinnerNumberModel.htm
// http://java.sun.com/j2se/1.4.2/docs/api/javax/swing/SpinnerNumberModel.html
// http://www.java2s.com/Tutorial/Java/0240__Swing/ListeningforJSpinnerEventswithaChangeListener.htm
// http://java.sun.com/products/jfc/tsc/tech_topics/jlist_1/jlist.html

public class BigVirtualList extends JFrame {

    public static void main(String[] args) {
        new BigVirtualList();
    }

    static final int initialLength = 1;
    final private JList list1 = new JList();
    final private BVLData bvldata = new BVLData(initialLength*1000);

    public BigVirtualList() {
        this.setTitle("Big virtual list");
        this.getContentPane().setLayout(new BorderLayout());
        this.setSize(new Dimension(400, 300));
        list1.setModel(bvldata);

        list1.setPrototypeCellValue(list1.getModel().getElementAt(0));


        SpinnerModel model1 = new SpinnerNumberModel(initialLength,1,1000000,1);
        final JSpinner spinner1 = new JSpinner(model1);

        this.getContentPane().add(new JScrollPane(list1), BorderLayout.CENTER);
        JLabel label1 = new JLabel("Length (1000s of items):");
        JPanel panel1 = new JPanel(new BorderLayout());
        panel1.add(label1, BorderLayout.WEST);
        panel1.add(spinner1, BorderLayout.CENTER);
        this.getContentPane().add(panel1, BorderLayout.SOUTH);      

        ChangeListener listener = new ChangeListener() {
            public void stateChanged(ChangeEvent e) {
                Integer newLength = (Integer)spinner1.getValue();
                bvldata.setLength(newLength*1000);
            }
        };

        spinner1.addChangeListener(listener);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setVisible(true);
    }
}

class BVLData extends AbstractListModel {
    public BVLData(int length) { this.length = length; }

    private int length;

    public int getLength() { return length; }
    public void setLength(int length) {
        int oldLength = getLength();
        this.length = length;
        int newLength = getLength();

        if (newLength > oldLength)
            fireIntervalAdded(this, oldLength+1, newLength);
        else if (newLength < oldLength)
            fireIntervalRemoved(this, newLength+1, oldLength);      
    }   

    @Override
    public Object getElementAt(int index) {
        return "Item "+index+" mod 107 = "+(index%107);
    }

    @Override
    public int getSize() { return getLength(); }

}

1 Ответ

6 голосов
/ 14 марта 2009

Ну, я не вижу цели, но все равно ...

Я предполагаю, что вы разбиваете целочисленное максимальное значение где-то на панели прокрутки:

System.out.println("CellHeight:"+list1.getFixedCellHeight());
System.out.println("CellHeight*119304000:"+NumberFormat.getNumberInstance().format(list1.getFixedCellHeight()*119304000.0));
System.out.println("MAXINT:"+Integer.MAX_VALUE);

Дает вам цифры:

CellHeight:18
CellHeight*119304000:2.147.472.000
MAXINT:2.147.483.647

Как видите, добавление еще 18000 пикселей в высоту превысит MAXINT ... это доставит вам неприятности.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...