Zurück   3DCenter Forum > Software-Hilfe Foren > Programmierung
Registrieren Hilfe Community Kalender Heutige Beiträge Suchen Uns unterstützen

Antwort
 
Themen-Optionen Ansicht
Alt 2004-11-28, 18:52:19   #1 (im Thread / einzeln)
lola
Gold Member
 
Registriert: 2003-08-12
Beiträge: 733
Problem mit singleton unter java

Hi

ich hab eine Singleton Klasse mit Namen Data erzeugt und möchte nun verhindern daß eine 2. Instanz erzeugt werden kann wenn schon eine erzeugt wurde.Kann mir da einer weiterhelfen wie ich das realisieren könnte? Hier der Code:

final class Data {
private String name;
// create the one and only object “in store”
private static Data theData = new Data ("NoName");

// a private constructor
private Data (String n) {this.name = n;}
// a reference to the object
public static Data createData() {
return theData;
}
// other methods of class Data
public void setName (String n) { name = n; }
public String getName () { return name;
}
}

Geändert von lola (2004-11-28 um 20:17:52 Uhr)
lola ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-28, 21:34:14   #2 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Tuts du ja schon. Der Konstrukor ist private, also wird verhindert, dass er von ausserhalb der Klasse aufgeraufen wird.
Ist im Moment nicht lazy aber das ist ja nicht Voraussetzung für ein singelton.

BTW, [ code ] Tags regeln.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-28, 21:54:12   #3 (im Thread / einzeln)
lola
Gold Member
Threadstarter
 
Registriert: 2003-08-12
Beiträge: 733
Re: Problem mit singleton unter java

Nee,ich meinte ich kann ja trotzdem 2 instanzen erzeugen,zB:

Code:
public static void main(String[] args) {

		Data firstData = Data.createData();
		System.out.println(firstData.getName());
		firstData.setName("one");
		System.out.println(firstData.getName());
		//firstData.Ausgabe();
		//firstData.Eingabe();
		//firstData.Ausgabe();
		Data secondData = Data.createData();
		System.out.println(secondData.getName());
		secondData.setName("two");
		System.out.println(secondData.getName());
		System.out.println(firstData.getName());
}
Wie könnte ich verhindern dass da die 2. Instanz erzeugt wird (also secondData)?
lola ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-28, 22:14:25   #4 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Sorry, aber genau dieser Code beweist, das du eben nicht neue Instanzen kreierst. Wie denn auch, createData verwendet ja auch nicht new sondern gibt bloss eine Klassenvariable, die zufälligerweise die singleton Instanz ist zurück.

Falls du mir immer noch nicht glaubst:
Code:
System.out.println("Are firstData and secondData the very same object? " + (firstData == secondData ? "Yes" : "No" ));
update:
Es geht eigentlich schon, aber nicht mit deinem Code und ich glaube nicht, dass du das meintest. Wenn du das verhindern willst, musst du entweder einen SecurityManager verwenden, der das verbietet oder im Konstrukor einen check machen, ob die singleton Instanz schon kreiert wurde. Falls ja wirfst du eine Exception.
Code:
public class NotSoSingleton {

    public static void main(String[] args) {
        try {
            Class<Data> dataClass = Data.class;
            Constructor<Data> dataConstructor = dataClass.getDeclaredConstructor(String.class);
            dataConstructor.setAccessible(true);
            Data data1 = dataConstructor.newInstance("one");
            Data data2 = dataConstructor.newInstance("two");
            System.out.println(data1 != data2 ? "OK" : "NOK");
        } catch (SecurityException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
    	} catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }
}

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.

Geändert von HellHorse (2004-11-29 um 14:58:45 Uhr)
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-29, 17:44:14   #5 (im Thread / einzeln)
lola
Gold Member
Threadstarter
 
Registriert: 2003-08-12
Beiträge: 733
Re: Problem mit singleton unter java

Zitat von HellHorse:
update:
Es geht eigentlich schon, aber nicht mit deinem Code und ich glaube nicht, dass du das meintest. Wenn du das verhindern willst, musst du entweder einen SecurityManager verwenden, der das verbietet oder im Konstrukor einen check machen, ob die singleton Instanz schon kreiert wurde. Falls ja wirfst du eine Exception.

