![]() |
Re: Verhindern eine Anwendung mehrfach zu starten
Ich persönlich würde erst garnicht soweit gehen, die Dateigröße o.Ä. zu verwenden, es reicht doch eigentlich völlig aus, wenn man einen Inviduellen String benutzt (ok, der sollte jetzt nicht nur 1 Zeichen lang sein :lol: ). Ich glaube, da sind die Wahrscheinlichkeiten, dass ein anderes Programm einen Mutex anlegt, den Wert prüft und dann auch noch den gleichen Wert hat, schon relativ klein.
|
Re: Verhindern eine Anwendung mehrfach zu starten
Ja, schätze ich auch so ein. Schön wäre es aber, wenn dieser individuelle String automatisch in der Unit erzeugt wird - die will ich ja einfach nur (ohne weitere Arbeit) in das Projekt einfügen....
// edit: Noch ein letztes Mal nachgedacht: Das Umbenennen der Exedatei ist nur kritisch, wenn das Programm nach dem ersten Start umbenannt und unter neuem Namen erneut aufgerufen würde - nur dann greift der Test ins Leere. So viel Vorsatz sollte dann auch dadurch belohnt werden, das die zweite Instanz tatsächlich gestartet wird. Mein Fazit: Ich lasse meine Unit wie sie ist :wink: |
AW: Re: Verhindern eine Anwendung mehrfach zu starten
Zitat:
|
AW: Verhindern eine Anwendung mehrfach zu starten
Ja sicher.
|
AW: Verhindern eine Anwendung mehrfach zu starten
Ich wollte die hier
![]() Carsten |
AW: Verhindern eine Anwendung mehrfach zu starten
Es lebe Unicode ... der Code war noch aus Zeiten von vor 2009 :stupid:
Hab's nicht getestet, aber ich denke mal so müsste es aussehn:
Delphi-Quellcode:
{ 604 OneInstance.pas }
SetString(S, PChar(lpData), cbData div 2); cbData:= StrLen(lpData) * 2;
Delphi-Quellcode:
Bzw. wenn man es ganz genau nimmt, dann statt
{ 149 OneInst.pas }
cbData := cbData * 2; Result := GetMemory(cbData);
Delphi-Quellcode:
besser ein
2
Delphi-Quellcode:
.
SizeOf(Char)
Aber 100% richtiger wäre es, wenn man den Code "richtig" fest auf UnicodeString/WideString, PWideChar und 2 umstellt, anstatt String, PChar und SizeOf(Char). |
AW: Verhindern eine Anwendung mehrfach zu starten
Falls das nicht funktioniert, ich nutze seit Ewigkeiten "PBOnceOnly" geschrieben von Dr. Peter Below, funktioniert bei mir tadellos.
Einfachmal nach "PBOnceOnly" hier oder per google suchen. Ich habe leider nur noch eine von mir modifizierte Version, was ich modifizierte, abgesehen vom unit namen, habe ich aus dummheit nicht notiert aber es arbeitet tadellos mit Delphi Rio zusammen. |
AW: Verhindern eine Anwendung mehrfach zu starten
|
AW: Verhindern eine Anwendung mehrfach zu starten
Ich habe eine Lösung nur für Windows.
Verwende ich seit vielen Jahren in allen Programmen die es nötig haben. Keine extra Unit, keine Komponente (wäre eh viel zu spät), keine Kopfschmerzen. Das kommt direkt in die DPR:
Delphi-Quellcode:
Natürlich sollte man den Mutex am Ende immer freigeben.
// direkt nach begin
Mutex := CreateMutex(nil, True, 'EindeutigerName'); // das sollte am besten eine selbst erzeugte Class ID sein. Dann ist "eindeutig" recht sicher if (Mutex <> 0) and (GetLastError = 0) then MutexError := false else MutexError := true; ... // alles was das programm so macht ... // direkt vorm end if Mutex <> 0 then CloseHandle(Mutex); Aber zum Glück räumt auch Windows alles weg wenn der Prozess mal unerwartet übern Jordan geht. |
AW: Verhindern eine Anwendung mehrfach zu starten
Das ist die einfachste Lösung, wenn man sicherstellen will, dass ein Programm nur einmal gestartet wird.
Was diese Lösung aber nicht kann: Die übergebenen Parameter an die erste Instanz weiterreichen, damit diese damit weiterarbeiten kann. Und genau da liegt ja aktuell das Problem. |
AW: Verhindern eine Anwendung mehrfach zu starten
Hmm..
Zitat:
Z.B. in der DPR (wie oben als Beispiel)
Delphi-Quellcode:
const
c_ProgStartMutexName = 'MeinProgrammName'; {Attempt to create a named mutex} CreateMutex(nil, false, c_ProgStartMutexName); {if it failed then there is another instance} if GetLastError = ERROR_ALREADY_EXISTS then begin {Send all windows our custom message - only our other} {instance will recognise it} SendMessage(HWND_BROADCAST, RegisterWindowMessage(c_ProgStartMutexName), 0, 0); {Lets quit} Halt(0); end; Application.Initialize; Dort, im Hauptfenster vorher:
Delphi-Quellcode:
const
c_ProgStartMutexName = 'MeinProgrammName'; var OldWindowProc : Pointer; {Variable for the old windows proc} MyMsg : Int64; {custom systemwide message} function NewWindowProc(WindowHandle : hWnd;TheMessage : LongInt; ParamW : LongInt; ParamL : LongInt) : LongInt stdcall; begin if TheMessage = MyMsg then begin {Tell the application to restore, let it restore the form} SendMessage(Application.handle, WM_SYSCOMMAND, SC_RESTORE, 0); SetForegroundWindow(Application.Handle); {We handled the message - we are done} Result := 0; exit; end; {Call the original winproc} Result := CallWindowProc(OldWindowProc, WindowHandle, TheMessage, ParamW, ParamL); end; procedure TFormMeineMainForm.FormCreate(Sender: TObject); begin //Register a custom windows message MyMsg := RegisterWindowMessage(c_ProgStartMutexName); //Set form1's windows proc to ours and remember the old window proc OldWindowProc := Pointer(SetWindowLong(FormHauptmenue.Handle, GWL_WNDPROC, LongInt(@NewWindowProc))); .. end; In diesem Beispiel wird der ersten Instanz einfach nur mitgeteilt, dass sie sich wieder nach vorne bringen soll.. Kannst aber auch eine andere Message oder IPC verwenden, um z.B. deinen Sting zu übergeben. |
AW: Verhindern eine Anwendung mehrfach zu starten
Und wo übergibst du dort die "Parameter" ?
Die beiden verlinkten Units nutzen dafür WM_COPYDATA, um quasi GetCommandLine zu übergeben, wobei dort der Speicher für den "String" (damals ANSI und jetzt Unicode) falsch vewaltet wird/wurde, weswegen es nun nicht mehr im "neuen" Delphi funktioniert(e). Man könnte es noch mit WM_SETTEXT/SendTextMessage versuchen, wo Windows notfalls die ANSI/Unicodeumwandlung vornehmen würde, falls nötig, wenn man zu doof ungeschickt ist WM_COPYDATA richtig zu verwenden. |
AW: Verhindern eine Anwendung mehrfach zu starten
Zitat:
Danke. Ich arbeite mit Named Pipes zur Kommunikation meiner Programteile untereinander. Das zuerst gestartete macht den Server auf, nach dem dann alle anderen Teile suchen. Das gleiche geht doch hier auch. Wobei natürlich PostMessage viel eleganter ist solange man an das passende Handle kommt. Die Datenübertragung erledigt eine serealisierte Memtable deren Daten als Bytestrom oder B64 String übertragen werden. |
AW: Verhindern eine Anwendung mehrfach zu starten
Hmm..
Zitat:
Zitat:
Ich würde auch nicht einfach die Parameter als Text hinschicken, sondern ggf. eine Recordstrucktur, welche dem Empfänger auch mitteilt, dass es sich um 'Parameter des Programmaufrufes' handelt. ;) |
AW: Verhindern eine Anwendung mehrfach zu starten
Hmm..
Zitat:
Dann kannst Du die Informationen direkt dorthin schicken, ist sogar besser, da nicht alle Messages mit UserData Prozess übergreifend verschickt werden. Ach ja: Kein Postmessage, sondern Sendmessage verwenden! Denn z.B. WM_COPYDATA funktioniert nur richtig damit.. |
AW: Verhindern eine Anwendung mehrfach zu starten
Zitat:
(ich wollte noch eine option haben die steuert ob programm a.exe parameter an b.exe übergibt, ob b.exe vor dem ".run" von a.exe terminiert werden soll usw... paar spielereien halt) Wie dem auch sei, das rad muss nicht neu erfunden werden, soll nicht bedeuten das die Quelltexte der anderen falsch sind, um gottes willen nein, nur mach ich mir darüber keine gedanken mehr. Lieber würde ich die PBOnceOnly.pas CrossPlatform fähig machen :-) |
AW: Verhindern eine Anwendung mehrfach zu starten
Zitat:
In dem Komponenten-Thema kann ich ja nicht schreiben, vielleicht kann es da jemand entsprechend aktualisieren? Carsten |
Alle Zeitangaben in WEZ +1. Es ist jetzt 21:15 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz