inline
-
Und wo ist da jetzt genau der Flaschenhals? Hast Du mal gemessen?
Ich kann dazu leider nur wenig sagen. Kenne mich mit solchen Datenbanksachen nicht aus. Zum StringBuffer kann ich dir aber schon 2 Tipps geben:
1. Nutze StringBuilder statt StringBuffer, wenn Du kannst.
2. Wenn Du die benötigte Größe des StringBuffers oder StringBuilders vorher kennst oder abschätzen kannst, dann erzeuge Dir gleich durch einen passenden Konstruktoraufruf einen StringBuffer oder StringBuilder, der groß genug ist.
-
für XML verwende ich immer JDOM http://www.jdom.org/.
Ich weis nicht ob das schneller als StringBuffer ist. Wenn es nicht Thread-Saved sein muss kannst du StringBuilder nehmen. Ist das gleiche wie StringBuffer nur halt nicht Thread-Save, aber schneller.vorher hatte ich noch ne hashmap eingesetzt, um die elemente aus der datenbank zu verwalten, das hat aber auch viel zu viel zeit gekostet (gibts da was flotteres?). zur zeit nehm ich n string array, aber das is halt so unkomfortabel.
Wenn du gezielt Elemente ansprechen willst, ist eine HashMap ziemlich flott. Also wenn du Name = Element hast, dann ist der Zugriff auf das Element über den Name bei einer HashMap schnell.
-
@in: Hallo??! Du machst Datenbank-Zugriffe und XML processing und sch... wegen inlining rum? Da fehlt es schon sehr viel weiter, als das irgendeine Funktion nicht geinlined wird. Mach dich mal mit der Benutzung eines Profilers vertraut und versuche, genauestens herauszufinden, was viel Zeit kostet. Die Methoden der Standard-Bibliothek sind in der Regel nicht so langsam implementiert, aber man muss sie korrekt benutzen.
Es kann dir auch keiner sagen, ob eine Hashmap jetzt besser oder schlechter ist, wenn du nicht sagst, wofür du sie verwenden willst.
-
Und wo ist da jetzt genau der Flaschenhals? Hast Du mal gemessen?
Java selber ist der Flaschenhals
Java sollte man immer nur als letzten Ausweg nehmen.
Aber im richtigen Forum bist du ja immerhin schon.
-
Ist jetzt aber nicht der Sinn von nem Stringbuffer dort wieder Strings zu "addieren".
stimmt, aber das problem liegt ja im StringBuffer ;). Das ist halt immer noch so langsam. Die meiste Zeit geht in der Klasse StringBuffer drauf. Halt im Konstruktor, in der append Methode (am allermeisten) und in der tostring methode.
Und wo ist da jetzt genau der Flaschenhals? Hast Du mal gemessen?
jip, wie gesagt, meine toxml methode braucht am meisten zeit (also auch die methode selber, nicht nur der stringbuffer).
1. Nutze StringBuilder statt StringBuffer, wenn Du kannst.
kann ich leider nicht, nutze java 1.4 , da scheints den noch nicht zu geben :(.
2. Wenn Du die benötigte Größe des StringBuffers oder StringBuilders vorher kennst oder abschätzen kannst, dann erzeuge Dir gleich durch einen passenden Konstruktoraufruf einen StringBuffer oder StringBuilder, der groß genug ist.
okay, das hilft etwas, die expand methode hat auch einiges an zeit verbraten.
für XML verwende ich immer JDOM http://www.jdom.org/.
ist eher zum parsen von bäumen, ich erstelle ja xml.
Wenn du gezielt Elemente ansprechen willst, ist eine HashMap ziemlich flott. Also wenn du Name = Element hast, dann ist der Zugriff auf das Element über den Name bei einer HashMap schnell.
naja, das war schon genau das was ich wollte (hab die spalten der tabelle als key verwendet, und die werte halt als attribute, aber die vielen zugriff waren zu langsam).
so, vorneweg jetzt erstmal danke für eure tipps, es wäre echt toll, wenn ich die geschichte noch etwas schneller kriegen könnte ;).
@in: Hallo??! Du machst Datenbank-Zugriffe und XML processing und sch... wegen inlining rum? Da fehlt es schon sehr viel weiter, als das irgendeine Funktion nicht geinlined wird. Mach dich mal mit der Benutzung eines Profilers vertraut und versuche, genauestens herauszufinden, was viel Zeit kostet. Die Methoden der Standard-Bibliothek sind in der Regel nicht so langsam implementiert, aber man muss sie korrekt benutzen.
Es kann dir auch keiner sagen, ob eine Hashmap jetzt besser oder schlechter ist, wenn du nicht sagst, wofür du sie verwenden willst.
das hilft mir wirklich weiter. ich bin nicht der super java crack, aber deine aussagen lassen auch keine besonderen kenntnisse spüren. ich hab geschrieben, das beim zusammenbau der xml struktur die meiste zeit flöten geht. dass das so sein würde, konnte ich auch schon vor der implementierung abschätzen. natürlich habe ich nachgemessen, wer genau wie schuldig ist. und hättest du meine posts richtig gelesen, hättest du das auch gemerkt. und das sind nunmal die methoden von java.lang. vielleicht benutze ich die nicht richtig, ich will mal im augenblick noch davon ausgehen, deswegen schreibe ich ja überhaupt hier. wenn es nicht an meiner fehlerhaften benutzung liegen würde, hätte c++ler wohl recht mit:
Java selber ist der Flaschenhals
und falls du noch n beispiel haben willst, wie ich mit einer hashmap elemente einer db verwalten möchte (wenn ich dafür von dir einen _nützlichen_ tipp erhalte), bittesehr:
public HashMap next() throws SQLException{ HashMap erg=new HashMap(); if (!res.next()) return false; int anz=resmeta.getColumnCount(); for( int i=1; i<=anz; i++ ){ erg.put(resmeta.getColumnName(i),res.getString(i)); } return true; }
so hatte ich mir ne connect klasse geschrieben, mit der ich recht angenehm und wiederverwendbar datenbankverwaltung ermögliche. aber das war halt grottenlangsam.
viele grüße,
in
-
c++ler schrieb:
Und wo ist da jetzt genau der Flaschenhals? Hast Du mal gemessen?
Java selber ist der Flaschenhals
Java sollte man immer nur als letzten Ausweg nehmen.
Aber im richtigen Forum bist du ja immerhin schon.Ach, Java ist inzwischen eigentlich schon ganz flott. ...und wird immer besser und schneller. Da kann man sich eigelich nicht beschweren, wenn man die Entwicklung der letzten Jahre betrachtet. Aber man muss halt, wie in jeder Sprache, wissen, wo die Schwachstellen in der Sprache oder der Standardbibliothek liegen, die man bei der Programmierung beachten sollte.
-
in schrieb:
nutze java 1.4 , da scheints den noch nicht zu geben :(.
Bist Du an Java 1.4 gebunden? Wenn nein, empfehle ich Dir in jedem Fall, auf eine aktuelle Javaversion umzusteigen. mit neueren Versionen kommen ja auch immer Verbesserung bezüglich der Performance der Standardbibliothek und der JVM im Allgemeinen. Wer weiß: Vielleicht beseitigt das alleine schon deinen Flaschenhals. ...ausprobieren macht klug.
-
Gregor schrieb:
c++ler schrieb:
Und wo ist da jetzt genau der Flaschenhals? Hast Du mal gemessen?
Java selber ist der Flaschenhals
Java sollte man immer nur als letzten Ausweg nehmen.
Aber im richtigen Forum bist du ja immerhin schon.Ach, Java ist inzwischen eigentlich schon ganz flott. ...und wird immer besser und schneller. Da kann man sich eigelich nicht beschweren, wenn man die Entwicklung der letzten Jahre betrachtet. Aber man muss halt, wie in jeder Sprache, wissen, wo die Schwachstellen in der Sprache oder der Standardbibliothek liegen, die man bei der Programmierung beachten sollte.
Aber sind String (oder Array) -Operationen nicht gerade eine Schwachstelle von Java? Die String Klasse ist doch total lahmer Müll. Und wenn man den StringBuffer mit den einfachen C Char Array (strcat...) vergleicht ist er wahrscheinlich auch deutlich langsamer. Und die ständigen ArrayIndexOutOfBound überprüfungen machen das ganze ja auch nicht gerade schnell.
-
rdgdfsgdf schrieb:
Aber sind String (oder Array) -Operationen nicht gerade eine Schwachstelle von Java? Die String Klasse ist doch total lahmer Müll. Und wenn man den StringBuffer mit den einfachen C Char Array (strcat...) vergleicht ist er wahrscheinlich auch deutlich langsamer. Und die ständigen ArrayIndexOutOfBound überprüfungen machen das ganze ja auch nicht gerade schnell.
Das ist eher kein Performanceproblem von Java. Die String-Klasse hat ihre Stärken und Schwächen. Um die Schwächen auszugleichen, gibt es StringBuffer oder auch StringBuilder. Wenn man mit Java programmiert, sollte man eigentlich recht schnell lernen, wo man das eine und wo man das andere einsetzt. Die String-Klasse ist halt immutable und nicht wie eine dynamische Datenstruktur aufgebaut. Das sollte man wissen.
Die Bereichsprüfungen bei Arrays sind auch kein echtes Performanceproblem. Du kannst ja mal ein C++ Programm und ein Java Programm vergleichen, die beide sehr viele Arrayzugriffe machen. Dann wirst Du feststellen, dass da kein wirklich großer Unterschied besteht. Vielleicht ein paar Prozent, aber nichts, was man bei "echtem Code" spürbar merkt.
Echte Performancebremsen sehen in Java anders aus. Zum Beispiel fehlen in der Standardbibliothek Containerklassen für primitive Datentypen. Da kommt man dann schonmal in die Versuchung, die Containerklassen für Objekte mittels der Wrapperklassen für primitive Datentypen zu missbrauchen. Das ist natürlich ein riesiger Overhead, der Zeit und Speicher kostet. ...oder ganz allgemein haben Objekte in Java einen Overhead von 8 Byte pro Objekt. Das ist in C++ so nicht gegeben. Aber man kann sich vorstellen, dass es dann nicht so gut ist, große Mengen, kleiner Objekte in Java zu erzeugen. (...und das hat auch nichts mit dem GC zu tun, der oft für Performenceprobleme verantwortlich gemacht wird, mir persönlich aber noch nie diesbezüglich negativ aufgefallen ist.)
(BTW: Die 8 Byte Overhead pro Objekt haben durchaus eine gewisse Berechtigung. Das ermöglicht Dinge wie Reflection usw.)
Naja, zumindest gibt es schon Eigenschaften von Java, die man kennen sollte, um performanten Code zu schreiben. Das sind aber eher nicht die Dinge, von denen man immer hört, wenn jemand sagt, dass Java unglaublich langsam sei.
BTW: Du kannst unter Java auch ganz bequem mit char-Arrays arbeiten, wenn Du willst. Die Umwandlung von und zu Strings ist da ganz einfach.
-
in schrieb:
das hilft mir wirklich weiter. ich bin nicht der super java crack, aber deine aussagen lassen auch keine besonderen kenntnisse spüren. ich hab geschrieben, das beim zusammenbau der xml struktur die meiste zeit flöten geht. dass das so sein würde, konnte ich auch schon vor der implementierung abschätzen. natürlich habe ich nachgemessen, wer genau wie schuldig ist. und hättest du meine posts richtig gelesen, hättest du das auch gemerkt.
Wenn dir das alles im Vorfeld doch schon so richtig klar war, wie erklärt sich dann dieser Thread? Jetzt schaun wir nochmal an den Anfang zurück:
in schrieb:
Hallo zusammen,
gibts in Java sowas wie inline bei C++? Also um den Methodenrumpf an die rufende Stelle reinzukopieren...
Gruß, in
Uiuiui.
Vielleicht hätte ich es netter ausdrücken sollen, das gebe ich schon zu. Aber dass du tatsächlich im Vorfeld über die tatsächlichen Probleme bescheid wusstest, konnte man bisher nicht so ganz erkennen, auch wenn du inzwischen der Sache schon näher kommst.und falls du noch n beispiel haben willst, wie ich mit einer hashmap elemente einer db verwalten möchte (wenn ich dafür von dir einen _nützlichen_ tipp erhalte), bittesehr:
public HashMap next() throws SQLException{ HashMap erg=new HashMap(); if (!res.next()) return false; int anz=resmeta.getColumnCount(); for( int i=1; i<=anz; i++ ){ erg.put(resmeta.getColumnName(i),res.getString(i)); } return true; }
so hatte ich mir ne connect klasse geschrieben, mit der ich recht angenehm und wiederverwendbar datenbankverwaltung ermögliche. aber das war halt grottenlangsam.
Sieht so aus, als würdest du Spaltennamen als Keys benutzen. Das skaliert recht schlecht bei vielen Elementen, O(n/k), k = Anzahl Spalten der Tabelle € O(n) bei der Suche. Nach Möglichkeit sollten Keys immer sehr unterschiedlich sein, viele unterschiedliche Spaltennamen wirst du nicht haben.
Allerdings weiß ich gar nicht, wie und wie viel du jetzt hier suchst. Damit das optimieren hier etwas bringt, müsstest du erstmal sichergehen, dass die Methoden der Hashmap das langsame sind und dann den Code von den Stellen posten, an denen sie aufgerufen werden. Bis jetzt hast du nur gezeigt, wie du die Hashmap befüllst, aber mir ist noch nicht klar, wie du das dann verwendest. Am besten, du beschreibst mal das typische Szenario, in dem du die Datenstruktur verwendest und was sie dann schnell erledigen können muss, dann finden wir schon das beste heraus.
-
Ich sehe grad, die Hashmap wird nur lokal referenziert und es wird nie aus ihr gelesen. Wofür dient sie überhaupt?
-
Gregor schrieb:
Die Bereichsprüfungen bei Arrays sind auch kein echtes Performanceproblem. Du kannst ja mal ein C++ Programm und ein Java Programm vergleichen, die beide sehr viele Arrayzugriffe machen. Dann wirst Du feststellen, dass da kein wirklich großer Unterschied besteht. Vielleicht ein paar Prozent, aber nichts, was man bei "echtem Code" spürbar merkt.
Ich muss mich korrigieren. Ich habe da gerade ein paar Testprogramme in dieser Art geschrieben und musste feststellen, dass die Unterschiede doch enorm sind. Allerdings nicht immer zu Ungunsten von Java. ...zur allgemeinen Provokation kommt hier mal ein Beispiel, wo Java bei mir deutlich besser abschneidet.
#include <iostream> int main() { const int size = 100000; int *array = new int[size](0); int a = 0; for (int j = 0 ; j < 5 ; ++j) { for (int i = 0 ; i < 2000000000 ; ++i) { array[i%size] += i; } } for (int i = 0 ; i < size ; ++i) { a += array[i]; } delete[] array; std::cout << a << std:: endl; }
vs.
public class ArrayTest { public static void main(String[] args) { final int size = 100000; int[] array = new int[size]; int a = 0; for (int j = 0 ; j < 5 ; ++j) { for (int i = 0 ; i < 2000000000 ; ++i) { array[i%size] += i; } } for (int i = 0 ; i < size ; ++i) { a += array[i]; } System.out.println(a); } }
Ergebnis:
gregor@linux:~/C++Projects/ArrayTest> /usr/java/jdk1.6.0/bin/javac ArrayTest.java gregor@linux:~/C++Projects/ArrayTest> time /usr/java/jdk1.6.0/bin/java -server ArrayTest 1608650240 real 1m13.778s user 1m11.931s sys 0m0.196s gregor@linux:~/C++Projects/ArrayTest> g++ -O3 -o ArrayTest ArrayTest.cpp gregor@linux:~/C++Projects/ArrayTest> time ./ArrayTest 1608650240 real 1m55.157s user 1m54.321s sys 0m0.105s
Keine Ahnung, an was das hier liegt.
-
Eigentlich sollte ein gut (optimal) optimierender Compiler deinen kompletten Quellcode wegoptimieren können (also zur Compilezeit die ganzen Schleifen auflösen und ausrechnen und nur das cout Konstante / System.out Konstante übriglassen).. Von daher könnte das ganze eher auf ein "welcher compiler optimiert besser" vergleich rauslaufen
-
public class ArrayTest { public static void main(String[] args) { int size = Integer.parseInt(args[0]); int num = Integer.parseInt(args[1]); int num2 = Integer.parseInt(args[2]); long start = System.currentTimeMillis(); int[] array = new int[size]; int a = 0; for (int j = 0; j < num; ++j) { for (int i = 0; i < num2; ++i) { array[i % size] += array[(j*i)%size] + i; } } for (int i = 0; i < size; ++i) { a += array[i]; } long time = System.currentTimeMillis() - start; System.out.println(size+ " " + num + " " +num2 + " erg: "+ a + " time: " + time); } }
100000 500 300000 erg: -1896966358 time: 21938
vs.
int main(int argc, char **argv){ int size = atoi(argv[1]); int num = atoi(argv[2]); int num2 = atoi(argv[3]); struct __timeb64 start; struct __timeb64 end; _ftime64( &start ); int* array = new int[size]; memset(array,0,size); int a = 0; for (int j = 0; j < num; ++j) { for (int i = 0; i < num2; ++i) { array[i % size] += array[(j*i)%size] + i; } } for (int i = 0; i < size; ++i) { a += array[i]; } _ftime64( &end ); long time = (end.time - start.time) * 1000 + (end.millitm - start.millitm); printf("%d %d %d erg: %d time: %d",size,num,num2,a, time); getch(); return 0; }
100000 500 300000 erg: -1896966358 time: 18375
java -server funktioniert bei mir nicht.
-
6458345 schrieb:
java -server funktioniert bei mir nicht.
Schade. Das kann nämlich einen enormen Unterschied machen. Man schaue sich zum Beispiel den Unterschied zwischen Client-VM und Server-VM da an:
http://weblogs.java.net/blog/opinali/archive/2005/11/mustangs_hotspo_1.html
...und welche Javaversion und welchen C++ Compiler nutzt Du eigentlich?
Naja, ist ja auch egal. Ich will das nicht unbedingt vertiefen. Letztendlich ist das hier alles ein bischen OT.
-
Und wenn man in C Pointer verwendet:
int main(int argc, char **argv){ int size = atoi(argv[1]); int num = atoi(argv[2]); int num2 = atoi(argv[3]); struct __timeb64 start; struct __timeb64 end; _ftime64( &start ); int* array = new int[size]; memset(array,0,size); int *ptr = &array[0]; int a = 0; int pos = 0; for (int j = 0; j < num; ++j) { for (int i = 0; i < num2; ++i) { (*ptr) += i; if(pos == size) { pos = 0; ptr = &array[0]; } } } for (int i = 0; i < size; ++i) { a += array[i]; } _ftime64( &end ); long time = (end.time - start.time) * 1000 + (end.millitm - start.millitm); printf("%d %d %d erg: %d time: %d",size,num,num2,a, time); getch(); return 0; }
100000 500 1000000 erg: -1706365568 time: 2406
public class ArrayTest { public static void main(String[] args) { int size = Integer.parseInt(args[0]); int num = Integer.parseInt(args[1]); int num2 = Integer.parseInt(args[2]); long start = System.currentTimeMillis(); int[] array = new int[size]; int a = 0; int pos = 0; for (int j = 0; j < num; ++j) { for (int i = 0; i < num2; ++i) { array[pos++] += i; if(pos == size) { pos = 0; } } } for (int i = 0; i < size; ++i) { a += array[i]; } long time = System.currentTimeMillis() - start; System.out.println(size+ " " + num + " " +num2 + " erg: "+ a + " time: " + time); } } 100000 500 1000000 erg: -1706365568 time: 5516
-
java version "1.5.0_06"
Java(TM) 2 Runtime Environment, Standard Edition (build 1.5.0_06-b05)
Java HotSpot(TM) Client VM (build 1.5.0_06-b05, mixed mode)Visual studio .NET 2003
-
ach ja. Ich hatte das ja auch nicht erwähnt:
gregor@linux:~> /usr/java/jdk1.6.0/bin/java -server -version java version "1.6.0-beta2" Java(TM) SE Runtime Environment (build 1.6.0-beta2-b78) Java HotSpot(TM) Server VM (build 1.6.0-beta2-b78, mixed mode) gregor@linux:~> g++ --version g++ (GCC) 3.3.5 20050117 (prerelease) (SUSE Linux) Copyright (C) 2003 Free Software Foundation, Inc. This is free software; see the source for copying conditions. There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
Der C++ Compiler ist bei mir also zugegebenermaßen nicht mehr so ganz neu. Ich weiß nicht, ob sich da inzwischen viel bezüglich der Performance getan hat. Aber Du hast ja auch nicht unbedingt die optimale VM für Java verwendet.
-
Verwende Prepared Statements (info bei SUN oder Javadoc),
ist unendlich viel schneller....