Genau das habe ich gemeint,danke für die Hilfe!
lola ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 10:53:10   #6 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Update2:
Das mit dem check im Konstruktor lässt sich auf die gleiche Weise aushebeln. Du musst also zwangläufig einen SecurityManager verwenden.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 11:22:08   #7 (im Thread / einzeln)
grakaman
Admiral Member
 
Registriert: 2002-01-13
Beiträge: 3.847
grakaman eine Nachricht über MSN schicken
Re: Problem mit singleton unter java

Den Check auf null macht man doch in der Factory-Methode.

Code:
class myclass
{
   private static myclass = null;

   private myclass()
   {

   }

   public static myclass CreateInstance()
   {
      if(myclass == null)
         myclass = new myclass();

      return myclass;
   }
}
Wenn du die Instanzierung in der Factory vornimmst, hast du den Vorteil über die Factory z.B. einen überladenen Konstruktor aufzurufen. Allerdings ist das Bsp. nicht Threadsafe. Wie das in Java Threadsafe implementiert wird, weiß ich nicht.

Geändert von grakaman (2004-11-30 um 11:45:47 Uhr)
grakaman ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 12:14:58   #8 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Zitat von grakaman:
Den Check auf null macht man doch in der Factory-Methode.
Wenn du es lazy willst, dann schon.

update:
Der check im Konstruktor war nur für den Fall gedacht, dass sich jemand den private Konstruktor über Reflection schnappt, als accessible marktiert und dann aufruft. Wie schon gesagt, auch dieser Test lässt sich aushebeln.
Zitat von grakaman:
Wenn du die Instanzierung in der Factory vornimmst, hast du den Vorteil über die Factory z.B. einen überladenen Konstruktor aufzurufen.
Kannst du sonst auch machen.
Zitat von grakaman:
Allerdings ist das Bsp. nicht Threadsafe. Wie das in Java Threadsafe implementiert wird, weiß ich nicht.
synchronzied und gut ist.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.

Geändert von HellHorse (2004-11-30 um 12:36:08 Uhr)
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 12:35:41   #9 (im Thread / einzeln)
grakaman
Admiral Member
 
Registriert: 2002-01-13
Beiträge: 3.847
grakaman eine Nachricht über MSN schicken
Re: Problem mit singleton unter java

Zitat von HellHorse:
Wenn du es lazy willst, dann schon.


Zitat:
Kannst du sonst auch machen.
Und wie?

Geändert von grakaman (2004-11-30 um 12:35:56 Uhr)
grakaman ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 14:03:00   #10 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Zitat von grakaman:
Normalerweise (so wie du es gepostet hast, und nicht so wie lola es gepostet hat) ist ein singleton lazy initialized. Will heissen die singleton Instanz wird erst kreiert, wenn das erste mal jemand die factory Methode aufruft und nicht schon wenn die Klasse geladen wird, daher ist auch der check nötig. Das ist aber nicht Anforderung an ein singleton.
Zitat von grakaman:
Und wie?
Hoffe wir reden vom gleichen:
Code:
final class Data {
    private String name;

    // create the one and only object \u201cin store\u201d
    private static Data theData = new Data("NoName");
    
    // a private default constructor    
    private Data() {
        this.name = "";
    }

    // a private constructor
    private Data(String n) {
        this.name = n;
    }
...
}
Anmerkung zu Update2:
Das Aushebeln geht allerdings nur, wenn theData nicht final ist.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.

Geändert von HellHorse (2004-11-30 um 14:04:59 Uhr)
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 14:21:42   #11 (im Thread / einzeln)
grakaman
Admiral Member
 
Registriert: 2002-01-13
Beiträge: 3.847
grakaman eine Nachricht über MSN schicken
Re: Problem mit singleton unter java

Na ich meinte eigentlich, wenn man in der/den Factory(s) die Instanz erstellt (deswegen heißt die wohl auch Factory), kann der, der die Klasse von "draußen" aufruft, festlegen mit welchen Konstruktor er sie initialisieren will. Das kann ich ja logischerweise nicht machen, wenn ich, wie lola, das Feld fest auf Klassenebene instanziere.

