![]() |
EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Liste der Anhänge anzeigen (Anzahl: 2)
Ich würde gerne mittels EnumPrinterDrivers die Liste der installierten Druckertreiber auslesen.
Das auslesen funktioniert auch, ich bekomme die korrekte Liste doch dann haut das Programm eine Zugriffsverletzung raus. Zitat:
Ich bin leider ein wenig aus der übung und habe auch früher solche API geschichten gemieden. Diesmal würde ich aber gerne verstehen wie es geht und wo da der Fehler ist. Im Debugger wirkt es irgendwie als hätte ich zu viel Speicher reserviert.
Delphi-Quellcode:
function GetPrinterDrivers(): boolean;
var arDriverList : array of TDriverInfo2; pBuffer : Pointer; pcbNeeded : DWORD; pcReturned : DWORD; ErrorTxt : array [0..500] of char; i : integer; begin // Aufruf mit Buffersize 0 soll die benötigte Größe in pcbNeeded schreiben. // ^--------------------------------------------+ if not EnumPrinterDrivers(nil, nil, 2, pBuffer, 0, pcbNeeded, pcReturned) then begin GetMem(pBuffer, pcbNeeded); end; result := EnumPrinterDrivers(nil, nil, 2, pBuffer, pcbNeeded, pcbNeeded, pcReturned); if result then begin arDriverList := pBuffer; for i := 0 to pcReturned -1 do form1.Memo1.Lines.add(arDriverList[i].pName); FreeMem(pBuffer, pcbNeeded); end else begin FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, nil, GetLastError, 0, ErrorTxt, 500, nil); ShowMessage(ErrorTxt); end; end; |
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Spar dir den Zeiger pBuffer und nimm direkt das Array:
Delphi-Quellcode:
Der Fehler liegt wohl daran dass du nicht einfach deinen rohen Speicherbereich von x Elementen (C-Array) in dein dynamisches Delphi-Array pressen kannst. Ein Delphi-Array enthält noch mehr Informationen (Anzahl der Elemtente, Referanzzähler). Wochenendlektüre dafür gibt es hier:
function GetPrinterDrivers(): boolean;
var arDriverList : array of TDriverInfo2; //pBuffer : Pointer; pcbNeeded : DWORD; pcReturned : DWORD; ErrorTxt : array [0..500] of char; i : integer; begin // Aufruf mit Buffersize 0 soll die benötigte Größe in pcbNeeded schreiben. // ^--------------------------------------------+ if not EnumPrinterDrivers(nil, nil, 2, nil, 0, pcbNeeded, pcReturned) then // gib ihm doch einen Nullzeiger statt einen nicht initialisierten! begin //GetMem(pBuffer, pcbNeeded); SetLength(arDriverList, pcbNeeded); end; result := EnumPrinterDrivers(nil, nil, 2, @arDriverList[0], pcbNeeded, pcbNeeded, pcReturned); if result then begin //arDriverList := pBuffer; // nope for i := 0 to pcReturned -1 do form10.Memo1.Lines.add(arDriverList[i].pName); //FreeMem(pBuffer, pcbNeeded); //macht das array nach verlassen der methode selber end else begin FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, nil, GetLastError, 0, ErrorTxt, 500, nil); ShowMessage(ErrorTxt); end end; ![]() In der Regel gibst du WinAPI-Aufrufen, wenn sie einen rohen Zeiger auf einen Speicherbereich haben wollen, die Adresse deines Delphi-Arrays mit Index 0 (siehe oben: "@arDriverList[0]"). PS: Ich glaube ich habe noch nie in meinem Leben (und ich mache jetzt vier Jahre Delphi) GetMem(..)/FreeMem(..) gebraucht... |
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Herzlichen Dank!
jetzt läuft wirklich rund. Und danke für den Link :-D |
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Warum eigentlich so kompliziert? Es gibt doch die Unit Printers dafür, oder zumindest bei meinem Delphi 2007.
Delphi-Quellcode:
lg,
unit Unit1;
interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, Printers, StdCtrls; type TForm1 = class(TForm) ComboBox1: TComboBox; procedure FormCreate(Sender: TObject); private { Private-Deklarationen } public { Public-Deklarationen } end; var Form1: TForm1; implementation {$R *.dfm} procedure TForm1.FormCreate(Sender: TObject); begin combobox1.Items:=Printer.Printers; combobox1.ItemIndex:=Printer.PrinterIndex; end; end. jus |
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Zitat:
|
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Das muß auch knallen.
* pBuffer ist beim ersten Aufruf nicht initialisiert (das sollte dir auch dein Compiler um die Ohren hauen) * GetMem ist nur bei einem ganz bestimmten "Fehler" richtig * ist nichts installiert, würde cbBuf=0 passen und es gibt keinen Fehler * und dann ist arDriverList ein Delphi-Array, was absolut garnichts mit dem C-Array zu zun hat ** du "sagst" dem Compiler aber, dass pBuffer dieses Array drin ist, das zufällg so lange gut geht, bis irgendwas auf die Control-Felder (Length und Type) des Delphi-Array zugreifen will, was z.B. am Ende der Prozedur passiert, wenn der Compiler dort den Speicher des Arrays freigeben will. ** das kann maximal in den Zeiger eines statischen
Delphi-Quellcode:
gecastet werden oder man muß umkopieren, bzw. die Daten direkt in ein bestehendes dynamisches Array reinschreiben
array[0..x] of TDriverInfo2
PS: Bei SetLength sollte man natürlich die Anzahl der Records und nicht der Bytes reingeben. |
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Zitat:
|
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Zitat:
Mache ich also ein SetLength(arDriverList, pcReturned); kommt beim zweiten / finalen aufruf die Fehlermeldung Zitat:
|
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
ich bin bisher immer noch nicht schlauer.
Der erste aufruf
Delphi-Quellcode:
liefert mir nur die benötigte größe in Byte - in meinem Fall sind das 10112 Byte (18 Treiber sind installiert)
EnumPrinterDrivers(nil, nil, 2, nil, 0, pcbNeeded, pcReturned)
mit
Delphi-Quellcode:
wird der Array 10112 Einträge groß - aber alles funktioniert.
SetLength(arDriverList, pcbNeeded);
Wie kann ich anhand der zurückgelieferten Bytes die Anzahl der Array Einträge ermitteln, die ich für ein SetLength benötige? Oder kann ich dem Array of auch sagen das dieses insgesamt 10112 Bytes groß sein soll? |
AW: EnumPrinterDrivers = Zugriffsverletzung (Anfängerfrage)
Zitat:
Delphi-Quellcode:
Strukturen zusätzlich auch die Strings (die in den einzelnen Einträgen per Pointer referenziert werden) oder andere Sachen im Buffer abgelegt?
DRIVER_INFO_2
Nein: Einfach die Anzahl der Bytes durch
Delphi-Quellcode:
teilen.
SizeOf(TDriverInfo2)
Ja: In diesem Falle würde ich von Anfang an mit einem untypisiertem Buffer (
Delphi-Quellcode:
) arbeiten. In
GetMem
Delphi-Quellcode:
gibt die API dir die Anzahl der Elemente zurück, also kannst du einfach iterieren:
pcReturned
Delphi-Quellcode:
var
Buf: PDriverInfo2; .. begin // Größe ermitteln, etc .. GetMem(Buf, pcbNeeded); try if EnumPrinterDrivers(...) then begin for I := 1 to pcReturned do begin // Mach was // .. // zum nächsten Element springen Inc(Buf); end; end; finally FreeMem(Buf); end; end; |
Alle Zeitangaben in WEZ +1. Es ist jetzt 06:22 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