Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Roborock Qrevo Curv 2 Flow: ora lava con un rullo
Roborock Qrevo Curv 2 Flow: ora lava con un rullo
Qrevo Curv 2 Flow è l'ultima novità di casa Roborock per la pulizia di casa: un robot completo, forte di un sistema di lavaggio dei pavimenti basato su rullo che si estende a seguire il profilo delle pareti abbinato ad un potente motore di aspirazione con doppia spazzola laterale
Alpine A290 alla prova: un'auto bella che ti fa innamorare, con qualche limite
Alpine A290 alla prova: un'auto bella che ti fa innamorare, con qualche limite
Abbiamo guidato per diversi giorni la Alpine A290, la prima elettrica del nuovo corso della marca. Non è solo una Renault 5 sotto steroidi, ha una sua identità e vuole farsi guidare
Recensione HONOR Magic 8 Lite: lo smartphone indistruttibile e instancabile
Recensione HONOR Magic 8 Lite: lo smartphone indistruttibile e instancabile
Abbiamo provato a fondo il nuovo Magic 8 Lite di HONOR, e per farlo siamo volati fino a Marrakech , dove abbiamo testato la resistenza di questo smartphone in ogni condizione possibile ed immaginabile. Il risultato? Uno smartphone praticamente indistruttibile e con un'autonomia davvero ottima. Ma c'è molto altro da sapere su Magic 8 Lite, ve lo raccontiamo in questa recensione completa.
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 06-06-2010, 14:51   #1
UnknownSoldier
Member
 
Iscritto dal: Aug 2008
Messaggi: 210
[Java] Problema con gestione immagini e heap

Salve a tutti. Sto creando un programma per gestire le immagini, quindi una volte selezionate delle immagini dal proprio hard disk, queste vengono salvate (in una forma rimpicciolita) come ImageIcon, per mostrare una anteprima. Però ho visto che ad un certo punto, ImageIO lancia un "OutOfMemoryError: Java heap space", molto probabilmente per lo spazio occupato dalle JLabel che montano le ImageIcon lette... ma a questo punto mi chiedo, come faccio a mostrare delle piccole anteprime di immagini in una schermata se poi ho problemi di spazio nella memoria? Inoltre ho notato che la lettura delle immagini è abbastanza lenta...

grazie mille!
UnknownSoldier è offline   Rispondi citando il messaggio o parte di esso
Old 06-06-2010, 15:36   #2
PGI-Bis
Senior Member
 
L'Avatar di PGI-Bis
 
Iscritto dal: Nov 2004
Città: Tra Verona e Mantova
Messaggi: 4553
Esaurire la memoria quando si tratta di immagini non è stranissimo in sè ma se le trasformi in icone relativamente piccole, contando che hai circa 50 megabyte a disposizione senza cambiare le impostazioni predefinite, dovrebbe essere un evento piuttosto "lontano" nel tempo.

Non è un risultato facile da ottenere in Java ma è possibile che per qualche ragione tu stia impedendo alla jvm di liberarsi dell'immagine originale dopo aver creato l'icona.

Mi spiego con un esempio. Dato l'URL di un'immagine creo un'icona:

Codice:
BufferedImage creaIcona(URL url, Dimension dimIcona) throws Throwable {
    BufferedImage originale = ImageIO.read(url);
    BufferedImage icona = new BufferedImage(dimIcona.width, dimIcona.height, original.getType());
    Image ridotta = original.getScaledInstance(dimIcona.width, dimIcona.height, Image.SCALE_FAST);
    icona.getGraphics().drawImage(ridotta, 0, 0, null);
    return icona;
}
Quel che conta è che originale e icona siano separate, altrimenti ti resta sul groppone. Nota che puoi restituire direttamente ridotta se non ti serve una BufferedImage o usare BufferedImageOp per non creare esplicitamente "icona", il succo resta quello.

Non mantenere riferimenti all'immagine di partenza: originale non è il file jpg da 500kb che hai sul disco, è la decompressione di quel file che può essere di parecchi megabyte (approssimativamente (altezza * larghezza * 3.815e-6) megabyte).

Circa la lentezza, uno non si rende mai conto di quanto schifo facciano gli hard disk finchè non carica immagini. Se le immagini che trasformi in icone hanno subito un accesso recente potresti essere fortunato e trovarle nella cache del disco. Ma se non sei così fortunato (e di solito non lo sei) ti becchi tutta la latenza del disco.

Vale a dire che non è la decompressione a produrre il ritardo che noti. Prova ne sia il fatto che se carichi due volte in successione un'immagine da file la seconda volta è un lampo.