Geändert von grakaman (2004-11-30 um 14:22:53 Uhr)
grakaman ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 15:15:28   #12 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Theoretisch, aber wie schon gesagt, bloss beim ersten mal, wenn die singleton Instanz noch nicht erstellt wurde. Falls sie schon besteht, geht es nicht. Es gibt also keine Garantie, dass es hinaut und ist daher sehr gefährlich. Und eigentlich ist es ja die Idee einer factory Methode unter anderem von solchem Zeugs, wie welcher Konstruktor verwendet wird zu abstrahieren.

Auf Klassenebene kannst du immer noch so was machen:
Code:
final class Data {
    private String name;

    // create the one and only object \u201cin store\u201d
    private final static Data theData;
    
    static {
        Random rand = new Random();
        if (rand.nextBoolean()) {
            theData = new Data("a String");
        } else {
            theData = new Data();
        }
    }

    // a private default constructor    
    private Data() {
        this.name = "";
    }

    // a private constructor
    private Data(String n) {
        this.name = n;
    }
    
    ....
}

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.

Geändert von HellHorse (2004-11-30 um 15:19:27 Uhr)
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 15:57:33   #13 (im Thread / einzeln)
grakaman
Admiral Member
 
Registriert: 2002-01-13
Beiträge: 3.847
grakaman eine Nachricht über MSN schicken
Re: Problem mit singleton unter java

Aber genau das machst du doch mit der Factory Klasse. Du bräuchtest ja auch keine Factory Klasse, wenn du die Instanzierung des Feld auf Klassenebene durchführst. Dann könntest du das Feld gleich public machen.
Die Sache ist jetzt die, dass du das Singleton beliebig instanzieren willst. Das verbietet ja ein Singleton nicht, es geht ja nur darum für einen bestimmten Zeitraum nur eine Instanz zu garantieren. Aber genau so gut könntest du ja in der Klasse eine statische Methode implementieren, die die Instanz wieder löscht. Nur wenn du die Instanzierung in einer extra Methode vornimmst, (Factory) kannst du von Außerhalb das Singleton mit beliebigen Konstruktoren aufrufen. Und in deinem Bsp. mag das zwar so gehen, aber wenn du verschiedene Konstruktoren mit unterschiedlichen Parametern hast, dann müssen die ja auch irgendwo herkommen. Und dann für alle Parameter noch Instanzfelder deklarieren werden und ggf. noch Getter- und Setter-Methoden, anstatt die gleich nur einer Methode zu übergeben und dort die Instanzierung vorzunehmen, sieht mir das ziemlich umständlicher aus. Aber im Endeffekt kann ich das auch nur so sagen, wie ich es eben immer gelesen habe. Kann sein, dass man das eben bei Java anders macht.

Geändert von grakaman (2004-11-30 um 16:00:11 Uhr)
grakaman ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 16:47:45   #14 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Zitat von grakaman:
Aber genau das machst du doch mit der Factory Klasse. Du bräuchtest ja auch keine Factory Klasse, wenn du die Instanzierung des Feld auf Klassenebene durchführst. Dann könntest du das Feld gleich public machen.
Ich kann ja auch später eine andere Klasse im static initializer instanzieren, kein Problem.
Zitat von grakaman:
Die Sache ist jetzt die, dass du das Singleton beliebig instanzieren willst. Das verbietet ja ein Singleton nicht, es geht ja nur darum für einen bestimmten Zeitraum nur eine Instanz zu garantieren.
Nein, es darum sicherzustellen, dass von einer Klasse, bloss eine Instanz existiert. Schlag sonst noch einmal in Design Patterns nach.
Zitat von grakaman:
Aber genau so gut könntest du ja in der Klasse eine statische Methode implementieren, die die Instanz wieder löscht.
Wenn dir deine Sprache erlaubt zu zählen wieviele Pointer auf ein Objekt existieren. Java tut es z.B nicht.
Zitat von grakaman:
Nur wenn du die Instanzierung in einer extra Methode vornimmst, (Factory) kannst du von Außerhalb das Singleton mit beliebigen Konstruktoren aufrufen.
Sollst du aber nicht können. Darum ist der Konstruktor auch private. Das hat aber nix damit zu tun, wann das singleton kreiert wird. Ich verweise hier noch einmal auf Design Patterns.
Zitat von grakaman:
Und in deinem Bsp. mag das zwar so gehen, aber wenn du verschiedene Konstruktoren mit unterschiedlichen Parametern hast, dann müssen die ja auch irgendwo herkommen. Und dann für alle Parameter noch Instanzfelder deklarieren werden und ggf. noch Getter- und Setter-Methoden, anstatt die gleich nur einer Methode zu übergeben und dort die Instanzierung vorzunehmen, sieht mir das ziemlich umständlicher aus.
Öh ja, der Konstruktor muss schon irgend woher kommen. Aber ich sehe nicht genau was du willst. Die Initialisation in der factory Methode statt im Konstruktor vornehmen oder was?
Zitat von grakaman:
Aber im Endeffekt kann ich das auch nur so sagen, wie ich es eben immer gelesen habe. Kann sein, dass man das eben bei Java anders macht.
"Eigentlich" macht man es schon so, wie du es gepostet hast. Bloss muss es halt eben nicht lazy sein, damit es ein singleton ist.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.

