Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Sony INZONE H6 Air: il primo headset open-back di Sony per giocatori
Sony INZONE H6 Air: il primo headset open-back di Sony per giocatori
Il primo headset open-back della linea INZONE arriva a 200 euro con driver derivati dalle cuffie da studio MDR-MV1 e un peso record di soli 199 grammi
Nutanix cambia pelle: dall’iperconvergenza alla piattaforma full stack per cloud ibrido e IA
Nutanix cambia pelle: dall’iperconvergenza alla piattaforma full stack per cloud ibrido e IA
Al .NEXT 2026 di Chicago, Nutanix ha mostrato quanto sia cambiata: una piattaforma software che gestisce VM, container e carichi di lavoro IA ovunque, dall’on-premise al cloud pubblico. Con un’esecuzione rapidissima sulle partnership e sulla migrazione da VMware
Recensione Xiaomi Pad 8 Pro: potenza bruta e HyperOS 3 per sfidare la fascia alta
Recensione Xiaomi Pad 8 Pro: potenza bruta e HyperOS 3 per sfidare la fascia alta
Xiaomi Pad 8 Pro adotta il potente Snapdragon 8 Elite all'interno di un corpo con spessore di soli 5,75 mm e pannello LCD a 144Hz flicker-free, per un tablet che può essere utilizzato con accessori dedicati di altissima qualità. Fra le caratteristiche esclusive, soprattutto per chi intende usarlo con la tastiera ufficiale, c'è la modalità Workstation di HyperOS 3, che trasforma Android in un sistema operativo con interfaccia a finestre
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


Sony INZONE H6 Air: il primo headset open-back di Sony per giocatori Sony INZONE H6 Air: il primo headset open-back d...
Nutanix cambia pelle: dall’iperconvergenza alla piattaforma full stack per cloud ibrido e IA Nutanix cambia pelle: dall’iperconvergenza alla ...
Recensione Xiaomi Pad 8 Pro: potenza bruta e HyperOS 3 per sfidare la fascia alta Recensione Xiaomi Pad 8 Pro: potenza bruta e Hyp...
NZXT H9 Flow RGB+, Kraken Elite 420 e F140X: abbiamo provato il tris d'assi di NZXT NZXT H9 Flow RGB+, Kraken Elite 420 e F140X: abb...
ASUS ROG Swift OLED PG34WCDN recensione: il primo QD-OLED RGB da 360 Hz ASUS ROG Swift OLED PG34WCDN recensione: il prim...
Ecovacs presenta la gamma 2026: paviment...
Efficienza energetica fino a 2.000 volte...
Lenovo 360: il programma di canale dell'...
Appena 10.000 qubit per rompere la critt...
Analisi dei transistor durante il funzio...
Attacco informatico a Booking.com: espos...
A quattro mesi dal divieto dei social ne...
NVIDIA GeForce RTX 5060 e 5060 Ti: in ar...
Rebellions, Arm e SK Telecom, nuova alle...
Modernizzazione delle app: Red Hat OpenS...
Nel mirino di Google c'è il back ...
PRAGMATA in bundle con GeForce RTX 5000:...
Le novità MOVA per il 2026: robot e impi...
Windows, stop all'attivazione telefonica...
ASUS porta la serie TUF nel formato Mini...
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: 01:06.


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