C# 4.0
In una query LINQ si presenta talvolta la necessità di aggiungere in modo condizionato uno o più parametri di ricerca in modo tale da creare una clausola Where pertinente.
Si può quindi procedere nel seguente modo utilizzando delle lambda expressions, dove in questo caso i parametri di ricerca sono rappresentati da code e salesDriveCode :
using (OrangeRepositoryEntities dcOrangeRepository = new OrangeRepositoryEntities())
{
var query =
from p in dcOrangeRepository.Promotion
select p;
if (!string.IsNullOrEmpty(code))
query = query.Where(p => p.Code.Contains(code));
if (!string.IsNullOrEmpty(salesDriveCode))
query = query.Where(p => p.SalesDriveCode.Contains(salesDriveCode));
foreach (var promotion in query)
{
…
}
}
During my day to day work, I frequently find useful information, from various blogs spread all over the Internet. In order to avoid others repeating my same researches, I decided to share my findings in this blog, with no claiming to be exhaustive nor teaching anything to anyone. These postings are provided 'AS IS' with no warranties, and confer no rights. The views expressed on this weblog are mine alone.
venerdì 27 agosto 2010
WPF - Larghezza automatica di un TabControl
C# 4.0
In una maschera WPF si vuole inserire un TabControl che abbia la larghezza impostata sulla stessa larghezza della maschera stessa.
Impostare quindi gli attributi Margin e Width dell’elemento XAML TabControl, nel modo seguente:
<TabControl Grid.Row="1" Name="tabMain" Margin="0,0,0,0" Width="Auto">
…
</TabControl>
Importante poi, NON impostare l’attributo HorizontalAlignment per non compromettere il corretto allineamento del controllo.
In una maschera WPF si vuole inserire un TabControl che abbia la larghezza impostata sulla stessa larghezza della maschera stessa.
Impostare quindi gli attributi Margin e Width dell’elemento XAML TabControl, nel modo seguente:
<TabControl Grid.Row="1" Name="tabMain" Margin="0,0,0,0" Width="Auto">
…
</TabControl>
Importante poi, NON impostare l’attributo HorizontalAlignment per non compromettere il corretto allineamento del controllo.
giovedì 26 agosto 2010
SQL Server 2005 - TSQL - Aggiornamento di una tabella tramite sub-query
Si deve aggiornare un campo di una tabella prendendo il valore da un’altra, dove il punto di contatto tra le due è rappresentato da un altro campo in comune tra la seconda ed una terza tabella.
In questo esempio il campo AuxiliaryCode della tabella FinalPromotion deve essere aggiornato con il valore del campo Code della DocumentCoupon che non ha FK verso FinalPromotion ma verso Document che a sua volta è legata a FinalPromotion.
Come sempre lo statement T-SQL che segue, potrebbe valere più di mille parole:
UPDATE FP
SET FP.AuxiliaryCode = A.Code
FROM FinalPromotion FP
INNER JOIN Document D
ON FP.DocumentGuid = D.Guid
INNER JOIN
(SELECT D.UpperDocumentGuid AS UDG, DC.Code FROM DocumentCoupon DC
INNER JOIN Document D
ON DC.Id = D.CouponId) A
ON D.UpperDocumentGuid = A.UDG
WHERE D.UpperDocumentGuid = 'b9464a81-b83c-4ce4-8758-b0e632240c2b'
In questo esempio il campo AuxiliaryCode della tabella FinalPromotion deve essere aggiornato con il valore del campo Code della DocumentCoupon che non ha FK verso FinalPromotion ma verso Document che a sua volta è legata a FinalPromotion.
Come sempre lo statement T-SQL che segue, potrebbe valere più di mille parole:
UPDATE FP
SET FP.AuxiliaryCode = A.Code
FROM FinalPromotion FP
INNER JOIN Document D
ON FP.DocumentGuid = D.Guid
INNER JOIN
(SELECT D.UpperDocumentGuid AS UDG, DC.Code FROM DocumentCoupon DC
INNER JOIN Document D
ON DC.Id = D.CouponId) A
ON D.UpperDocumentGuid = A.UDG
WHERE D.UpperDocumentGuid = 'b9464a81-b83c-4ce4-8758-b0e632240c2b'
mercoledì 25 agosto 2010
NHibernate - Gestione della concorrenza
C# 2.0
In alcune applicazioni c’è la necessità di gestire un corretto accesso ai dati, evitando eventuali problemi di concorrenza. Tipico esempio potrebbe essere quello delle gestione degli ordini, dove due utenti stanno lavorando contemporaneamente sullo stesso ordine e non si vuole rischiare di sovrascrivere informazioni dell’uno o dell’altro operatore.
Con NHibernate quindi è abbastanza semplice gestire tale esigenza, agendo nel seguente modo:
1. Aggiungere nella tabella degli ordini il campo DataUpdate di tipo datetime.
2. Modificare il file di mapping della tabella interessata, aggiungendo l’elemento timestamp subito dopo quello dell’identificativo della tabella, ed è importante che sia in questa posizione, prima della definizione delle altre proprietà affinché tutto funzioni correttamente.
…
<id name="Id" type="Guid">
<column name="Id" length="16" sql-type="uniqueidentifier" not-null="true" unique="true" index="PK_Ordine"/>
<generator class="guid" />
</id>
<timestamp column ="DataUpdate" name ="DataUpdate"/>
<property name="IdAssegnazione" type="Guid">
<column name="IdAssegnazione" length="16" sql-type="uniqueidentifier" not-null="false"/>
</property>
…
3. Gestire l’eccezione di tipo StaleObjectStateException nel metodo di aggiornamento dei dati.
try
{
using (ISession s = SessionController.Instance.NHSession)
{
s.SaveOrUpdate(ordine);
s.Flush();
}
}
}
}
catch (NHibernate.StaleObjectStateException ex)
{
MessageBox.Show("Attenzione! L’ordine è già stato modificato da un altro utente.\rControllare il file di Log. ", "Gestione Ordini", MessageBoxButtons.OK, MessageBoxIcon.Warning);
// Eventuale scrittura su file di log
…
}
catch (Exception ex)
{
…
}
finally
{
…
}
In alcune applicazioni c’è la necessità di gestire un corretto accesso ai dati, evitando eventuali problemi di concorrenza. Tipico esempio potrebbe essere quello delle gestione degli ordini, dove due utenti stanno lavorando contemporaneamente sullo stesso ordine e non si vuole rischiare di sovrascrivere informazioni dell’uno o dell’altro operatore.
Con NHibernate quindi è abbastanza semplice gestire tale esigenza, agendo nel seguente modo:
1. Aggiungere nella tabella degli ordini il campo DataUpdate di tipo datetime.
2. Modificare il file di mapping della tabella interessata, aggiungendo l’elemento timestamp subito dopo quello dell’identificativo della tabella, ed è importante che sia in questa posizione, prima della definizione delle altre proprietà affinché tutto funzioni correttamente.
…
<id name="Id" type="Guid">
<column name="Id" length="16" sql-type="uniqueidentifier" not-null="true" unique="true" index="PK_Ordine"/>
<generator class="guid" />
</id>
<timestamp column ="DataUpdate" name ="DataUpdate"/>
<property name="IdAssegnazione" type="Guid">
<column name="IdAssegnazione" length="16" sql-type="uniqueidentifier" not-null="false"/>
</property>
…
3. Gestire l’eccezione di tipo StaleObjectStateException nel metodo di aggiornamento dei dati.
try
{
using (ISession s = SessionController.Instance.NHSession)
{
s.SaveOrUpdate(ordine);
s.Flush();
}
}
}
}
catch (NHibernate.StaleObjectStateException ex)
{
MessageBox.Show("Attenzione! L’ordine è già stato modificato da un altro utente.\rControllare il file di Log. ", "Gestione Ordini", MessageBoxButtons.OK, MessageBoxIcon.Warning);
// Eventuale scrittura su file di log
…
}
catch (Exception ex)
{
…
}
finally
{
…
}
martedì 24 agosto 2010
WPF - Anteprima di stampa con XPSDocument
C# 4.0
Questo metodo esegue l’anteprima di stampa di una stringa o di un oggetto visual, utilizzando l’oggetto XPSDocument.
///
/// Prints the preview.
///
///
The content.
public static void PrintPreview(string content)
{
byte[] bytes = Encoding.UTF8.GetBytes(content);
MemoryStream memoryStream = new MemoryStream(bytes.Length);
Package package = Package.Open(memoryStream, FileMode.Create, FileAccess.ReadWrite);
Uri uri = new Uri("pack://PackTemporaryUri.xps");
if (PackageStore.GetPackage(uri) == null)
PackageStore.AddPackage(uri, package);
XpsDocument xpsDocument = new XpsDocument(package, CompressionOption.NotCompressed, uri.AbsoluteUri);
FixedDocument fixedDocument = new FixedDocument();
PageContent pageContent = new PageContent();
FixedPage fixedPage = new FixedPage { Background = Brushes.White, Width = 96 * 8.5, Height = 96 * 11 };
TextBlock tbTitle = new TextBlock { Text = content };
FixedPage.SetLeft(tbTitle, 96 * 0.75); // left margin
FixedPage.SetTop(tbTitle, 96 * 0.75); // top margin
fixedPage.Children.Add(tbTitle);
((IAddChild)pageContent).AddChild(fixedPage);
fixedDocument.Pages.Add(pageContent);
XpsDocumentWriter writer = XpsDocument.CreateXpsDocumentWriter(xpsDocument);
writer.Write(fixedDocument);
DocumentViewer documentViewer = new DocumentViewer
{
Document = xpsDocument.GetFixedDocumentSequence()
};
xpsDocument.Close();
// Viene create dinamicamente una window per la visualizzazione dell'anteprima di stampa.
Window window = new Window
{
Title = "Print Preview",
WindowStartupLocation = WindowStartupLocation.CenterScreen
};
Uri uriIcon = new Uri("pack://application:,,,/Admin.Console;component/Content/Print.png");
window.Icon = BitmapFrame.Create(uriIcon);
Grid grid = new Grid();
grid.Children.Add(documentViewer);
window.Content = grid;
window.ShowDialog();
}
///
/// Prints the visual preview.
///
///
The content.
public static void PrintPreview(Visual content)
{
MemoryStream memoryStream = new MemoryStream();
Package package = Package.Open(memoryStream, FileMode.Create, FileAccess.ReadWrite);
Uri uri = new Uri("pack://PackTemporaryUri.xps");
if (PackageStore.GetPackage(uri) == null)
PackageStore.AddPackage(uri, package);
XpsDocument xpsDocument = new XpsDocument(package, CompressionOption.NotCompressed, uri.AbsoluteUri);
XpsDocumentWriter writer = XpsDocument.CreateXpsDocumentWriter(xpsDocument);
writer.Write(content);
DocumentViewer documentViewer = new DocumentViewer
{
Document = xpsDocument.GetFixedDocumentSequence()
};
xpsDocument.Close();
// Viene create dinamicamente una window per la visualizzazione dell'anteprima di stampa.
Window window = new Window
{
Title = "Print Preview",
WindowStartupLocation = WindowStartupLocation.CenterScreen
};
Uri uriIcon = new Uri("pack://application:,,,/Admin.Console;component/Content/Print.png");
window.Icon = BitmapFrame.Create(uriIcon);
Grid grid = new Grid();
grid.Children.Add(documentViewer);
window.Content = grid;
window.ShowDialog();
}
Questo metodo esegue l’anteprima di stampa di una stringa o di un oggetto visual, utilizzando l’oggetto XPSDocument.
///
/// Prints the preview.
///
///
The content.
public static void PrintPreview(string content)
{
byte[] bytes = Encoding.UTF8.GetBytes(content);
MemoryStream memoryStream = new MemoryStream(bytes.Length);
Package package = Package.Open(memoryStream, FileMode.Create, FileAccess.ReadWrite);
Uri uri = new Uri("pack://PackTemporaryUri.xps");
if (PackageStore.GetPackage(uri) == null)
PackageStore.AddPackage(uri, package);
XpsDocument xpsDocument = new XpsDocument(package, CompressionOption.NotCompressed, uri.AbsoluteUri);
FixedDocument fixedDocument = new FixedDocument();
PageContent pageContent = new PageContent();
FixedPage fixedPage = new FixedPage { Background = Brushes.White, Width = 96 * 8.5, Height = 96 * 11 };
TextBlock tbTitle = new TextBlock { Text = content };
FixedPage.SetLeft(tbTitle, 96 * 0.75); // left margin
FixedPage.SetTop(tbTitle, 96 * 0.75); // top margin
fixedPage.Children.Add(tbTitle);
((IAddChild)pageContent).AddChild(fixedPage);
fixedDocument.Pages.Add(pageContent);
XpsDocumentWriter writer = XpsDocument.CreateXpsDocumentWriter(xpsDocument);
writer.Write(fixedDocument);
DocumentViewer documentViewer = new DocumentViewer
{
Document = xpsDocument.GetFixedDocumentSequence()
};
xpsDocument.Close();
// Viene create dinamicamente una window per la visualizzazione dell'anteprima di stampa.
Window window = new Window
{
Title = "Print Preview",
WindowStartupLocation = WindowStartupLocation.CenterScreen
};
Uri uriIcon = new Uri("pack://application:,,,/Admin.Console;component/Content/Print.png");
window.Icon = BitmapFrame.Create(uriIcon);
Grid grid = new Grid();
grid.Children.Add(documentViewer);
window.Content = grid;
window.ShowDialog();
}
///
/// Prints the visual preview.
///
///
The content.
public static void PrintPreview(Visual content)
{
MemoryStream memoryStream = new MemoryStream();
Package package = Package.Open(memoryStream, FileMode.Create, FileAccess.ReadWrite);
Uri uri = new Uri("pack://PackTemporaryUri.xps");
if (PackageStore.GetPackage(uri) == null)
PackageStore.AddPackage(uri, package);
XpsDocument xpsDocument = new XpsDocument(package, CompressionOption.NotCompressed, uri.AbsoluteUri);
XpsDocumentWriter writer = XpsDocument.CreateXpsDocumentWriter(xpsDocument);
writer.Write(content);
DocumentViewer documentViewer = new DocumentViewer
{
Document = xpsDocument.GetFixedDocumentSequence()
};
xpsDocument.Close();
// Viene create dinamicamente una window per la visualizzazione dell'anteprima di stampa.
Window window = new Window
{
Title = "Print Preview",
WindowStartupLocation = WindowStartupLocation.CenterScreen
};
Uri uriIcon = new Uri("pack://application:,,,/Admin.Console;component/Content/Print.png");
window.Icon = BitmapFrame.Create(uriIcon);
Grid grid = new Grid();
grid.Children.Add(documentViewer);
window.Content = grid;
window.ShowDialog();
}
NHibernate - a different object with the same identifier value was already associated with ...
C# 2.0
Questo messaggio d’errore è molto probabilmente dovuto ad un errore di mappatura, specialmente nella cancellazione a cascata: cascade="all-delete-orphan", nel senso che questo attributo non dovrebbe essere presente.
Questo messaggio d’errore è molto probabilmente dovuto ad un errore di mappatura, specialmente nella cancellazione a cascata: cascade="all-delete-orphan", nel senso che questo attributo non dovrebbe essere presente.
NHibernate - Combo SelectedItem binding
C# 2.0
Non sempre assegnando direttamente un oggetto alla proprietà binding SelectedItem di un combo, questo viene sincronizzato correttamente: per ovviare a questo inconveniente bisogna esporre l’Id dell’oggetto e valorizzare la proprietà SelectedValue del combo a questo Id.
Inoltre se in un combo, il binding impostato su SelectedItem sembri non sentire il change della base dati agganciata, verificare che la tabella di riferimento nel file di mapping abbia impostato il lazy a false:
<many-to-one name="TipoNota" lazy ="false" class="Pratiche.DataLayer.Database.Entities.TipoNota, Pratiche.DataLayer">
<column name="TipoNotaId" length="4" sql-type="int" not-null="false"/>
</many-to-one>
Non sempre assegnando direttamente un oggetto alla proprietà binding SelectedItem di un combo, questo viene sincronizzato correttamente: per ovviare a questo inconveniente bisogna esporre l’Id dell’oggetto e valorizzare la proprietà SelectedValue del combo a questo Id.
Inoltre se in un combo, il binding impostato su SelectedItem sembri non sentire il change della base dati agganciata, verificare che la tabella di riferimento nel file di mapping abbia impostato il lazy a false:
<many-to-one name="TipoNota" lazy ="false" class="Pratiche.DataLayer.Database.Entities.TipoNota, Pratiche.DataLayer">
<column name="TipoNotaId" length="4" sql-type="int" not-null="false"/>
</many-to-one>
NHibernate - Oggetti su istanze diverse
C# 2.0
A volte gli oggetti appartengono ad istanze diverse, e pur essendo dello stesso tipo possono risultare diversi, per questo quindi occorre fare l’override dell’oggetto in questione:
public override bool Equals(object obj)
{
if (obj != null && obj.ToString() != string.Empty)
{
return _id == ((TipoNota)obj).Id;
}
else
{
return base.Equals(obj);
}
}
A volte gli oggetti appartengono ad istanze diverse, e pur essendo dello stesso tipo possono risultare diversi, per questo quindi occorre fare l’override dell’oggetto in questione:
public override bool Equals(object obj)
{
if (obj != null && obj.ToString() != string.Empty)
{
return _id == ((TipoNota)obj).Id;
}
else
{
return base.Equals(obj);
}
}
NHibernate - Collezioni di tipo Bag e Set
C# 2.0
Le collezioni di tipo Bag permettono l’inserimento di valori duplicati, questo quindi può comportare dei problemi in fase di Refresh, in pratica vengono duplicati i valori all’interno della collection. A questo punto è consigliabile una collezione di tipo Set, la quale può anche essere ordinata:
<set name="PraNoteScadenzes" order-by="DataScadenza desc" lazy ="false" inverse ="true" cascade ="all-delete-orphan">
<key column="PraId"/>
<one-to-many class="Pratiche.DataLayer.Database.Entities.NoteScadenze, Pratiche.DataLayer"/>
</set>
Nella classe poi, la proprietà deve essere codificata così:
public virtual Iesi.Collections.ISet PraNoteScadenzes
{
get
{
if (_praNoteScadenzes == null)
{
_praNoteScadenzes = new SortedSet();
}
return _praNoteScadenzes;
}
set { _praNoteScadenzes = value; }
}
Le collezioni di tipo Bag permettono l’inserimento di valori duplicati, questo quindi può comportare dei problemi in fase di Refresh, in pratica vengono duplicati i valori all’interno della collection. A questo punto è consigliabile una collezione di tipo Set, la quale può anche essere ordinata:
<set name="PraNoteScadenzes" order-by="DataScadenza desc" lazy ="false" inverse ="true" cascade ="all-delete-orphan">
<key column="PraId"/>
<one-to-many class="Pratiche.DataLayer.Database.Entities.NoteScadenze, Pratiche.DataLayer"/>
</set>
Nella classe poi, la proprietà deve essere codificata così:
public virtual Iesi.Collections.ISet PraNoteScadenzes
{
get
{
if (_praNoteScadenzes == null)
{
_praNoteScadenzes = new SortedSet();
}
return _praNoteScadenzes;
}
set { _praNoteScadenzes = value; }
}
NHibernate - "Unexpected row count: 0; expected: 1"
C# 2.0
Con questo errore verificare nel file di mapping che:
1. L’attributo “class” dell’elemento “generator”, nel caso di Guid deve essere <generator class="guid" />
2. Togliere anche l’attributo unsaved-value="null".
Inoltre togliere l’eventuale default nel campo chiave del DB SQL.
Con questo errore verificare nel file di mapping che:
1. L’attributo “class” dell’elemento “generator”, nel caso di Guid deve essere <generator class="guid" />
2. Togliere anche l’attributo unsaved-value="null".
Inoltre togliere l’eventuale default nel campo chiave del DB SQL.
SQL Server 2005 - Remote connections
Può succedere che da un WebService non si riesca ad accedere a SQL Server.
Controllare quindi, tramite SQL Server 2005 Surface Area Configuration, l'impostazione di "Remote Connections" del gruppo "Database Engine". Le opzioni devono essere impostati su: "Local and remote connections" e "Using TCP/IP only".
Controllare quindi, tramite SQL Server 2005 Surface Area Configuration, l'impostazione di "Remote Connections" del gruppo "Database Engine". Le opzioni devono essere impostati su: "Local and remote connections" e "Using TCP/IP only".
Presentazione
Molto spesso nel quotidiano svolgimento della mia
professione ho trovato informazioni utilissime in diversi blog sparsi per il
mondo, talvolta scovati nei siti più remoti della rete grazie a “potenti motori
di ricerca”.
Così mi son posto il quesito, che qualcun altro a sua volta,
potesse effettuare lo stesso tipo di ricerca, così ecco questo blog: che non ha
nessuna pretesa di insegnare niente a nessuno o distribuire il “verbo”. Ha solamente
lo scopo di mettere su “carta” alcune soluzioni pratiche ai problemi che ci
affliggono giorno per giorno.
Diego Parolin aka Didacus
Iscriviti a:
Post (Atom)