Geändert von HellHorse (2004-11-30 um 16:48:22 Uhr)
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 18:37:20   #15 (im Thread / einzeln)
grakaman
Admiral Member
 
Registriert: 2002-01-13
Beiträge: 3.847
grakaman eine Nachricht über MSN schicken
Re: Problem mit singleton unter java

Der Konstruktor ist private, damit nicht beliebig viele Instanzen erstellt werden können, sondern nur eine. Das hat aber nun nichts damit zu tun, dass keiner von außerhalb das Objekt instanzieren bzw. verschieden initialisieren darf. Es muss lediglich gewährleistet werden, dass eben von der Klasse nur eine Instanz vorhanden ist. Wer da den Anfgang macht, ist ja unerheblich. Und das muss ja auch nicht die komplette App-Lebenszeit betreffen. Ich kann ja genau so ein System implementieren, dass auf ein Ereignis plötzlich komplexe Operationen macht und während dessen, sollen alle partizipierenden Objekte auf ein und die selbe Instanz zugreifen. Wenn die Operation fertig ist und ein neues Ereignis ausgelöst wird, möchte ich vielleicht ein anders initialisiertes Singleton Objekt, bei dem auch wieder alle partizipierenden Teilobjekte nur auf eine Instanz zugreifen sollen. Das ganze kann ja voll und ganz sequenziell ablaufen, dabei muss es sich doch nicht zwangsweise um irgend welche Remoteobjekte handeln, auf das mehrere Clients gleichzeitig zugreifen.

Geändert von grakaman (2004-11-30 um 19:00:37 Uhr)
grakaman ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 20:31:40   #16 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Zitat von grakaman:
Der Konstruktor ist private, damit nicht beliebig viele Instanzen erstellt werden können, sondern nur eine.
Der Konstruktor ist private, damit "von aussen" keine Objekte generiert werden können.
Zitat von grakaman:
Das hat aber nun nichts damit zu tun, dass keiner von außerhalb das Objekt instanzieren bzw. verschieden initialisieren darf.
Doch. Schau dir mal die Anmerkungen zur C++ Implementation in Design Patterns doch noch einmal an. Die sind diesbezüglich sehr klar.
Zitat von grakaman:
Es muss lediglich gewährleistet werden, dass eben von der Klasse nur eine Instanz vorhanden ist. Wer da den Anfgang macht, ist ja unerheblich.
Eben, es muss nicht lazy sein.
Zitat von grakaman:
Und das muss ja auch nicht die komplette App-Lebenszeit betreffen. Ich kann ja genau so ein System implementieren, dass auf ein Ereignis plötzlich komplexe Operationen macht und während dessen, sollen alle partizipierenden Objekte auf ein und die selbe Instanz zugreifen. Wenn die Operation fertig ist und ein neues Ereignis ausgelöst wird, möchte ich vielleicht ein anders initialisiertes Singleton Objekt, bei dem auch wieder alle partizipierenden Teilobjekte nur auf eine Instanz zugreifen sollen. Das ganze kann ja voll und ganz sequenziell ablaufen, dabei muss es sich doch nicht zwangsweise um irgend welche Remoteobjekte handeln, auf das mehrere Clients gleichzeitig zugreifen.
Und was machst du, wenn jemand irgendwo noch eine Referenz auf das "alte" singelton hat? Muss nicht remote sein.
Klar, in Smalltalk kannst du z.B. alle Pointer, die auf ein best. Objekt zeigen auf ein anderes Zeigen lassen wenn du bloss das referenzierte Objekt kennst. Java erlaubt dir solche Spässe nicht.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-11-30, 23:08:53   #17 (im Thread / einzeln)
grakaman
Admiral Member
 
