![]() |
Datenbankverbindung mittels ODBC (System.Data.Odbc)
Hallo zusammen,
da ich heute nachmittag selbst recht lang an einer moeglichen Loesung probiert habe, hier mein Code um eine externe oder lokale MySQL Datenbank zu verbinden, und die enthaltenen Daten der Tabelle in eine XML-Struktur zu laden. Dabei muessen zusaetzlich die folgenden Namespaces in die Uses-Anweisung eingebunden werden:
Delphi-Quellcode:
System.Xml,
System.Data, System.Data.Odbc;
Delphi-Quellcode:
Wer jetzt einen guten Tip hat um die XML-Nodes in ein Array zu bekommen,
procedure frmMain.DBConnect_cb;
var //connString: String; connector: OdbcConnection; cmd: OdbcCommand; adapter: OdbcDataAdapter; data: DataSet; xmlparser: XmlDataDocument; xmlnode: XmlNodeList; i: Integer; begin //connString := 'DRIVER={MySQL ODBC 3.51 Driver};' + // 'SERVER=localhost;' + // 'DATABASE=myDB;' + // 'UID=Benutzername;' + // 'PASSWORD=Passwort;' + // 'OPTION=3'; connector := OdbcConnection.Create(); connector.ConnectionString := 'DSN=myDSN'; connector.Open; adapter := OdbcDataAdapter.Create(); adapter.SelectCommand := OdbcCommand.Create('SELECT * FROM table1',connector); data := DataSet.Create; adapter.Fill(data); xmlparser := XmlDataDocument.Create(data); xmlnode := xmlparser.GetElementsByTagName('spalte1'); MessageBox.Show(IntToStr(xmlnode.Count)); // Anzahl der Datensaetze data.Free; adapter.Free; connector.Close; end; sodass man die Daten bearbeiten kann, waere eine sample Application bereits komplett :-) Gruss, MrZweig |
Re: Datenbankverbindung mittels ODBC (System.Data.Odbc)
Ich glaube da fehlt noch etwas "thinking ADO.Net". ;)
Du hast doch deine Daten in ein DataSet geschrieben. Was hindert dich daran, es zu bearbeiten? Zum Speichern in eine XML-Datei kannst du ganz easy
Delphi-Quellcode:
verwenden.
DataSet.WriteXML([DateiName]);
Hast du Constrains oder Verknüpfungen in deinem DataSet, würde ich mit
Delphi-Quellcode:
... das Schema als *.xsd speichern und erst danach mit ...
DataSet.WriteXmlSchema([DateiName]);
Delphi-Quellcode:
... die Daten als *.xml.
DataSet.WriteXml([DateiName], XmlWriteMode.IgnoreSchema);
Laden kannst du ein Schema mit:
Delphi-Quellcode:
Bei nur einer Tabelle klappt das absolut problemlos:
DeinDataSet.ReadXmlSchema([DateiName]);
DeinDataSet.ReadXml([DateiName], XmlReadMode.IgnoreSchema);
Delphi-Quellcode:
DeinDataSet.ReadXml([DateiName]);
|
Re: Datenbankverbindung mittels ODBC (System.Data.Odbc)
Hall Robert,
Deine Gedanken dazu sind sehr hilfreich und richtig :-) Allerdings will ich im Prinzip, die Daten aus dem DataSet primaer in einem Array "behandeln", um diese zum Beispiel weiterzuverarbeiten. Der Weg ueber XML ist mehr eine "hilfsmethode", da ich nicht so recht nachvollziehen kann, wie ich die Daten aus dem DataSet Objekt in ein Arrayobjekt ueberfuehren kann. Hierzu vielleicht eine Anregung? Gruss, Sascha aka MrZweig |
Re: Datenbankverbindung mittels ODBC (System.Data.Odbc)
Das ist eigentlich ziemlich easy (wie fast alles unter .Net ;) )
Delphi-Quellcode:
Nachtrag:
var
enumTable :IEnumerator; dtrEnum :DataRow; begin // hole dir einen Enumerator für die Einträge der Tabelle enumTable := DeinDataSet.Tables['DeinTabelle'].Rows.GetEnumerator; // Zurücksetzen des Enumerators enumTable.Reset; // Iteriere durch den Enumerator while enumTable.MoveNext do begin // Typecase of DataRow dtrEnum := enumTable.Current as DataRow; // Mache irgendwas mit der DataRow.... end; end; Willst du wirklich alles in einem Array haben:
Delphi-Quellcode:
var
DeinArray :Array of DataRow; begin setLength(DeinArray, DeinDataSet.Tables['DeineTabelle'].Rows.Count); DeinDataSet.Tables['DeinTabelle'].Rows.CopyTo(DeinArray, 0); end; |
Re: Datenbankverbindung mittels ODBC (System.Data.Odbc)
Ah wunderbar :)
Allerdings hinkt es bei mir an folgender Stelle, die ich nicht ganz nochvollziehen kann: Wenn ich vorher mit
Delphi-Quellcode:
das DataSet fuelle, sind saemtliche Rows darin enthalten (das
adapter := OdbcDataAdapter.Create();
adapter.SelectCommand := OdbcCommand.Create('SELECT * FROM table1',connector); data := DataSet.Create; adapter.Fill(data); sehe ich anhand der XML Ausgabe). Wenn ich aber nun mittels
Delphi-Quellcode:
die Daten versuche in das Array zu schieben, bekomme ich eine NullReference-
data.Tables['table1'].Rows.CopyTo(Array, 0);
Exception (uebrigens fast die einzige Compilermeldung mit der ich irgendwie immer in Konfrontation gerate ;). Es sieht so aus als ob ich in dem DataSet den Tablenamen nicht erreichen kann, weil ich im SELECT bereits die Daten einer Tabelle auslese. Denkfehler meinerseits? UPDATE: Ich habe jetzt mal data.Tables[0] benutzt, damit geht es dann auch. Aber eigentlich sollte man doch auch ueber den Tablenamen die Daten abrufen koennen. Etwas merkwuerdig. Schoenen Abend, Sascha |
Re: Datenbankverbindung mittels ODBC (System.Data.Odbc)
:oops: Ich hatte nicht bemerkt, dass du geantwortet hast.
Das hier habe ich eben in C# getippt (Sorry ich hatte gerade nicht genug Nerven für D8 ;) ) Ersetze das Wort "Oracle" mit "ODBC", die Tabelle mit deiner und den Primärschlüssel mit dem aus deiner Tabelle -> schon müsste es laufen. :angle2: Ich persönlich finde einen Array in Sachen Datenbanken in .Net vollkommen fehl am Platz. Selbst mit DataTable.LoadDataRow kann man nicht wirklich das absolut geniale DataSet.Merge nachbilden...
Code:
void ButtonClick(object sender, System.EventArgs e)
{ OracleDataAdapter daSelData = new OracleDataAdapter(); daSelData.SelectCommand = new OracleCommand("SELECT *"+ '\n'+ "FROM Scott.Emp t" ,connUser); DataTable tblTemp = new DataTable("Emp"); //Hier unnötig, da wir die Tabelle bereits benannt erzeugen //daSelData.TableMappings.Add(new DataTableMapping("Table1", "Emp")); tblTemp .BeginLoadData(); try { daSelData.Fill(tblTemp); } finally { tblTemp.EndLoadData(); } // der PrimKey verhindert das Einfügen doppelter Datensätze // Mit DataSet.Merge sind dadurch noch ganz andere Spielereien möglich ;) tblTemp.Constraints.Add(new UniqueConstraint("primKeyEmpNo" ,tblTemp.Columns["EMPNO"] ,true)); if (!DeinDataSet.Tables.Contains(tblTemp.TableName)) { // legt die Tabelle im ZielDataSet an, falls sie nicht schon existiert, durch Clone bekommt sie alle Spalten und Constraints, // aber KEINE Daten DeinDataSet.Tables.Add(tblTemp.Clone()); } // der folgende Code könnte durch diese EINE Zeile ersetzt werden // DeinDataSet.Merge(dsTemp, false); DataRow[] RowsArray = (DataRow[])Array.CreateInstance(typeof(DataRow) ,tblTemp.Rows.Count); tblTemp.Rows.CopyTo(RowsArray, 0); foreach (DataRow dtrArray in RowsArray) { // Bei LoadDataRow wird durch den PrimKey auch ein doppeltes Einfügen verhindert. // Eine wirkliche Synchronisierung wie bei Merge klappt damit nicht. :/ DeinDataSet.Tables[tblTemp.TableName].LoadDataRow(dtrArray.ItemArray, true); } } |
Alle Zeitangaben in WEZ +1. Es ist jetzt 11:08 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