Non ci puoi fare molto ma devi prestare attenzione a non cementare la GUI. Se carichi tante icone in sequenza nel thread che gestisce la gui ammazzi la percezione di scorrevolezza della tua interfaccia da parte dell'utente. Devi usare un thread a parte, cosa che puoi tranquillamente fare perchè il caricamento di immagini non coinvolge componenti AWT/Swing. SwingWorker ti offre la meccanica di base per fare tutto.
__________________
Uilliam Scecspir ti fa un baffo? Gioffri Cioser era uno straccione? E allora blogga anche tu, in inglese come me!
PGI-Bis è offline   Rispondi citando il messaggio o parte di esso
Old 06-06-2010, 18:42   #3
UnknownSoldier
Member
 
Iscritto dal: Aug 2008
Messaggi: 210
Ma infatti io uso una BufferedImage per leggere e una ImageIcon che salvo in memoria da usare con la JLabel...

Codice:
    public void createCenterPanel()
    {
        Object[] objs = ProgressBar.showDeterminateBar(this, "Caricamento...", 0, images.size());
        JProgressBar bar = (JProgressBar)objs[1];
        for (ImageX image : images)
        {
            JPanel imagePanel = new JPanel();
            imagePanel.setPreferredSize(new Dimension(160, 200));
            imagePanel.add(new JLabel(image.getImagePreview()));
            imagePanel.add(new JLabel(image.getFile().getName()));
            imagePanel.add(new JLabel(image.getFile().length() / 1024 + " KB"));
            centerPanel.add(imagePanel);
            bar.setValue(bar.getValue() + 1);
        }
        JDialog dialog = (JDialog)objs[0];
        dialog.dispose();
        centerPanel.revalidate();
    }
E il metodo getImagePreview():
Codice:
    public ImageIcon getImagePreview()
    {
        BufferedImage image = null;
        
        try
        {
            image = ImageIO.read(file);
        }
        catch (Exception exception)
        {
            JOptionPane.showMessageDialog(null,
                    "Si è verificato un errore durante la lettura del file \"" + file.getName() + "\"",
                    "Errore anteprima immagine", JOptionPane.ERROR_MESSAGE);
        }

        ImageIcon imagePreview = new ImageIcon(image.getScaledInstance(100, 85, Image.SCALE_FAST));

        return imagePreview;
    }
Se invece di ritornare imagePreview, modifico e faccio ritornare null, riesco a leggere tutte le immagini senza alcun problema! Quindi a questo punto viene spontaneo pensare che la colpa sia causata dalle JLabel che diventano troppo pesanti... ma quali alternative ho?
UnknownSoldier è offline   Rispondi citando il messaggio o parte di esso
Old 06-06-2010, 20:02   #4
PGI-Bis
Senior Member
 
L'Avatar di PGI-Bis
 
Iscritto dal: Nov 2004
Città: Tra Verona e Mantova
Messaggi: 4553
Il "peso" di una JLabel non dipende dall'immagine. Verifica con un profiler chi spende la memoria perchè in quel pezzo di codice non mi sembra ci siano problemi.
__________________
Uilliam Scecspir ti fa un baffo? Gioffri Cioser era uno straccione? E allora blogga anche tu, in inglese come me!
PGI-Bis è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Roborock Qrevo Curv 2 Flow: ora lava con un rullo Roborock Qrevo Curv 2 Flow: ora lava con un rull...
Alpine A290 alla prova: un'auto bella che ti fa innamorare, con qualche limite Alpine A290 alla prova: un'auto bella che ti fa ...
Recensione HONOR Magic 8 Lite: lo smartphone indistruttibile e instancabile Recensione HONOR Magic 8 Lite: lo smartphone ind...
Sony WF-1000X M6: le cuffie in-ear di riferimento migliorano ancora Sony WF-1000X M6: le cuffie in-ear di riferiment...
Snowflake porta l'IA dove sono i dati, anche grazie a un accordo con OpenAI Snowflake porta l'IA dove sono i dati, anche gra...
AOC propone un QD-OLED al prezzo di un I...
AGCOM segnalerà la AI Mode di Goo...
L'Estonia porta ChatGPT in classe: firma...
2 nm giapponesi contro TSMC: il piano di...
Questa scopa elettrica Tineco iFLOOR 5 è...
FRITZ!Box 6825 4G sbarca in Italia a 129...
Red Hat presenta Digital Sovereignty Rea...
AMD Zen 6 'Olympic Ridge': sette configu...
Amazon batte Walmart dopo 30 anni: il so...
Digos nel mirino dell'intelligence cines...
Sony non vuole aumentare il prezzo della...
Energia geotermica: la (costosa) rispost...
MG ha già venduto 1 milione di au...
"Something Big Is Happening" —...
Renault chiude il 2025 con una perdita r...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 21:08.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Served by www3v