Java Swing Workers и проблема с comboBox - PullRequest
0 голосов
/ 14 мая 2011

Я использую Swing для создания приложения, которое использует базу данных. У меня странная проблема, которую я не понимаю, как ее решить.

В определенной панели у меня есть comboBox, который я хочу заполнить записями, используя запрос к БД. после завершения построения JPanel, который содержит comboBox, я выполняю Worker

private void populateLists(){
    GetSimpleRecordsWorker worker = new GetSimpleRecordsWorker(Tables.characters,panelLeftDetails);
    worker.execute();
}

Его doInBackground выполняет запрос SELECT и после получения результата - в методах done() Рабочего - обновляет модель панели и вызывает refreshFromModel.

панели.
charI.setSimpleModel(simpleModel);
charI.refreshFromModel();

refreshFromModel выполняет следующие действия:

private void refreshFromSimpleModel(){
    charSelectCombo.removeAllItems();
    Pair[] pairs = simpleModel.getRecords();
    for (int i=0; i<pairs.length; i++){
        charSelectCombo.addItem(pairs[i]);
    }
    charSelectCombo.setSelectedItem(null);
}

Теперь проблема в том, что после завершения refreshFromModel - графический интерфейс застрял / завис. Я НЕ ИМЕЮ ПОНЯТИЯ ПОЧЕМУ! Странно, в другом кадре, я использую ту же методологию - и она работает без проблем .

Кроме того, если я попробую removeAll вместо removeAllItems, это сработает, но ComboBox потеряет стрелку и будет странно действовать.

Я взял трассировку кадров в отладчике - там есть такие вещи, как removeSourceEvent или Unsafe.park.

 GUI.GuiHandler at localhost:53547 (Suspended)  
Daemon System Thread [Attach Listener] (Suspended)  
Daemon System Thread [Signal Dispatcher] (Suspended)    
Daemon System Thread [Finalizer] (Suspended)    
    Object.wait(long) line: not available [native method]   
    ReferenceQueue<T>.remove(long) line: not available  
    ReferenceQueue<T>.remove() line: not available  
    Finalizer$FinalizerThread.run() line: not available 
Daemon System Thread [Reference Handler] (Suspended)    
    Object.wait(long) line: not available [native method]   
    Reference$Lock(Object).wait() line: 485 
    Reference$ReferenceHandler.run() line: not available    
Thread [AWT-Shutdown] (Suspended)   
    Object.wait(long) line: not available [native method]   
    Object.wait() line: 485 
    AWTAutoShutdown.run() line: not available   
    Thread.run() line: not available    
Daemon System Thread [Java2D Disposer] (Suspended)  
    Object.wait(long) line: not available [native method]   
    ReferenceQueue<T>.remove(long) line: not available  
    ReferenceQueue<T>.remove() line: not available  
    Disposer.run() line: not available  
    Thread.run() line: not available    
Daemon Thread [AWT-Windows] (Suspended) 
    WToolkit.eventLoop() line: not available [native method]    
    WToolkit.run() line: not available  
    Thread.run() line: not available    