Registriert: 2002-01-13
Beiträge: 3.847
grakaman eine Nachricht über MSN schicken
Re: Problem mit singleton unter java

Hier mal ein Zitat aus dem Buch "Enterprise Solutions Patterns using Microsoft.NET":

zu static Initialization bei Singleton:

...The only downside of this approach is that you have less control over the mechanics of the instantiaten. In the Design Pattern form, you were able to use a nonedefault constructor or perform other tasks before the instantiation..

Wie gesagt, es schreibt ja niemand vor für wie lange die Instanz erhalten sein soll, sie muss nur für einen bestimmten Zeitraum eben für alle Objekte gelten. Das kann für immer sein oder eben nur für einen bestimmten Zeitraum und danach erstellt die Software vielleicht eine veränderte Instanz, die jetzt plötzlich gelten muss. Bsp.: Ich habe einen Puffer, der bestimmte Daten sammeln soll und immer ausgeben soll. Vielleicht möchte ich ein Programm starten und zuvor den Puffer individuell anpassen (Größe, Ausgabetyp Konsole/Popup etc.). Oder es werden regelmäßig recht komplexe Operationen getriggert (sequenziell) und vor jedem Aufruf muss aber das alte Singleton gelöscht werden und ein anderes instanziert und initialisiert werden. Das ganze könnte natürlich auch parallel passieren, z.B. in verschiedenen App Domains (.NET) oder eben Prozessen. Oder man will einfach nur für einen bestimmten Zeitraum aus performancegründen die Instanzierung von mehreren gleichen Objekten verhindern. Es gibt da recht viele Anwendungsmöglichkeiten für ein Singleton, ich würde das nicht ganz so eng sehen.

Geändert von grakaman (2004-11-30 um 23:10:29 Uhr)
grakaman ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Alt 2004-12-01, 12:30:48   #18 (im Thread / einzeln)
HellHorse
Admiral Member
 
Benutzerbild von HellHorse
 
Registriert: 2003-03-27
Beiträge: 3.357
Re: Problem mit singleton unter java

Zitat von grakaman:
Hier mal ein Zitat aus dem Buch "Enterprise Solutions Patterns using Microsoft.NET":
Es geht hier weder um "Enterprise Patterns" noch um .NET
Zitat von grakaman:
zu static Initialization bei Singleton:

...The only downside of this approach is that you have less control
Natürlich ist statische Initialisierung weniger flexibel, das bestreite ich ja auch nicht. Aber gerade dadurch dass lola ja die statische Methode verwendet um auf das singleton zuzugreiffen anstatt das Feld public zu machen, behält er sich ja die Flexibilität das später zu ändern ohne dass es Änderungen in anderen Klassen erfordert.

Zitat:
.. and it’s not “oh just thread your application”. Anyone that says that is basically an idiot, not appreciating the problems.
HellHorse ist offline   Mit Zitat antworten Beitrag zum Zitieren auswählen
Antwort

Lesezeichen
  • Dieses Thema bei Twitter speichern
  • Dieses Thema bei Facebook speichern


Forumregeln
Es ist Ihnen erlaubt, neue Themen zu verfassen.
Es ist Ihnen erlaubt, auf Beiträge zu antworten.
Es ist Ihnen nicht erlaubt, Anhänge hochzuladen.
Es ist Ihnen nicht erlaubt, Ihre Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.

Gehe zu


Alle Zeitangaben in WEZ +1. Es ist jetzt 04:36:44 Uhr.


Powered by vBulletin® (Deutsch)
Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.