La presentazione è in caricamento. Aspetta per favore

La presentazione è in caricamento. Aspetta per favore

Da VB a VB.NET Andrea Saltarello [IDevGuro] Corrado Cavalli [DevGuro(Of T)]

Presentazioni simili


Presentazione sul tema: "Da VB a VB.NET Andrea Saltarello [IDevGuro] Corrado Cavalli [DevGuro(Of T)]"— Transcript della presentazione:

1 Da VB a VB.NET Andrea Saltarello [IDevGuro] Corrado Cavalli [DevGuro(Of T)]

2 Sponsor

3 Golden migration rule....Non migrare!.Non migrare! VB.NET è un nuovo linguaggio Il wizard fa del suo meglio ma non fa i miracoli

4 Scelta di un Project Template Windows Application Class Library Windows Control Library ASP.NET Web Application / Service / Control Library Console Application Windows Service Smart Device App Office App (VSTO)

5 Overview del.NET Framework Visual Studio.NET Common Language Specification VisualBasicC++C#J#….NET Framework

6 Struttura di un Progetto Solution Files (.sln,.suo) Project Files (.vbproj) Local Project Items Classi, Form, Moduli, ecc. (.vb) Web Project Items XML Web services (.asmx) Web Form (.aspx) Global application classes (.asax)

7 Parte #1 Il linguaggio Visual Basic.NET

8 Definire classi In VB6, una classe: è un file.cls Ha come nome il nome del file (es: Customer.cls) In VB.NET, una classe: È un blocco di codice Class…End Class In un unico file ci possono essere più classi

9 VB.NET=OOP! Class Car … End Class Class Car … End Class File Class1.vb Public Class Car Public Sub Start() Public Sub Start() End Sub End Sub End ClassVB 2005 File Class2.vb Public Partial Class Car Public Sub Stop() Public Sub Stop() End Sub End Sub End Class VB 2005 Dim T As New Car() T.Start() T.Stop() VB 2005

10 Uso dei Costruttori Sub New rimpiazza Class_Initialize Viene eseguita quando loggetto è istanziato Possiamo specificare più costruttori per la stessa classe (Overload) Public Sub New( ) intValue = 1 End Sub Public Sub New(ByVal i As Integer) intValue = i End Sub

11 Distruzione degli oggetti Sub Finalize rimpiazza levento Class_Terminate La usiamo per liberare le risorse usate dalloggetto Viene eseguita quando il GC distrugge loggetto Usiamo x = Nothing per permettere la Garbage Collection Attenzione: la distruzione potrebbe non avvenire immediatamente! Protected Overrides Sub Finalize( ) Rilascio delle risorse End Sub

12 Dichiarazione di Funzioni Stessa Sintassi di Visual Basic 6.0 Una classe può contenere metodi con stesso nome e signature differente una alternativa ai parametri Optional più robusti: errori a compile time Non deve esistere ambiguità I metodi non possono differire solo per il tipo di ritorno o argomenti opzionali Public Function ToInt(ByVal o as Object) As Integer End Sub Public Function ToInt(ByVal s as String) As Integer End Sub

13 Dichiarazione di Proprietà Sintassi differente da quella di Visual Basic 6.0 Keyword ReadOnly e WriteOnly Public ReadOnly Property MyProperty( ) As Integer Get Return _MyProperty End Get End Property Public Property MyData( ) As Integer Get Return intMyData 'Restituisce una variabile locale End Get Set (ByVal Value As Integer) intMyData = ValueMemorizza il valore in una variabile End Set End Property

14 VB.NET=Ereditarietà Classi con funzionalità e interfacce molto simili ad un altra Modifica della classe (errori,interfaccia non congrua) Copia e Incolla Il codice potrebbe non essere disponibile Eredità per delega (favour object composition over inheritance – cit.) Parecchio codice per il wrapping della classe interna Car SuperCar Start Start Faster

15 Ereditarietà Creare una classe che eredita da un altra Eredita Interfaccia e Implementazione Inherits Non è necessario Source Code Stesso linguaggio (C# VB.NET) Dim sc as New SuperCar sc.Start()Sc.Faster() Public Class SuperCar Inherits Car Inherits Car End Class Public Class SuperCar Inherits Car Inherits Car Public Sub Turbo Public Sub Turbo End Sub End Sub End Class Dim sc as New SuperCar sc.Start()Car sc.Faster()Car sc.Turbo()SuperCar

16 Ereditarietà Concetti La classe da cui si eredita è definita classe Base Posso ereditare da qualsiasi classe.NET (che lo permette) Tutte le classi in.NET ereditano da System.Object Si definisce una relazione Is a (é un) In.NET tutto è Object Dim c as Car= New SuperCar c.Start() c.Turbo()Errore Public Class SuperCar Inherits Car Inherits Car Public Sub Turbo Public Sub Turbo End Sub End Sub End Class Una classe può ereditare da una sola classe base Single Inheritance

17 Override La classe che eredita può modificare il comportamento della classe base Ridefinendone i metodi/proprieta/eventi La classe base deve permetterne la redifinizione Overridable Public Class Car Public Overridable Sub Faster Public Overridable Sub Faster Console.WriteLine(base) Console.WriteLine(base) End Sub End Sub End Class Public Class SuperCar Inherits Car Inherits Car Public Overrides Sub Faster Public Overrides Sub Faster Console.WriteLine(Inherited) Console.WriteLine(Inherited) End Sub End Sub End Class Dim c as New SuperCar c.Faster()Inherited

18 Override Un metodo dichiarato Overrides: E automaticamente Overridable Si può usare NotOverridable per modificarne il comportamento Public Class SuperCar: Inherits Car Public NotOverridable Overrides Sub Faster Public NotOverridable Overrides Sub Faster Console.WriteLine(Inherited) Console.WriteLine(Inherited) End Sub End Sub End Class MyBase Rappresenta la classe da cui si sta ereditando Public Class SuperCar: Inherits Car Public Overrides ReadOnly Property Model as String Public Overrides ReadOnly Property Model as String Return MyBase.Model + Super Return MyBase.Model + Super End Sub End Sub End Class

19 Override I costruttori non vengono ereditati No problem se non avete dei costruttori parametrici (usa default) Dovete invocarli esplicitamente (prima istruzione) Public Class SuperCar: Inherits Car Public Sub New (cilindrata as Integer) Public Sub New (cilindrata as Integer) MyBase.New(cilindrata)Inizializzo la classe base MyBase.New(cilindrata)Inizializzo la classe base End Sub End Sub End Class Public Class Car Public Sub New (cilindrata as Integer) Public Sub New (cilindrata as Integer) End Sub End Sub End Class

20 Public Class Car Private Sub IncreaseSpeed Me.Faster() Private Sub IncreaseSpeed Me.Faster() End Sub End Sub End Class Override, classe base e Me Quando associate un istanza di una classe ereditata ad una variabile di tipo base, la classe base vede limplementazione dei metodi della classe ereditata Public Class Car Public Overridable Sub Faster Public Overridable Sub Faster Console.WriteLine(base) Console.WriteLine(base) End Sub End Sub End Class Public Class SuperCar: Inherits Car Public Overrides Sub Faster Public Overrides Sub Faster Console.WriteLine(Inherited) Console.WriteLine(Inherited) End Sub End Sub End Class Dim c as Car= New SuperCar c.Faster()Inherited Me non indica il contesto corrente

21 Public Class Car Private Sub IncreaseSpeed MyClass.Faster() Private Sub IncreaseSpeed MyClass.Faster() End Sub End Sub End Class MyClass MyClass permette di specificare che il contesto di esecuzione sia quello della classe base e non quello di eventuali overrides in classi ereditanti da essa Viene eseguito il metodo Faster della classe Car Non disponibile in C#

22 Public Class SuperCar: Inherits Car Public Shadows Sub Faster Public Shadows Sub Faster Console.WriteLine(Inherited) Console.WriteLine(Inherited) End Sub End Sub End Class Shadows Overrides funziona se il metodo della classe base è virtual Potrebbe non esserlo (I metodi Overridable sono più lenti) Potremmo non poterlo modificare Shadows Permette di ridefinire metodi già presenti nella classe base Sono a tutti gli effetti delle nuove implementazioni Sono ignorate dalla classe base (no ereditarietà) Va usato per ridefinire membri Shared Dim c as Car= New SuperCar c.Faster()base In alcuni casi potremmo non volere che qualcuno erediti la nostra classe

23 NotInheritable e MustInherit NotInheritable Indica che la classe non è ereditabile (sealed class) Public NotInheritable Class Car End Class Public Class SuperCar Inherits Car Errore Inherits Car Errore End Class MustInherit Non può essere istanziata (abstract class) Può essere usata solo ereditando da essa Spesso contiene dei metodi definiti MustOverride Permette di definire delle classi che hanno funzionalità parziale Public MustInherit Class Car Public MustOverride Sub Faster Public MustOverride Sub Faster End Sub End Sub End Class

24 Classe Astratta: CollectionBase System.Collection.CollectionBase Permette di creare collezioni tipizzate Public Class Cars : Inherits CollectionBase Public Sub Add(ByVal item As Car) Public Sub Add(ByVal item As Car) MyBase.InnerList.Add(item) MyBase.InnerList.Add(item) End Sub End Sub Default Property Item(ByVal index As Integer) As Car Default Property Item(ByVal index As Integer) As Car Get Get Return DirectCast(MyBase.InnerList(index), Car) Return DirectCast(MyBase.InnerList(index), Car) End Get End Get Set(ByVal Value As Car) Set(ByVal Value As Car) MyBase.InnerList(index) = Value MyBase.InnerList(index) = Value End Set End Set End Property End Property End Class Dim l as New Cars() l.Add(new Car)ok l.Add (New Truck) Errore

25 Override di eventi La classe base deve essere strutturata in maniera opportuna Public Class Car Public Event LowOil() Public Event LowOil() Public Overridable Sub Faster Public Overridable Sub Faster If (Me.OilLevel<5) Then Me.OnLowOil() If (Me.OilLevel<5) Then Me.OnLowOil() End Sub End Sub Protected Overridable Sub OnLowOil() Protected Overridable Sub OnLowOil() RaiseEvent LowOil() RaiseEvent LowOil() End Sub End Sub End Class Public Class SuperCar: Inherits Car Protected Overrides Sub OnLowOil() Protected Overrides Sub OnLowOil() Me.Signal(Low oil) Me.Signal(Low oil) MyBase.OnLowOil() MyBase.OnLowOil() End Sub End Sub End Class

26 VB Generics Permettono di definire dei tipi generici che verrano completati a compile time Public Class Car Private _Navigator As T Public ReadOnly Property Navigator As T Get Return _Navigator Return _Navigator End Get End Class Placeholder Public Class Car(Of T) Private _Navigator As T Public ReadOnly Property Navigator As T Get Return _Navigator Return _Navigator End Get End Class Public Class TomTom Public Sub RouteTo():End Sub Public Sub RouteTo():End Sub End Class Public Class Garmin Public Sub NavigateTo():End Sub End Class Dim Car1 As New Car(Of TomTom) Dim Car2 As New Car(Of Garmin) Car1.Navigator.

27 VB 2005 – Generics Collections Public Class List Private elements() As Object Private elements() As Object Private count As Integer Private count As Integer Public Sub Add(ByVal element As Object) Public Sub Add(ByVal element As Object) If count = elements.Length Then Resize(count * 2) If count = elements.Length Then Resize(count * 2) elements(count) = element : count += 1 elements(count) = element : count += 1 End Sub End Sub Public Default Property Item(ByVal Index _ Public Default Property Item(ByVal Index _ As Integer) As Object As Integer) As Object Get Get Return elements(index) Return elements(index) End Get End Get Set (value as object) elements(index) = value Set (value as object) elements(index) = value End Set End Set End Property End Property End Class Dim intList As New List() intList.Add(1) Boxing! intList.Add(2) Boxing! intList.Add(Pippo") No error… intList.Add(Pippo") No error… i = CInt(intList(0)) Richiesto cast Public Class List(Of T) Private elements() As T Private elements() As T Private count As Integer Private count As Integer Public Sub Add(ByVal element As T) Public Sub Add(ByVal element As T) If count = elements.Length Then Resize(count * 2) If count = elements.Length Then Resize(count * 2) elements(count) = element : count += 1 elements(count) = element : count += 1 End Sub End Sub Public Default Property Item(ByVal Index _ Public Default Property Item(ByVal Index _ As Integer) As T As Integer) As T Get Get Return elements(index) Return elements(index) End Get End Get Set (value as T) elements(index) = value Set (value as T) elements(index) = value End Set End Set End Property End Property End Class Dim intList As New List(Of Int32) intList.Add(1) No Boxing! intList.Add(2) No Boxing! intList.Add(Pippo") Error… intList.Add(Pippo") Error… i = intList(0) Non serve castare

28 Generics Collections Collection Classes List (of T) Dictionary (Of K,V) SortedDictionary (Of K,V) Stack(Of T) Queue(Of T) Collection Interfaces List (of T) IDictionary (Of K,V) ICollection (Of T) IEnumerable (Of T) IEnumerator (Of T) IComparable(Of T) IComparer(Of T) Collection Base Classes Collection (Of T) KeyedCollection (Of T) ReadOnlyCollection(Of T) Public Class TypedCollection(Of T) _ Inherits System.Collections.ObjectModel.Collection(Of T)... End Class Dim Coll as New TypedCollection(Of DataTable) Coll.Add(new DataTable()) Dim coll As New List(Of DataTable)

29 VB 2005 – Generics Methods Dim X As Int32 = 3 Dim Y As Int32 = 4 Dim Z As Int32 = IIf(X>Y,X,Y) Error Dim Z As Int32 = MyIIF(Of Int32)(X>Y,X,Y) No cast! Dim Z As Int32 = MyIIF(X>Y,X,Y) Type inferencing Public Function MyIIF (Of T)(cond As Boolean, _ trueP As T, falseP As T) As T If (cond) Then Return trueP If (cond) Then Return trueP Else Return falseP End If Else Return falseP End If End Sub

30 VB 2005 – Generics Constraints Public Class Car(Of T) Private _Navigator As T Public ReadOnly Property Navigator As T Get Return _Navigator Return _Navigator End Get End Class Dim Car1 As New Car(Of TomTom) Car1.Navigator.RouteTo() Error Public Class Car _ (Of T As {New,BaseClass,I1,I2,Class,Structure}) Private _Navigator As New T Public ReadOnly Property Navigator As T Get Return _Navigator Return _Navigator End Get End Class Public Class Car(Of T) Private _Navigator As New T Public ReadOnly Property Navigator As T Get Return _Navigator Return _Navigator End Get End Class Public Class Car(Of T As New) Private _Navigator As New T Public ReadOnly Property Navigator As T Get Return _Navigator Return _Navigator End Get End Class

31 VB.NET: Approfondimenti Da VB a VB.NET, Andrea Saltarello, workshop 14/11/2001 workshop 14/11/2001workshop 14/11/2001 Language Revolution, Corrado Cavalli e Raffaele Rialdi, Whidbey workshop Whidbey workshopWhidbey workshop An Introduction to Visual Basic 2005, Jay Roxe e Sean Draine, Whidbey (Reloaded) workshop Whidbey (Reloaded) workshopWhidbey (Reloaded) workshop

32 Classi Attributo Le classi attributo: dal punto di vista dellutilizzatore, sono espressioni utili per decorare porzioni di codice Dal punto di vista del realizzatore, sono classi che specializzano System.Attribute E raccomandato luso del suffisso Attribute nella definizione del nome delle suddette classi Il suffisso può essere omesso in fase di decorazione del codice Public Class ReportManager _ _ Public Sub PrintLabel() Public Sub PrintLabel() PrintLabel(1) PrintLabel(1) End Sub End Sub Public Sub PrintLabel(ByVal copies As Integer) Public Sub PrintLabel(ByVal copies As Integer) End Sub End Sub End Class

33 Attributi: approfondimenti Componenti "Alternativi": le classi Attributo, Andrea Saltarello, workshop Component Development workshop Component Developmentworkshop Component Development

34 Parte #2 VB.NET vs. Windows Forms

35 VB6 Aggiunta di un file.res al progetto Utilizzo del resource editor Localizzare testo usando LoadResStringLoadResPictureLoadResData No local/region aware Difficile integrare culture diverse Difficile estendere la localizzazione No locale adaptive UI Localizzazione in VB6

36 VB.NET Si appoggia su files xml (.resx) ResGen.exe.resx ->.resources ResourceManager Cerca le assembly satellite utilizzando la cultura della UI Probing Neutral / Default culture ToolsWinRes.exe Lutz Roeders Resourcer (http://www.aisto.com/roeder/dotnet/) Al.exe Localizzazione in.NET

37

38 Localizzazione: Approfondimenti Applicazioni Windows Forms localizzate, Corrado Cavalli, articolo. articolo

39 Applicazioni MDI Proprietà IsMdiContainer = True per promuovere un form allo stato di contenitore MDI Nessun limite sul numero di form MDI per applicazione Proprietà MdiParent = mdiform per incastrare il form MDI child un form MDI può ospitare *qualsiasi* controllo sulla sua superficie

40

41 VB6: SetWindowLong (GWL_WNDPROC ) SubClassing in VB6 Private Sub Form_Load() _OldWindowProc = SetWindowLong(Me.hwnd, GWL_WNDPROC, _ AddressOf NewWindowProc) AddressOf NewWindowProc) End Sub Public Function NewWindowProc(ByVal hwnd As Long, ByVal msg As Long, ByVal wParam As Long, lParam As Any) As Long... NewWindowProc = CallWindowProc( _ OldWindowProc, hwnd, msg, wParam, lParam) End Function Private Sub Form_Unload(Cancel As Integer) SetWindowLong _ hwnd, GWL_WNDPROC, _ OldWindowProc SetWindowLong _ hwnd, GWL_WNDPROC, _ OldWindowProc End Sub

42 Svantaggi IDE Molto fragile Debug difficile Facile sbagliare AlternativeMsgHookMsgBlaster SubClassing in VB6

43 La classe Control espone WndProc (protected) Alla WndProc viene passata la struttura Message E possibile intercettare i messaggi, modificarli e/o passarli alla classe base NativeWindow Permette il subclassing quando non si ha accesso alla WndProc Permette il riutilizzo del subclassing IMessageFilter Permette di intercettare messaggi prima che vengano inviati a Form/Controllo Application.AddMessageFilterApplication.RemoveMessageFilter SubClassing in.NET

44

45 Alcuni controlli (es: Menu, ListBox e ComboBox) espongono la proprietà DrawMode, che può essere impostata a: NormalOwnerDrawFixedOwnerDrawVariable L'evento DrawItem riceve un riferimento allelemento (es: lindice) e un oggetto Graphics Controlli Owner drawn

46

47 Delegate Le delegate sono dei puntatori a funzione: Managed. Non abbiamo in mano un vero puntatore a funzione, ma un oggetto in grado di rintracciarlo. Il GC non ci preoccupa: CLR e FX sono in combutta e mantengono valido il puntatore type-safe. Le funzioni devono rispettare la firma della delegate

48

49 Multithreading Il framework offre la classe Thread, che accetta in ingresso una istanza della delegate ThreadStart

50

51 Asyncronous Programming Il cliente vuole applicazioni veloci Ottimizzare la velocità apparente Collo di bottiglia... Attendere prego... Applicazione Freezed Application.DoEvents() Eseguire loperazione in un thread separato Non è consentito accedere a controlli Windows da threads diversi da quello in cui il controllo è stato creato Gli unici metodi thread-safe di un controllo windows sono InvokeBeginInvokeEndInvokeCreateGraphicsISyncronizeInvokeInvokeRequired,Invoke,BeginInvoke,EndInvoke Control implementa ISyncronizeInvoke

52 InvokeRequired public readonly property InvokeRequired() as boolean get get Dim procId as Integer Dim WinThId = GetWindowThreadProcessId(Me.Handle, procId) Dim procId as Integer Dim WinThId = GetWindowThreadProcessId(Me.Handle, procId) return AppDomain.GetCurrentThreadId()<>WinThId return AppDomain.GetCurrentThreadId()<>WinThId end get end get End property Ritorna True se il thread che invoca la proprietà è diverso da quello che ha creato il controllo Non siamo sicuri se è safe accedere al controllo Metodi always safe

53 Asyncronous Delegates I delegates possono essere utilizzati per eseguire operazioni asincrone BeginInvoke Ritorna IAsyncResult IAsyncResult.IsCompleted BeginInvoke (AsyncCallback,Object) Invoca AsyncCallback al termine delloperazione AsyncCallback(ByVal ar as IAsyncResult) Object viene passato alla procedura di callback In AsyncCallback deve essere invocato EndInvoke AsyncCallback viene invocata in un thread separato Il thread proviene dal ThreadPool BeginInvoke/EndInvoke posson generare delle eccezioni Non è possibile interropere BeginInvoke

54

55 GUI Asincrona: Approfondimenti Accesso Thread-Safe a controlli Windows Forms, Corrado Cavalli, articolo.Accesso Thread-Safe a controlli Windows Forms, Corrado Cavalli, articolo. articolo

56 Parte #3 Accesso ai dati

57 Prerequisiti v1.x: deploy manuale di MDAC 2.7+ v2.0: non necessitano di MDAC: Classi base, comuni (System.Data.Common) e disconnesse.NET managed provider per SQL Server e Oracle v2.0: usano MDAC, senza particolari requisiti: Managed provider OLEDB e ODBC Vanno bene le versioni 2.6, 2.7, 2.8 o… 9.0

58 Struttura di ADO.NET ADO.NET è composto da: Namespace System.Data: racchiude le primitive indipendenti dalla tipologia di base dati. Sono tutte disconnesse.NET Managed Provider: implementano le primitive necessarie allaccesso a specifiche basi dati System.Data.SqlClient per laccesso a SQL Server 7+ System.Data.OleDb per connettersi usando un OleDb Provider System.Data.OracleClientSystem.Data.Odbc

59 Common Provider Model ADO.NET v1.0/1.1 è basato su alcune interfacce E problematico scrivere codice indipendente dalla base dati E problematico scrivere codice indipendente dalla base dati ADO.NET 2.0 è basato su classi base condivise dai provider E una estensione, non introduce incompatibilità E una estensione, non introduce incompatibilità La sintassi SQL è comunque specifica per la base dati! La sintassi SQL è comunque specifica per la base dati! Architettura basata sul pattern Factory Architettura basata sul pattern Factory

60 .NET Data Providers.NET Data Providers Implementano un insieme comune di interfacce esponendo le classi: Connection: permette la connessione ad una base dati Command: permette lesecuzione di comandi SQL e Stored Procedure DataReader: implementa un cursore forward- only, read-only, client side DataAdapter: permette di riempire un contenitore disconnesso

61 Classe Connection Simile ad ADO old-style La Connection String: Può essere specificata mediante un costruttore parametrico Utilizza le stesse keyword di ADODB Richiede attenzione nella specifica del Provider: SqlClient non lo accetta System.Data.OleDb non supporta ODBC Dim conSQL As New SqlConnection( ) conSQL.ConnectionString = "Integrated Security=True;" & _ "Data Source=LocalHost;Initial Catalog=Pubs;" conSQL.Open( )

62 Classe Command Possiamo creare un comando: Mediante il costruttore Mediante metodo CreateCommand Possiamo eseguire un comando mediante i metodi: ExecuteReader: restituisce il DataReader in base ad una query ExecuteScalar: è il metodo preferibile se il risultato è un singleton ExecuteNonQuery: esegue un comando di azione Dim commSQL As New SqlCommand( ) commSQL.Connection = conSQL commSQL.CommandText = "Select Count(*) from Authors" MessageBox.Show(commSQL.ExecuteScalar( ).ToString)

63 Invocare Stored Procedure 1.Creare un oggetto Command 2.Impostare CommandType al valore StoredProcedure 3.Impostare la proprietà CommandText 4.Usare il metodo Add per creare e aggiungere parametri 5.Impostare la proprietà ParameterDirection 6.Invocare ExecuteReader 7.Consumare i record, e chiudere il DataReader 8.Leggere i parametri di output e il valore di ritorno

64 Usare un DataReader Per usare un DataReader, possiamo: Avanzare alla posizione successiva mediante il metodo Read(), che ritorna True finchè non si sono consumati tutti i dati Leggere i valori dei campi mediante la proprietà Item, oppure mediante i metodi GetXYZ() Un DataReader impegna la propria connessione, quindi: Non è possibile utilizzarla per eseguire comandi Dobbiamo ricordarci di chiuderlo mediante il metodo Close()

65

66 M.A.R.S. In v2.0, Multiple Active Results Sets: Mantiene disponibile una connessione quando apriamo un SqlDataReader al fine di poter: Eseguire unaltra query per ottenere un DataReader/XmlReader Eseguire comandi DML permette differenti result set contemporaeamente attivi: alternare fetch ad ogni reader Alternare query che non restituiscono reader Mediante MARS, è sufficiente una sola connessione se: I dati risiedono nello stesso database Usiamo SQL Server 2005/MDAC9

67 M.A.R.S. Dim parentReader As DataReader = Command1.ExecuteReader() While parentReader.Read() ' process parent row data here ' process parent row data here ' then get rowset from child table ' then get rowset from child table = parentReader("id") = parentReader("id") Dim childReader As DataReader = Command2.ExecuteReader() Dim childReader As DataReader = Command2.ExecuteReader() ' process child rows here ' process child rows here childReader.Close() childReader.Close() End While parentReader.Close()

68 La classe DataTable E il contenitore dati disconnessi di ADO.NET La classe DataTable espone le proprietà: Columns: è una collezione di istanze di DataColumn Rows: è una collezione di istanze della classe DataRow Una istanza della classe DataRow: Mette a disposizione il contenuto delle proprie colonne mediante la proprietà Item

69 Popolare una DataTable Popolare una DataTable accedendo ad un RDBMS Creare una DataTable in via programmatica Dim tblAuthors As DataTable = New DataTable("authors") tblAuthors.Columns.Add("AuthorID", GetType (Integer)) Dim adaptSQL As New SqlClient.SqlDataAdapter( _ "Select * from authors", conSQL) Dim datPubs As DataTable = New DataTable( ) adaptSQL.Fill(datPubs, "NewTable")

70 Scorrere una DataTable Dim numeroRighe As Int32 = unaTabella.Rows.Count Dim indiceRiga As Int32 Dim unaRiga As DataRow For indiceRiga = 0 To numeroRighe - 1 unaRiga = unaTabella.Rows(indiceRiga) Dim nomeAutore As String = _ unaRiga.Item("au_fname").ToString() Next

71 Modificare una DataTable Aggiungere righe Modificare righe Cancellare righe myDataTable.Rows.Remove(drDelRow) drChangeRow.BeginEdit( ) drChangeRow("Title") = drChangeRow("Title").ToString & " 1" drChangeRow.EndEdit( ) Dim drNewRow As DataRow = myDataTable.NewRow 'Populate columns datPubs.Tables("Titles").Rows.Add(drNewRow)

72

73 La classe DataAdapter E il collegamento tra il mondo connesso e quello disconnesso Può riempire DataSet/DataTable avvalendosi di una connessione chiusa Dim adaptSQL As New SqlClient.SqlDataAdapter( _ "Select * from authors", conSQL) Dim datPubs As New DataSet( ) adaptSQL.Fill(datPubs, "miaTabella") ' Accesso ai dati adaptSQL.Update (datPubs, "miaTabella")

74 Aggiornare il database Possiamo ripercuotere le modifiche effettuate al DataSet sul DB: Esplicitando il comando di aggiornamento Generando automaticamente il comando di update Dim sqlCommBuild As New SqlCommandBuilder(adaptSQL) MsgBox(sqlCommBuild.GetInsertCommand.CommandText) adaptSQL.Update(datPubs, "titles") Dim comm As New SqlClient.SqlCommand("Insert titles" & _ "(title_id, title, type) adaptSQL.InsertCommand = comm adaptSQL.Update(datPubs, "titles")

75 La classe DataSet E un contenitore disconnesso E assimilabile ad un vettore di matrici Permette di specificare vincoli e relazioni tra i dati contenuti E accessibile in scrittura Permette di propagare le modifiche su un DB Supporta nativamente la (de)serializzazione in formato XML

76 ADO.NET: Approfondimenti ADO.NET Primer, Andrea Saltarello, workshop Data Management. workshop Data Management. workshop Data Management. ADO.NET 2.0: whats new, Andrea Saltarello, workshop Data Management. workshop Data Management.workshop Data Management.

77 Big picture... In ADO.NET le sorgenti dati disconnesse non hanno più il concetto di record corrente. Il binding di dotnet è gestito da un intermediario tra controllo e sorgente dati Qualunque oggetto che implementi IList può essere utilizzato come sorgente dati La presenza di un unico intermediario per bindare più controlli garantisce il sync tra questi BindingManagerBase Data Sync

78 Sincronizzazione tra controlli E se non volessimo i due controlli sincronizzati? E se volessimo scorrere i dati indipendentemente sui due controlli? Risposta: bisogna avere due intermediari Data Sync BindingManagerBase

79 Simple binding Associa un qualsiasi tipo ad un controllo in modo da semplificare la presentazione di un valore e poterlo aggiornare Il binding con un singolo elemento implica l'uso di PropertyManager La proprietà Position sarà sempre 0 Il binding con una lista di elementi implica l'uso di CurrencyManager che ha il concetto di 'record corrente'. Si usa Position per navigare le righe mostrate Non si usa Position per leggere la posizione perchè la lista potrebbe contenere elementi che non vengono mostrati (es. filtro sulla dataview) Si usa Current per leggere l'elemento nella lista sottostante (datasource) int i=5; myLabel.DataBindings.Add("Text", i, null); int i=5; myLabel.DataBindings.Add("Text", i, null); Proprietà controllo DataSource DataMember se è null viene usato ToString()

80

81 Data binding: Approfondimenti Windows Forms Databinding, Raffaele Rialdi, workshop Data Management. workshop Data Management.workshop Data Management.

82 Denkiù!

83 Links


Scaricare ppt "Da VB a VB.NET Andrea Saltarello [IDevGuro] Corrado Cavalli [DevGuro(Of T)]"

Presentazioni simili


Annunci Google