Thread [AWT-EventQueue-0] (Suspended)   
    EventQueue.removeSourceEvents(Object, boolean) line: not available  
    WindowsComboBoxUI$WindowsComboBoxRenderer(Component).removeNotify() line: not available 
    WindowsComboBoxUI$WindowsComboBoxRenderer(Container).removeNotify() line: not available 
    WindowsComboBoxUI$WindowsComboBoxRenderer(JComponent).removeNotify() line: not available    
    CellRendererPane(Container).remove(int) line: not available 
    CellRendererPane(Container).remove(Component) line: not available   
    WindowsComboBoxUI(BasicComboBoxUI).getSizeForComponent(Component) line: not available   
    WindowsComboBoxUI(BasicComboBoxUI).getDisplaySize() line: not available 
    WindowsComboBoxUI(BasicComboBoxUI).getMinimumSize(JComponent) line: not available   
    AutoCompleteComboBox(JComponent).getMinimumSize() line: not available   
    FlowLayout.minimumLayoutSize(Container) line: not available 
    JPanel(Container).minimumSize() line: not available 
    JPanel(Container).getMinimumSize() line: not available  
    JPanel(JComponent).getMinimumSize() line: not available 
    BoxLayout.checkRequests() line: not available   
    BoxLayout.minimumLayoutSize(Container) line: not available  
    JPanel(Container).minimumSize() line: not available 
    JPanel(Container).getMinimumSize() line: not available  
    JPanel(JComponent).getMinimumSize() line: not available 
    BoxLayout.checkRequests() line: not available   
    BoxLayout.preferredLayoutSize(Container) line: not available    
    JPanel(Container).preferredSize() line: not available   
    JPanel(Container).getPreferredSize() line: not available    
    JPanel(JComponent).getPreferredSize() line: not available   
    BorderLayout.layoutContainer(Container) line: not available 
    JPanel(Container).layout() line: not available  
    JPanel(Container).doLayout() line: not available    
    JPanel(Container).validateTree() line: not available    
    JPanel(Container).validateTree() line: not available    
    JPanel(Container).validateTree() line: not available    
    JLayeredPane(Container).validateTree() line: not available  
    JRootPane(Container).validateTree() line: not available 
    JRootPane(Container).validate() line: not available 
    RepaintManager.validateInvalidComponents() line: not available  
    SystemEventQueueUtilities$ComponentWorkRequest.run() line: not available    
    InvocationEvent.dispatch() line: not available  
    EventQueue.dispatchEvent(AWTEvent) line: not available  
    EventDispatchThread.pumpOneEventForFilters(int) line: not available 
    EventDispatchThread.pumpEventsForFilter(int, Conditional, EventFilter) line: not available  
    EventDispatchThread.pumpEventsForHierarchy(int, Conditional, Component) line: not available 
    EventDispatchThread.pumpEvents(int, Conditional) line: not available    
    EventDispatchThread.pumpEvents(Conditional) line: not available 
    EventDispatchThread.run() line: not available   
Thread [DestroyJavaVM] (Suspended)  
Daemon System Thread [D3D Screen Updater] (Suspended)   
    Object.wait(long) line: not available [native method]   
    D3DScreenUpdateManager.run() line: not available    
    Thread.run() line: not available    
Daemon System Thread [TimerQueue] (Suspended)   
    Object.wait(long) line: not available [native method]   
    TimerQueue.run() line: not available    
    Thread.run() line: not available    
Thread [Thread-3] (Suspended)   
    Thread.sleep(long) line: not available [native method]  
    JDCConnectionPool$ConnectionReaper.run() line: 93   
Daemon Thread [SwingWorker-pool-1-thread-1] (Suspended) 
    Unsafe.park(boolean, long) line: not available [native method]  
    LockSupport.park(Object) line: not available    
    AbstractQueuedSynchronizer$ConditionObject.await() line: not available  
    LinkedBlockingQueue<E>.take() line: not available   
    ThreadPoolExecutor.getTask() line: not available    
    ThreadPoolExecutor$Worker.run() line: not available 
    Thread.run() line: not available    

Большое спасибо заранее.

Ответы [ 3 ]

1 голос
/ 15 мая 2011

Добавление 100 000 элементов в поле со списком, вероятно, является причиной вашей проблемы.

Вам также нужно задаться вопросом, как один пользователь может иметь дело с 100 000 элементов в поле со списком.

JComboBox обычно не должен иметь дело с таким большим количеством предметов.

Один из возможных способов оптимизации вашего кода - явный вызов setPrototypeDisplayValue(), таким образом, вы не будете JComboBox проверять измерения ВСЕХ своих элементов модели для вычисления собственных измерений.Это, вероятно, текущая причина, почему ваша система выглядит замороженной.

1 голос
/ 14 мая 2011

его doInBackground выполняет запрос SELECT, а после получения результата обновляет модель панели и вызывает функцию refreshFromModel панели.

Запрос к БД выполняется в doInBackground ().

Обновление модели необходимо выполнить в методе process ().

Прочтите раздел из учебника по Swing на Параллелизм для получения дополнительной информации.

0 голосов
/ 14 мая 2011

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

Возможно, вы можете взглянуть на библиотеку glazedlists . Его основная функция заключается в предоставлении наблюдаемых списков, но он очень гибкий. Одна из вещей, которую вы можете сделать, - это создать ComboBoxModel на основе такого списка и включить автозаполнение для комбинированного списка (через класс AutoCompleteSupport).

Этот подход может помочь избежать проблем с производительностью, связанных с ComboBox, в addItem. Эта часть будет перемещена в списки остекления.

...