Category: Java

  • How to use TableModels and ListModel with NetBeans GUI Builder

    A default JTable or JList comes with it’s own pre initalized model. Okay – but: how can we modify this model? Which type of model is usually pre initialized?
    In the following I’ll just list some of the may possible ways to work with tables and lists and the NetBeans Gui Builder:
    (more…)

  • java.sql.SQLException: No suitable driver found for ‘jdbc:derby:…

    Kleine Gemeinheit im “Using dblook” guide (Übersichtslink):

    dblook -verbose -d ‘jdbc:derby:pathToDBDBName’

    — Zeitmarke: 2010-01-04 12:27:34.578
    — Quellendatenbank: pathToDBDBName
    — Verbindungs-URL: ‘jdbc:derby:pathToDBDBName
    — appendLogs: false

    java.sql.SQLException: No suitable driver found for ‘jdbc:derby:pathToDBDBName
    at java.sql.DriverManager.getConnection(DriverManager.java:602)
    at java.sql.DriverManager.getConnection(DriverManager.java:207)
    at org.apache.derby.tools.dblook.go(Unknown Source)
    at org.apache.derby.tools.dblook.(Unknown Source)
    at org.apache.derby.tools.dblook.main(Unknown Source)
    — **–> DEBUG: No suitable driver found for ‘jdbc:derby:pathToDBDBName

    (Fast) Derselbe Aufruf geht via ij und NetBeans …

    Lösung: dblook -d jdbc:derby:pathToDBDBName
    Unterschied: einfach ohne die Hochkommata, auch wenn es im Guide immer wieder mal mit Gänsefüßchen steht.

  • Escape analysis, Lock Coarsening und Biased locking

    Die Ausgabe 179 des “The Java Specialists’ Newsletter” stellt ein paar interessante – wenn auch noch experimentelle – Features der Server-VM vor:

    Escape analysis: Damit kann die JVM prüfen, ob ein Objekt einen bestimmten Scope nicht verlässt (z.B. nur in einer Methode verwendet wird) und dieses Objekt dann direkt auf dem Stack anlegen.

    Lock Coarsening: Fordert ein Thread aufeinanderfolgend mehrere Locks auf ein Objekt an und gibt sie dann wieder frei (wie z.B. bei der Verwendung von Vector), kann die JVM diese wiederholten, teuren Anfragen zu einem lock/release zusammenfassen,

    Biased locking: Wird ein Objekt von nur einem Thread ge’lock’ed, kann auf die Sperre ebenso verzichtet werden.

    Im The Java Specialists’ Newsletter werden einige eindrucksvolle MicroBenchmarks gezeigt, die einiges an Performance bringen können. Aber: es sind nur MicroBenchmarks, die den Effekt sehr gut zeigen. In komplexen Applikationen kann der Benefit natürlich deutlich schlechter ausfallen. Soweit ich das aus anderen Seiten gelesen habe, sind die Optionen derzeit noch als experimentell zu betrachten – aber sie sind schon ein schöner Vorgeschmack.

    Ein sehr schönes Statement im Zusammenhang mit Performance Tuning, das den Nagel auf den Kopf trifft: “Assumption is the mother of all f* ups”. Der Spruch drückt sehr schön das aus, was ich bei Performance-Optimierungen immer wieder sage: Erst Messen, dann Tunen. Und niemals Tunen ohne zu Messen. Andernfalls kann man schnell mal Stunden damit zubringen, ein Programmstück auf 5% Ausführungszeit zu drücken … das aber in der Gesamtausführung nahezu keine Zeit verbraucht – womit die Verbesserung quasi nicht existent ist. Oder schlimmer: man denkt, ein Programmteil wäre langsam, optimiert aber erfolglos an der Ursache vorbei.

    Relevante Links:

    http://profiler.netbeans.org/
  • Avoid too much sorting

    “Java is slow” is the sentence that I heard very often when I began studying computer science – and I forunately never really believed it. But why the predjudice? Well Java CAN be slow if it’s just handeled wrong. Often it’s just convenience of just the missing knowledge of implemntations that makes code slow, so I’ll try to post once in a while whenever I come across such code parts in my hobby programming or at my programmings at work.

    So my first issue is about sorting and autoboxing: About last week we profiled some code that felt just sloooow. It turned out that we lost most of the time within a certain loop that was executed very often. The critical part of the code was (stripped from all other stuff) like this :

    ArrayList<Double> list = new ArrayList<Double>(20); // keep 20 smallest calculated values
    while (condition) {
      double value = calculate(args);
      if (list.size < 20 || value < list.get(19)){
        list.add(value);
        Collections.sort(list)
      }
      // strip elements if size is > 20
    }

    So what’s the issue here?

    1. condition holds true for a LOT of iterations (well, can’t change this)
    2. the list is small (just 20) BUT it is to be sorted completely for each insert
    3. could autoboxing be an issue here?

    Okay, what did we change?

    We changed the ArrayList to a SortedDoubleArray (an implementation that I coded some time ago) that inserts the value already in the correct place using Arrays#binaraySearch() and System.arrayCopy(). As I wasn’t quite sure whether or not autoboxing could be an issue here, I created a copy of the class that operates on Doubles instead of the double primitives.

    The Test

    In order to compare the 3 methods (using Collections.sort(), and the SortedArrays using double and Double), I inserted 1,000,000 random double values into the structures and measured the times. The results are:

    • Collection.sort(): 2907 ms (=100%)
    • SortedDoubleArray (with Double-autoboxed values):  93 ms (~3%)
    • SortedDoubleArray (with double primitives):  94 ms (~3%)

    Conclusion

    • Using Collections.sort() is convenient and in most cases absolutely okay! But if you use it in critical locations within the code (for example in loops that are executed very often), you might want to check if there isn’t a better solution.
    • Autoboxing does not hurt in our case

    But never forget: Profile first, then tune. Otherwise you might tune code that has almost no impact to the overall execution time (for example, if the for-loop above is just executed 10 times).  And just change one issue after the other and perform measurements between each step so that you can identify the changes with the most impact.
    If you have no profiler at hand, you might want to try the NetBeans profier.

    value
  • How to load images in Java

    Every once in a while there is a question on the NetBeans mailinglist about how to load an image (or other ressource, like a text file) relative to the executing JAR.

    The Java Tutorial gives the answer in the chapter “How to use Icons“:

    java.net.URL imgURL = getClass().getResource("inSamePackage.png");
    java.net.URL imgURL = getClass().getResource("/de/foo/otherPackage.png");

    (more…)

  • NetBeans mit mehr RAM starten

    Gerade eine Frage auf der NetBeans Mailinglist die Frage gelesen, wie man NetBeans mit mehr Ram starten könne.

    Eine Google-Anfrage später die Antwort:
    Eine entsprechende Einstellung lässt sich in  [NetBeans Installationspfad]/etc/netbeans.conf vornehmen.
    Aber .. welche Größe hat NetBeans denn nun per Default? Ein Blick in eben genannte Conf zeigt:

    # Note that a default -Xmx is selected for you automatically.
    # You can find this value in var/log/messages.log file in your userdir.
    # The automatically selected value can be overridden by specifying -J-Xmx here
    # or on the command line.
    

    Aha. [Userverzeichnis]/.netbeans/6.7/var/log/messages.log soll also Auskunft geben – und tut es auch. Dort steht bei mir (unter anderem)  “-Xmx407m”.

    Fragt sich, wann dieser Wert gesetzt wird. Zum Installationszeitpunkt hatte ich 1024Mb Ram verbaut – später auf 2048Mb erweitert. Auf meinem Arbeitsrechner (2048Mb Ram) steht der Wert auf “-Xmx409m” – also fast gleich. Die Vermutung liegt nahe, dass die Einstellung beim Start automatisch angepasst wird.

  • How to display images with rounded Corners / Borders

    Bilder in Java darzustellen ist keine große Kunst, dazu gibt es auch ein Tutorial von Sun. Etwas schöner wären aber abgerundete Ecken. Und noch schöner wäre es, wenn wir einfach eine kleine Componente hätten die das alles selber macht, die wiederverwendbar wäre und die im GUI-Editor einfach zu bedienen wäre. Gesagt getan.

    Die Anforderung ist also eine Gui Componente, die im Gui Editor (NetBeans in meinem Falle) wiederverwendbar ist (also ein JavaBean), der man ein Bild übergeben kann und bei der die Ecken abgerundet sind. Wenn möglich, wollen wir sogar noch einen Rahmen setzen können. Ein kurzer Blick auf die Seite “Painting in AWT and Swing” zeigt, dass wir paintComponent() überschreiben sollten. Die Componente soll sich der Einfachheit halber der Größe des Bildes automatisch anpassen.

    Die fertige Klasse sieht dann so aus:

    public class ThumbPanel extends JPanel {
    
        protected BufferedImage image = null;
        public static final String PROP_IMAGE = "image";
        protected int roundness = 10;
        public static final String PROP_ROUNDNESS = "roundness";
    
        public ThumbPanel() {
            init();
        }
    
        private void init() {
            setOpaque(false);
        }
    
        protected void update() {
            if (image != null) {
                setSize(image.getWidth(), image.getHeight());
                setPreferredSize(new Dimension(image.getWidth(), image.getHeight()));
            }
        }
    
        @Override
        protected void paintComponent(Graphics g) {
            if (image == null) {
                return;
            }
            g.setClip(new RoundRectangle2D.Double(0, 0, image.getWidth(), image.getHeight(), roundness, roundness));
            g.drawImage(image, 0, 0, null);
            g.setClip(null);
        }
    
        public int getRoundness() {
            return roundness;
        }
    
        public void setRoundness(int roundness) {
            int oldRoundness = this.roundness;
            this.roundness = roundness;
            firePropertyChange(PROP_ROUNDNESS, oldRoundness, roundness);
            update();
        }
    
        public BufferedImage getImage() {
            return image;
        }
    
        public void setImage(BufferedImage image) {
            BufferedImage oldImage = this.image;
            this.image = image;
            firePropertyChange(PROP_IMAGE, oldImage, image);
            update();
        }
    }

    Getter und Setter sind selbsterklärend. Update() passt die Komponente der aktuellen Größe an. Ein Null-Images possiert in meinem Anwendungsfall nicht, wäre aber offenbar kein Problem das anzupassen. PaintComponent() setzt die Clip-Eigenschaft und malt dann das Bild auf die Komponente – fertig! Das ganze sieht dann so aus:

    rounded corners ISo weit so gut. Nun will man aber vielleicht noch einen Rahmen (=Border) dazufügen. Standard Borders sehen dabei nicht ganz so praktikabel aus, da sie an den Ecken abgeschnitten werden. Also muss wohl oder übel eine eigene Border her. Schön wäre ein Rahmen, bei dem man Farbe, Dicke und Rundung einstellen kann. Der Versuch, die LineBorder zu extenden war nicht wirklich von Erfolg gekrönt, da man dabei Dicke und Rundung nicht separat einstellen kann.

    Also selber malen. Linien mit RoundRectangle2Ds zu zeichnen, wollte nie so wirklich schön werden. Insbesondere sobald die Dicke > 1 Pixel sein sollte. Alternative: Roundrect fill und innen wieder Clip’en.  Nur dumm, dass sich das Clip dann auch auf das Bild übertragen hatte. Also per AlphaComposite das Innere einfach ausschneiden. Damit auch bei mehrfachen repaints, nicht immer ein BufferedImage für den Rahmen erzeugt werden muss, wird das Ergebnis einfach gecached. Die ganze klasse sieht dann so aus:

    public class MyRoundBorder implements Border {
    
        private PropertyChangeSupport propertyChangeSupport = new PropertyChangeSupport(this);
        protected int roundness = 10;
        public static final String PROP_ROUNDNESS = "roundness";
        protected Color color = Color.BLACK;
        public static final String PROP_COLOR = "color";
        protected int thickness = 1;
        public static final String PROP_THICKNESS = "thickness";
        // buffer the created image because recreating a complete image could waste precious time
        protected SoftReference cache = new SoftReference(null);
        protected Rectangle oldR = new Rectangle();
        protected Rectangle oldC = new Rectangle();
    
        public MyRoundBorder() {
        }
    
        public MyRoundBorder(int roundness, Color color, int thickness) {
            this.roundness = roundness;
            this.color = color;
            this.thickness = thickness;
        }
    
        public int getThickness() {
            return thickness;
        }
    
        public void setThickness(int thickness) {
            int oldThickness = this.thickness;
            this.thickness = thickness;
            propertyChangeSupport.firePropertyChange(PROP_THICKNESS, oldThickness, thickness);
            cache = new SoftReference(null);
        }
    
        public Color getColor() {
            return color;
        }
    
        public void setColor(Color color) {
            Color oldColor = this.color;
            this.color = color;
            propertyChangeSupport.firePropertyChange(PROP_COLOR, oldColor, color);
            cache = new SoftReference(null);
        }
    
        public int getRoundness() {
            return roundness;
        }
    
        public void setRoundness(int roundness) {
            int oldRoundness = this.roundness;
            this.roundness = roundness;
            propertyChangeSupport.firePropertyChange(PROP_ROUNDNESS, oldRoundness, roundness);
            cache = new SoftReference(null);
        }
    
        public void addPropertyChangeListener(PropertyChangeListener listener) {
            propertyChangeSupport.addPropertyChangeListener(listener);
        }
    
        public void removePropertyChangeListener(PropertyChangeListener listener) {
            propertyChangeSupport.removePropertyChangeListener(listener);
        }
    
        @Override
        public void paintBorder(Component c, Graphics g1, int x, int y, int width, int height) {
            Graphics2D g = (Graphics2D) g1;
    
            BufferedImage buffered = cache.get();
            Rectangle newR = new Rectangle(x, y, width, height);
            if (buffered == null || !c.getBounds().equals(oldC) || !oldR.equals(newR)) {
                buffered = new BufferedImage(c.getWidth(), c.getHeight(), BufferedImage.TYPE_INT_ARGB);
                Graphics2D tmpg = buffered.createGraphics();
                tmpg.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
                tmpg.setClip(g1.getClip());
                tmpg.setColor(color);
                // fill area
                tmpg.fillRoundRect(x, y, width, height, roundness, roundness);
                // cut out inner
                tmpg.setComposite(AlphaComposite.getInstance(AlphaComposite.DST_OUT));
                tmpg.fillRoundRect(x + thickness, y + thickness, width - (2 * thickness), height - (2 * thickness), roundness, roundness);
                tmpg.dispose();
    
                cache = new SoftReference(buffered);
                c.getBounds(oldC);
                oldR = newR;
            }
            // draw border upon image
            g.drawImage(buffered, 0, 0, null);
        }
    
        @Override
        public Insets getBorderInsets(Component c) {
            return new Insets(thickness, thickness, thickness, thickness);
        }
    
        @Override
        public boolean isBorderOpaque() {
            return true;
        }
    }

    Die Getter/Setter sind wieder dazu da, eine schöne JavaBean zu bauen, damit die Border auch im GUI Editor einfach zu verwenden ist.

    Da normale Borders verwendet werden können, kann natürlich auch eine CompoundBorder verwendet werden – um zum Beispiel anzuzeigen, wenn ein Bild selektiert ist (dann zB mit zusätzlichem weißen Innenrahmen), was dann so aussieht (links einfach, rechts Compond):

    CompoundBorder II

  • Erste Schritte mit JavaX JXMapKit

    Update 2014: Mittlerweile kann es durchaus schlauer sein JavaFX zu verwenden, so wie es hier beschrieben ist.


    Nachdem ich festgestellt habe, dass Nasa WorldWind zum Anzeigen von Kartenpositionen vielleicht doch ein bisschen Overkill ist, habe ich mir JXMapKit des SwingLabs-Projekts angesehen.

    Um das Beispiel überhaupt zum Laufen zu bekommen, benötigen wir natürlich die richtigen Libraries. Das wären dann SwingX und SwingX-ws. Derzeit wird man mit der Kombination nicht ganz glücklich, da in SwingX 1.0 (mindestens) eine Methode entfernt wurde, die in SwingX-ws benötigt wird. Der zugehörige Bug ist zwar reported, aber natürlich noch nicht in der aktuellsten Version eingebaut (Stand 30.7.2009). Eine gepatchte Version habe ich hier online gestellt: Jar / Quellen.

    Wenn die Libraries erst einmal im Classpath liegen ist es im Prinzip ganz einfach (ich erkläre wie immer anhand von NetBeans aufgrund des besseren GUI-Editors):

    1. JFrame-Form erstellen
    2. JXMapKit in den Frame ziehen
    3. die JXMapKit Komponente anklicken und in den Properties den defaultProvider auf OpenStreetMap stellen, andernfalls bekommt man nur Exceptions.
    4. fertig!

    Das ganze sollte dann so aussehen:

    Screenshot von JXMapKit
    Screenshot von JXMapKit

    Der Code dazu sieht folgendermaßen aus:

    public class MapViewer extends javax.swing.JFrame {
        public MapViewer() {
            initComponents();
        }
    
        private void initComponents() {
            jXMapKit1 = new org.jdesktop.swingx.JXMapKit();
            setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
            jXMapKit1.setDefaultProvider(org.jdesktop.swingx.JXMapKit.DefaultProviders.OpenStreetMaps);
            getContentPane().add(jXMapKit1, java.awt.BorderLayout.CENTER);
            pack();
        }
    
        public static void main(String args[]) {
            java.awt.EventQueue.invokeLater(new Runnable() {
                public void run() {
                    new MapViewer().setVisible(true);
                }
            });
        }
        private org.jdesktop.swingx.JXMapKit jXMapKit1;
    }

    OpenStreetMap ist ja ganz schön. Aber eine Karte wäre ja auch fein. Das erreicht man, indem eine neue TileFactory erstellt wird. Dazu wird schnell der Konstruktor geändert:

        public MapViewer() {
            initComponents();
            WMSService wms = new WMSService();
            wms.setLayer("BMNG");
            wms.setBaseUrl("http://wms.jpl.nasa.gov/wms.cgi?");
            TileFactory fact = new WMSTileFactory(wms);
            jXMapKit1.setTileFactory(fact);
        }
    Screenshot von JXMapKit mit BlueMarble
    Screenshot von JXMapKit mit BlueMarble der NASA

    Und schon sieht’s so aus:

    Beim Starten kann es sein, dass man erst mal nur ein blaues Fenster sieht. Ändert sich leicht, indem man ein paar mal auf den Minus-Button drückt und etwas abwartet, da das Laden der Bilder vom Nasa-Server etwas dauern kann. Bei genauem Hinsehen, wird man im obigen Screenshot auch bemerken, dass der markierte Ausschnitt im rechten unteren Teil nicht mit der tatsächlichen Darstellung übereinstimmt sondern (in dem Fall) einen Ausschnitt anzeigt, der deutlich südlicher liegt (im Fenster sieht man nämlich eigentlich die Nordspitze Schottlands.

    Nützliche Links:

  • erste Schritte mit Nasa World Wind

    Photos auf einer Karte anzuzeigen kann so schwer nicht sein möchte man meinen. Anbindung an Google Maps oder Google Earth und gut is.

    Will man diese Kartenanzeige jetzt noch in ein Java-Programm integrieren, sieht’s schon anders aus. Google Maps wäre kein Problem, wenn denn JWebPane schon fertig wäre. Ist es aber nicht. Also bleiben derzeit nur noch 2 Methoden: Nasa WorldWind einbinden oder JXMapViewer benutzen.

    Der erste Test mit Nasa WorldWind ging erheblich schneller als erwartet: Das NetBeansWiki beschreibt die wenigen nötigen Schritte.

    1. Nasa Worldwind Java SDK herunterladen
    2. In Netbeans eine Library mit den Dateien worldwind.jar, jogl.jar und gluegen-rt.jar anlegen
    3. die Library zum Projekt hinzufügen
    4. Ein JFrame-Form erstellen
    5. (optional einige JavaBeans in die Palette des GUI Managers hinzufügen)
    6. WorldWindowGLCanvas in den JFrame ziehen
    7. folgende Imports hinzufügen:
      import gov.nasa.worldwind.*;
      import gov.nasa.worldwind.avlist.AVKey;
    8. und folgenden Code unter den initComponents() Aufruf des Konstruktors:
      Model m = (Model) WorldWind.createConfigurationComponent(AVKey.MODEL_CLASS_NAME);
      worldWindowGLCanvas1.setModel(m);
    9. In den Projekteigenschaften noch folgende JVM-Property setzen: -Djava.library.path=c:pfadzumnasaworldwindsdk
    10. fertig!

    Die ganze Klasse sieht dann so aus:

    import gov.nasa.worldwind.*;
    import gov.nasa.worldwind.avlist.AVKey;
    
    public class NWW extends javax.swing.JFrame {
    
        public NWW() {
            initComponents();
            Model m = (Model) WorldWind.createConfigurationComponent(AVKey.MODEL_CLASS_NAME);
            worldWindowGLCanvas1.setModel(m);
        }
    
        private void initComponents() {
            worldWindowGLCanvas1 = new gov.nasa.worldwind.awt.WorldWindowGLCanvas();
            setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
            setTitle("Nasa World WInd");
            setMinimumSize(new java.awt.Dimension(640, 480));
            getContentPane().add(worldWindowGLCanvas1, java.awt.BorderLayout.CENTER);
            pack();
        }
    
        public static void main(String args[]) {
            java.awt.EventQueue.invokeLater(new Runnable() {
                public void run() {
                    new NWW().setVisible(true);
                }
            });
        }
        private gov.nasa.worldwind.awt.WorldWindowGLCanvas worldWindowGLCanvas1;
    }

    Nützliche Links: