Torna indietro   Hardware Upgrade Forum > Software > Programmazione

iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile
iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile
C'è tanta sostanza nel nuovo smartphone della Mela dedicato ai creator digitali. Nuovo telaio in alluminio, sistema di raffreddamento vapor chamber e tre fotocamere da 48 megapixel: non è un semplice smartphone, ma uno studio di produzione digitale on-the-go
Intel Panther Lake: i processori per i notebook del 2026
Intel Panther Lake: i processori per i notebook del 2026
Panther Lake è il nome in codice della prossima generazione di processori Intel Core Ultra, che vedremo al debutto da inizio 2026 nei notebook e nei sistemi desktop più compatti. Nuovi core, nuove GPU e soprattutto una struttura a tile che vede per la prima volta l'utilizzo della tecnologia produttiva Intel 18A: tanta potenza in più, ma senza perdere in efficienza
Intel Xeon 6+: è tempo di Clearwater Forest
Intel Xeon 6+: è tempo di Clearwater Forest
Intel ha annunciato la prossima generazione di processori Xeon dotati di E-Core, quelli per la massima efficienza energetica e densità di elaborazione. Grazie al processo produttivo Intel 18A, i core passano a un massimo di 288 per ogni socket, con aumento della potenza di calcolo e dell'efficienza complessiva.
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 10-10-2010, 21:15   #1
RaouL_BennetH
Senior Member
 
L'Avatar di RaouL_BennetH
 
Iscritto dal: Sep 2004
Messaggi: 3967
[.Net 4.0]Operazioni Thread Safe (winforms)

Ciao a tutti

Sto lavorando con l'entity framework ed ho dei problemi (credo i classici...) a gestire determinate operazioni.

Attualmente il progetto è di tipo WinForms.

Il problema che ho è il seguente:

Ho il classico form contenente alcune caselle di input ed una griglia.

Durante l'inserimento/modifica dei dati , ho un semplice form di attesa, e dopo l'inserimento/modifica la mia griglia deve aggiornarsi.

Attualmente il codice è questo:

Codice:
//context è il mio EntityModel

public MyForm()
{
    InitializeComponent();
    base.SaveClicked += new ButtonClickHandler(SaveEntity);
    base.UpdateClicked += new ButtonClickHandler(UpdateEntity);
    context = new EFHelper();
    validator = new InputValidator();
}

private void MyForm_Load(object sender, EventArgs e)
{
     //popolo la griglia 
     GridRefresh();
}

void GridRefresh()
{
    var result = (from contratti in context.Categoria_Contratti
          select contratti).ToList();
     contrattiBindingSource.DataSource = result;
}

private void SaveEntity()
{
      context.AddToCategoria_Contratti(contratto);
      context.SaveChanges();
      GridRefresh();   
}
In questo caso, tutto va bene e non ci sono problemi.

Io invece volevo strafare e mi sono incasinato...

Codice:
public class Worker()
{
     private BackGroundWorker bw;
     private FormWait wf;

     public void WorkAsync()
     {
         wf = new FormWait();
         wf.Show();
         bw = new BackGroundWorker();
         bw.DoWork += new DoWorkEventHandler(bw_DoWork);
         bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
         bw.RunWorkerAsync();
}

private void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
        wf.Close();
}

private void bw_DoWork(object sender, DoWorkEventArgs e)
{
        devo_far_partire_un_metodo_che_qui_non_conosco
}
In parte ci sono anche riuscito con qualche bruttura:

Codice:
public delegate void ProvaChiamata();
public ProvaChiamata metodoInvocato;

//nel bw_DoWork...
metodoInvocato();
Per quanto riguarda l'inserimento dei dati, la mia SaveEntity diventa quindi:

Codice:
private void SaveEntity()
{
    Worker worker = new Worker();
    worker.metodoInvocato = delegate()
    {
       context.AddToCategoria_Contratti(contratto);
       context.SaveChanges();
       GridRefresh();
     };
     worker.WorkAsync();
}
Bene.. cioè, male, l'inserimento me lo effettua, ma quando deve aggiornare la griglia mi solleva l'eccezione di: "Operazione cross-thread non valida: è stato eseguito l'accesso al controllo 'gridView' da un thread diverso da quello da cui è stata eseguita la creazione"
__________________
Dai wafer di silicio nasce: LoHacker... il primo biscotto Geek
RaouL_BennetH è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 08:37   #2
banryu79
Senior Member
 
L'Avatar di banryu79
 
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
Quote:
Originariamente inviato da RaouL_BennetH Guarda i messaggi
...
Bene.. cioè, male, l'inserimento me lo effettua, ma quando deve aggiornare la griglia mi solleva l'eccezione di: "Operazione cross-thread non valida: è stato eseguito l'accesso al controllo 'gridView' da un thread diverso da quello da cui è stata eseguita la creazione"
Suppongo (perchè la piattaforma .NET non la conosco ma so che è single-threaded) che il problema è dato dal fatto che stai accedendo ad un oggetto della GUI (gridView) da un thread diverso dallo specifico thread dedicato.

Prova a cercare informazioni specifiche in merito, oppure incrocia le dita, magari un utente .Net informato passa di qua, ti legge e decide di risponderti.
__________________

As long as you are basically literate in programming, you should be able to express any logical relationship you understand.
If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it.
(Chris Crawford)
banryu79 è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 08:45   #3
gugoXX
Senior Member
 
L'Avatar di gugoXX
 
Iscritto dal: May 2004
Città: Londra (Torino)
Messaggi: 3692
E' davvero tanto che non uso Winforms, e soprattutto ancora di piu' che non lo uso senza pattern come MVC.
Ad occhio pero' direi che dovresti chiamare la UpdateGrid dal thread che ha creato la finestra.

BeginInvoke( new Action( ()=>GridRefresh() ) ,null, null);

O qualcosa di simile, sto scrivendo al volo.
Magari
BeginInvoke( GridRefresh ,null, null);
viene compilata ed e' piu' leggibile.
__________________
Se pensi che il tuo codice sia troppo complesso da capire senza commenti, e' segno che molto probabilmente il tuo codice e' semplicemente mal scritto.
E se pensi di avere bisogno di un nuovo commento, significa che ti manca almeno un test.
gugoXX è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 08:49   #4
Kralizek
Senior Member
 
L'Avatar di Kralizek
 
Iscritto dal: Feb 2003
Città: Stockholm (SE)
Messaggi: 1343
se sei in .net-4 e vuoi parallelizzare il tuo lavoro, perchè non usare la TPL?

ecco un bel tutorial da cui prendere spunto: http://blogs.msdn.com/b/csharpfaq/ar...g-started.aspx
Kralizek è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 09:10   #5
astorcas
Senior Member
 
L'Avatar di astorcas
 
Iscritto dal: Jan 2005
Città: Siena
Messaggi: 1313
Quote:
Originariamente inviato da gugoXX Guarda i messaggi
E' davvero tanto che non uso Winforms, e soprattutto ancora di piu' che non lo uso senza pattern come MVC.
Ad occhio pero' direi che dovresti chiamare la UpdateGrid dal thread che ha creato la finestra.

BeginInvoke( new Action( ()=>GridRefresh() ) ,null, null);

O qualcosa di simile, sto scrivendo al volo.
Magari
BeginInvoke( GridRefresh ,null, null);
viene compilata ed e' piu' leggibile.
Si io direi una cosa del genere:

Codice:
public delegate void InvokeDelegate();

void GridRefresh()
{
       if(contrattiBindingSource.InvokeRequired){
                BeginInvoke(new InvokeDelegate(GridRefresh));
       }
       else{
           var result = (from contratti in context.Categoria_Contratti
             select contratti).ToList();
           contrattiBindingSource.DataSource = result;
      }
}
In questo modo è il metodo stesso a gestire le chiamate da thread esterni o esegue direttamente il codice se la chiamata arriva dal thread principale.

Ho messo il delegato InvokeDelegate ma sono quasi certo che ce ne sia uno con la stessa firma, se lo trovi usa quello

EDIT: Non serve creare un delegate apposta (InvokeDelegate nel mio caso), ne esiste uno già bello e pronto: MethodInvoker

Ultima modifica di astorcas : 11-10-2010 alle 09:41.
astorcas è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 10:56   #6
RaouL_BennetH
Senior Member
 
L'Avatar di RaouL_BennetH
 
Iscritto dal: Sep 2004
Messaggi: 3967
Innanzitutto, grazie a tutti per le risposte

@gugoXX e @astorcas

Sebbene leggermente diversi, i vostri suggerimenti funzionano entrambi

e quindi vi chiedo:

ma è per qualsiasi tipo di controllo che ci sono queste limitazioni o soltanto per quelli che hanno un legame con una base dati ?


@Kralizek
Se l'eseguibile finale dovrà girare anche su pc con processori vecchio "stile", single core, posso comunque prendere in considerazione l'approfondire TPL ?

Grazie a tutti

RaouL.
__________________
Dai wafer di silicio nasce: LoHacker... il primo biscotto Geek
RaouL_BennetH è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 11:13   #7
astorcas
Senior Member
 
L'Avatar di astorcas
 
Iscritto dal: Jan 2005
Città: Siena
Messaggi: 1313
qualsiasi controllo ha bisogno di essere aggiornato dallo stesso thread da cui è stato creato. Per verificare che ci si si trovi in un caso o nell'altro ogni controllo può far affidamento alla proprietà InvokeRequired (infatti la trovi proprio nella classe Control).
Il "pattern" è sempre il solito, almeno io uso sempre il solito.

Codice:
public voiid NomeFunzione()
{
      if(controlloDaModificare.InvokeRequired)
      {
          BeginInvoke(new MethodInvoker(NomeFunzione));
      }
      else
      {
          //Corpo del metodo che modifica il controllo
      }

}
In pratica questo si traduce in:
"Sono il thread principale?"

Si? Allora modifico direttamente (vado nell'else)

No? Allora chiedo al thread principale di farlo per me (quindi quando il codice girerà nel thread principale andrà di nuovo nell'else)

Ecco ti ho esposto, più o meno, il mio punto di vista
astorcas è offline   Rispondi citando il messaggio o parte di esso
Old 11-10-2010, 11:28   #8
Kralizek
Senior Member
 
L'Avatar di Kralizek
 
Iscritto dal: Feb 2003
Città: Stockholm (SE)
Messaggi: 1343
Quote:
Originariamente inviato da RaouL_BennetH Guarda i messaggi
@Kralizek
Se l'eseguibile finale dovrà girare anche su pc con processori vecchio "stile", single core, posso comunque prendere in considerazione l'approfondire TPL ?
diciamo che la parallelizzazione serve solo in parte a gestire multi core. nella maggior parte dei casi serve per comunicare al SO i tempi morti e come gestirli aumentando l'efficienza dell'uso del processore quando ti viene assegnato.

Per fare un esempio:

mettiamo che fare una query al database significa compiere queste operazioni:
- inoltro del comando al server
- esecuzione del comando (eseguito sul server)
- reperimento dei risultati dal server

e che devi eseguire 4 query bene o male indipendenti tra di loro. nella programmazione classica ti troveresti ad aspettare la query n-1 prima di poter eseguire la query n.

usando un minimo di parallelismo, anche su un processore single-core, tu comunichi al sistema operativo che nei tempi morti (ad es l'esecuzione del comando) può iniziare a preparare il comando della seconda query ed inoltrarlo al database e mettersi in attesa del risultato.

spero di essere stato chiaro
Kralizek è offline   Rispondi citando il messaggio o parte di esso
Old 12-10-2010, 09:47   #9
RaouL_BennetH
Senior Member
 
L'Avatar di RaouL_BennetH
 
Iscritto dal: Sep 2004
Messaggi: 3967
@Kralizek: il concetto è chiarissimo

@astorcas: non per ripetermi ma.... il concetto è chiarissimo


Grazie mille ragazzi !

RaouL.
__________________
Dai wafer di silicio nasce: LoHacker... il primo biscotto Geek
RaouL_BennetH è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile iPhone 17 Pro: più di uno smartphone. &Eg...
Intel Panther Lake: i processori per i notebook del 2026 Intel Panther Lake: i processori per i notebook ...
Intel Xeon 6+: è tempo di Clearwater Forest Intel Xeon 6+: è tempo di Clearwater Fore...
4K a 160Hz o Full HD a 320Hz? Titan Army P2712V, a un prezzo molto basso 4K a 160Hz o Full HD a 320Hz? Titan Army P2712V,...
Recensione Google Pixel Watch 4: basta sollevarlo e si ha Gemini sempre al polso Recensione Google Pixel Watch 4: basta sollevarl...
Nano Banana si espande: l’AI di Google p...
Che fare con i Tesla Cybertruck invendut...
Simucube 3 Sport, Pro e Ultimate ufficia...
Facebook rilancia le offerte di lavoro: ...
Hisense PT1: il cinema in casa con la po...
Pixel 10: come risolvere (forse) i crash...
Plenitude lancia la sua Fibra ottica: fi...
Apple TV+ elimina il 'plus' dal nome: or...
Prezzi da outlet in saldo su 23 articoli...
Death 2 Spotify: a Oakland nasce il movi...
Vivo presenta X300 e X300 Pro: due flags...
iPad mini con chip A17 Pro: potenza da M...
Samsung cresce oltre le attese grazie al...
Microsoft presenta MAI-Image-1, il suo p...
AirPods Pro 3 contro AirPods Pro 2: Appl...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 10:42.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
Served by www3v