DataTable ou DataAdapter et update/insert -goto en C# oo- [.NET] - C#/.NET managed - Programmation
Marsh Posté le 10-07-2006 à 21:05:50
Youpi tralala !
Ca marche
Code :
|
Merci bien à ce lien :
http://samples.gotdotnet.com/quick [...] romDB.aspx
--edit--
1/ Mise à jour du code : maintenant, seules les lignes non vides sont ajoutées. Deplus, seules les lignes réellement modifiées sont mises à jour.
2/ Nouvelle mise à jour : maintenant on peut aussi supprimer des lignes
Marsh Posté le 12-07-2006 à 14:48:22
Bon, je me permet de remonter ce topic en réouvrant la question, puisque ma solution actuelle n'est pas satisfaisante.
En effet, retrouver les lignes dans mon datatable via leur "position" n'est pas satisfaisant, dans la mesure où, pendant que je modifie la ligne 2 par exemple, une personne peut ajouter une nouvelle ligne, qui, à cause de ma clause de tri, va se trouver en position numéro deux.
Ainsi, quand je vais soumettre la requête "modifier la ligne 2", je vais modifier la mauvaise ligne.
J'ai vu une propriété "primaryKey" dans datatable. Vu qu'on est obligé d'avoir une PK dans une datatable, à priori c'est une source plus fiable.
Cependant, j'ai toujours le souci de généricité. Ainsi, la clé peut être composées ou autre, ce qui ne facilite en rien mon truc...
Si vous voyez une meilleur solution (ou avez une idée précise de comment implémenter cette dernière, je suis preneur)
Marsh Posté le 12-07-2006 à 15:32:18
En fait, tu veux recoder l'Enterprise Manager de SQL Server ?
Déjà, le fait que tu utilises des string ça t'empêche de considérer toutes les clefs primaires possibles, donc niveau généricité, ça marchera que dans 99% des cas. (mais c déjà pas mal
Les clefs primaires c bien : ca t'identifie une ligne de façon unique.
Si tu as plusieurs clefs primaires, bah il faut faire un système pour les passer toutes en paramètres pour identifier tes lignes.
Pour éviter de faire une using à gaz, je collerai la gestion de la clef primaire dans un object spécialisé (DataIdentifier ?) qui gère les clefs simples et composées de façon transparent au niveau de ton code 'fonctionnel'.
Cet objet à pour but de lire les clefs primaires d'un DataRow et les coller dans une chaine à la demande, et vice-versa. Avec une redéfinition de Equals tu peux aussi vérifier l'égalité de deux lignes, quel que soit le nombre de clefs qui le compose.
Chui clair ou pas ?
Marsh Posté le 12-07-2006 à 15:46:36
Dans mon flux XML, je récupère le type et les infos du type. Donc y'a pas de souci, je sais dans tous les cas recomposer ma clé.
Par contre, je viens de me rendre compte d'un truc très con : "primaryKey" de mon datatable est vide. Il est nécessaire d'avoir une PK dans la requête qui l'allimente, mais il n'est pas foutu de le retrouver dans l'objet comment faire
Marsh Posté le 12-07-2006 à 15:47:47
sinon, une clé primaire, par définition, y'en a qu'une par table, après ce sont des clés alternatives si je fais le choix d'utiliser la pk je n'ai pas à me soucier de savoir combien il y a de clé
Marsh Posté le 12-07-2006 à 21:37:14
Ca y est !
J'ai donc trouvé comme mettre à jour selon la PK.
Ce fût quelque peut difficile, mais finalement c'est plutôt simple
Pour récupérer la PK dans l'objet DataTable, il suffit de faire un "FillSchema()" à partir du DataAdapter.
Pour le reste, je vous laisse contempler le bordel
Edit :
Correction des bugs, et surtout, rechargement des données en cas de modification. Désolé pour le goto --
Edit :
Correction d'un nouveau bug : Si la requête ne retourne pas de PK identifiable, alors le .FillSchema() vide la DataTable. Il faut donc ensuite la recharger sans schema
Cette correction est largement améliorable (ici, on doit obligatoirement faire deux fois la requête, avec toutes les requêtes imbriquées !). Mais ça demande la duplication d'une grande partie du code...
Code :
|
Marsh Posté le 12-07-2006 à 22:14:43
je suis certain que j'arriverais à faire la même chose en faisant 5x moins de ligne. Il y a une technique bien particulière pour cela. Traiter déjà des event sur le dataset. Après, ça fait tout à fait la même chose que tu fais la, en plus facile et en plus cours
Marsh Posté le 12-07-2006 à 22:31:29
yep, sauf que moi je ne bosse pas en mode connecté et j'ai pas de databind, j'ai pas de viewstate ni rien, donc pour les event...
Marsh Posté le 12-07-2006 à 22:33:11
ceci dit, il y a pas mal d'optimisations à faire. si tu te sens le courrage, je suis preneur d'optimisations (le premier, ce serait de faire un Request.Params["toto%"] afin d'éviter de tous me les palucher un peu partout... Là les boucles imbriquées qui servent à rien, y'en a une trippoté
Marsh Posté le 12-07-2006 à 23:14:51
Groumpf...
Je viens de m'appercevoir que pour les cas tordus, genre je met à jours les données d'une table liée, pour chaque ligne de la table principale, je vais avoir un truc très con : pas capable de retrouver la PK correspondant à chaque champ
Bon, là je suis mal, y'a plus qu'à réécrire une grande partie du code... Dans le principe, ça reste pareil, sauf qu'il faut que j'inclue la PK dans le nom du champ si je veux pouvoir le retrouver
Edit :
Ceci dit, je me rend compte aussi que pour les INSERT, je suis encore plus dans la merde dans le cas d'une table liée... Obligé dans tous les cas de passer par un JS pour pas avoir d'embrouille... Et là je ne vois pas de solution simple pour faire une correction qui colle...
Du coup, une solution JS est possible pour le coup du INSERT et une autre, côté XSL, pour le UPDATE et DELETE... du coup ben... Ouais, en fait c'est bien comme ça
Marsh Posté le 13-07-2006 à 00:12:18
FillSchema, c'est trop trop trop con comme fonction
Non seulement ça vite le DT s'il est vide, mais en plus, si je filtre par la PK lors de la requête, il doit décider que l'information de PK est inutile, donc il la vire
Marsh Posté le 13-07-2006 à 01:05:33
c'est même pire que tout...
si je met id = 1 en dur, j'ai bien une pk
si je met id = @idval alors ça marche plus, y'a plus de pk
en fait, il me vire le schema si un paramètre porte sur une composante de la pk
Marsh Posté le 13-07-2006 à 01:06:33
à moins que ce ne soit carrément dès que j'ai un paramètre...
ouais, en fait ça c'est. dès que j'utilise un paramètre à ma requêtes, j'ai plus d'info de pk...
Marsh Posté le 13-07-2006 à 09:27:19
Arjuna a écrit : yep, sauf que moi je ne bosse pas en mode connecté et j'ai pas de databind, j'ai pas de viewstate ni rien, donc pour les event... |
ma technique consisterait également à travailler en mode déconnecté. Bien entendu, j'aurais besoin du viewstate pour conserver les data d'un passage à l'autre.
Cependant, je ne l'ai jms implémenté en Web, donc finalement, ça deviendrait peut etre marteau burin aussi...
Marsh Posté le 13-07-2006 à 10:25:40
ouais, chais pas trop.
d'un autre côté, mon truc est récursif, on peut donc avoir n requêtes imbriquées et m requêtes à la queue leu-leue. Du coup, pour retrouver quelque chose dans le viewstate... j'ai préféré écarter cette option dès le départ, jugeant les problèmes d'accès mémoire trop compliqués par rapport à réxécuter les requêtes.
de plus, à la sortie, je crée un flux XML que je transforme en XSL, je n'ai donc pas la possibilité de lier un contrôle à chaque datatable comme on ferait avec des datarepeater.
dans tous les cas, ça ne résoud pas mon problème de fillschema qui ne marche pas quand j'ai un paramètre dans ma requête...
Marsh Posté le 13-07-2006 à 13:23:09
Et le goto , tu l'as enlevé ?
Pourquoi tu découpes pas ton code en fonctions ? Ca te simplifiera la vie méchamment !
Les méthodes 'usines à gaz' de 200 lignes de long c'est super chiant à maintenir, tu passes des lustres à essayer de re-comprendre ce que tu as fait quand tu les reprends un mois plus tard, tu factorises aucun traitement et du coup tu multiplies les copier-coller là ou tu pourrais l'éviter, etc...
Vive les méthodes privée !
Marsh Posté le 13-07-2006 à 14:26:35
=> dans le dernier code que j'ai posté en date, non, il est toujours présent le gogo.
=> pour les fonction, je suis d'accord que ça deviens un peu usine à gaz. ceci dit, c'est carrément du oneshot à chaque fois : même si parfois je fais deux fois un traîtement similaire, il y a suffisament de différences pour ne pas le rendre factorisable. du coup j'ai tout laissé en pense de vache, mais je ne vois pas comment faire mieux, du moins si je veux conserver le peu d'optimisations qu'il y a dans le code...
sinon, voici ma bidouille pour contourner le problème du fillschema... d'un côté, j'ai fortement optimisé le code, et surtout, il n'y a plus de goto... mais d'un autre côté, là je lance systématiquement la requête une première fois sans filtre ! du coup je vous raconte pas les perfs si y'a beaucoup de ligne
j'espère qu'une bonne âme voudra bien me donner une explication de ce "bug" de fillschema, afin de retrouver un semblant de performances dans mon truc...
du coup, j'en suis à me demander si ce ne serait pas plus simple de tout charger, et de faire les filtres dans le DataTable au lieu de les faire en SQL. A priori, ça serait pas moins performant
Voilà le code complet qui marche.
Y'a plus de bugs. Et je peux mettre à jour n'importe quelle requête du moment qu'il n'y a pas d'agrégations dedans (ni de jointures complexes, ça va de soit)
Code :
|
Marsh Posté le 13-07-2006 à 16:27:46
Suite à la correction de ce problème :
http://forum.hardware.fr/forum2.ph [...] 0#t1406039
Maintenant :
- ça marche
- c'est rapide
- c'est super optimisé (autant faire se peut)
Marsh Posté le 10-07-2006 à 19:36:26
Bonjour,
A la façon d'un RecordSet avec ADODB, je voudrais pouvoir modifier/ajouter des données dans un DataTable retourné par un "select ...".
Voici mon code pour le moment (ça ne fait que le select) :
Je ne connais pas la structure de la table lue. Cependant, j'utilise SQL Server, qui permet tout à fait de modifier des données retournées dans un select (via des mises à jours sur curseur).
Je dois modifier à la fois les données de mon DataTable, ainsi que celle de la table lue.
Message édité par Arjuna le 12-07-2006 à 22:29:48