|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
[C++/Java] Come si fa il testing con thread concorrenti ?
Usando JUnit e CPPUnit in una applicazione multithread, come si fa a testare e verificare la correttezza della gestione della concorrenza ?
Cioè mettiamo che ci siano due thread, A e B, che devono accedere ad una classe condivisa, come si fa a scrivere un test che verifica che se A è in sezione critica B non entra in sezione critica se prima A non ne è uscito ? |
|
|
|
|
|
#2 |
|
Senior Member
Iscritto dal: Jul 2002
Messaggi: 4334
|
In caso di java e' il linguaggio stesso che se ne preoccupa,
quando un blocco e' synchronized il programmatore sa che solo un thread alla volta ci entrera'. Per il C++ non so se il supporto e' a livello cosi' "basso". Se invece intendi dimostrare che un programma non avra' mai problemi di deadlock e affini mi sembra che il problema non abbia soluzione per un prog. generico, ma solo per casi (molto) banali.
__________________
|Java Base| |
|
|
|
|
|
#3 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
In effetti non avevo pensato ai metodi synchronized...
In C++ non ci sono costrutti a livello di linguaggio che permettono di esprimere la mutua esclusione, ma questa deve essere implementata con le syscall del sistema o con alcune librerie non standard... Quindi limitandoci al C++ e al caso riportato sopra di due o più thread in mutua esclusione come si fa ? Che poi sarebbe anche il caso in cui un metodo non sia synchronized, ma la sincornizzazione avviene su un determinato oggetto all'interno del metodo... Non voglio dimostrare l'assenza di deadlock, ma volevo esprimere tramite test la necessità di una mutua esclusione... Schematicamente: MioThread A, B; ClasseCondivisa cl; A.avvia(cl); B.avvia(cl); //mettiamo che A e B girino accedendo a cl diverse volte while(A.isRunning() || B.isRunning()); assert(cl è stata acceduta in mutua esclusione); Quali costrutti si potrebbero usare per un simile test ? L'unica cosa che mi viene in mente è registrare gli accessi alla classe.... |
|
|
|
|
|
#4 |
|
Senior Member
Iscritto dal: Jul 2002
Messaggi: 4334
|
La prima cosa che mi viene in mente:
all'inizio della sezione critica ci metti: control = <qui metti qualcosa che indica univocamente il thread, tipo il nome della classe, ecc.> alla fine fai un assert che controlla che control sia ancora nel suo stato, altrimenti vuol dire che e' stata "sporcata" da un altro thread. Potrebbe essere del tutto scorretto, magari aspetta qualche "guru" della concorrenza, tipo PGI-Bis
__________________
|Java Base| |
|
|
|
|
|
#5 |
|
Senior Member
Iscritto dal: Nov 2004
Città: Tra Verona e Mantova
Messaggi: 4553
|
Ma che guru, manco ghiro!
Son talmente guru che, incuriosito dalla faccenda ho provato, in Java, a monitorare l'accesso di un Thread ad un metodo registrando codice hash e tempo di accesso e controllando, ogni volta che un Thread accede al metodo, se questo accesso sia sovrapposto a quello di un altro Thread (entrato e non ancora uscito o entrato e uscito in un periodo intersecante il tempo di accesso del nuovo Thread). Risultato del test da guinnes dei primati: due thread non sono mai in conflitto o lo sono sempre! A prescindere dalla sincronizzazione! E i mutex e i lock e i semaphores...TUTTE PALLE! C'ho qui il test c'ho! Carta canta signori! Ho quadrato il cerchio della concorrenza ho quadrato! Avete problemi di concorrenza? ANTICHI!!! Buttate nel codice il test di PGI-GURU DELLA CONCORRENZA e per magia i vostri Thread o non saranno mai in conflitto o lo saranno sempre! Ci faccio i miglioni ci faccio! |
|
|
|
|
|
#6 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
Posta un po' per curiosità |
|
|
|
|
|
#7 |
|
Senior Member
Iscritto dal: Jul 2002
Messaggi: 4334
|
Ma va guru dei miei stivali... Sei 'na sola...
__________________
|Java Base| |
|
|
|
|
|
#8 |
|
Senior Member
Iscritto dal: Dec 2000
Città: bologna
Messaggi: 1309
|
Mockare la classe condivisa? o i thread che ci accedono?
l'obbiettivo sarebbe quello di creare con un test una situazione sporca data dalla cattiva concorrenza, ma in pratica non saprei bene come fare... |
|
|
|
|
|
#9 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
Avevo pensato anche io a mockare la classe condivisa, ma è un lavoro dispendioso...soprattutto se le classi condivise sono tante...
Diciamo che non voglio testare le situazioni "sporche", anche perchè srebbe impossibile, soprattutto per le catene di lock, che sono troppo time-depend. Principalmente volevo esprimere la necessità di inserire la gestione della concorrenza.... L'obiettivo è di essere obbligati dal test ad inserire la gestione della mutua esclusione (immaginatevi di essere in un ambito Test Driven) Ultima modifica di cionci : 18-10-2006 alle 11:06. |
|
|
|
|
|
#10 | |
|
Senior Member
Iscritto dal: Dec 2000
Città: bologna
Messaggi: 1309
|
Quote:
|
|
|
|
|
|
|
#11 | |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
Quote:
Il test ovviamente deve fallire sempre, se non è gestita la mutua esclusione...ma questa condizione non è garantita... Vediamo forse qualcosa mi viene in mente per testarlo...lo elaboro oggi quando torno a casa... Però mi sa che porta ad un deadlock... |
|
|
|
|
|
|
#12 |
|
Senior Member
Iscritto dal: Nov 2004
Città: Tra Verona e Mantova
Messaggi: 4553
|
Posto ma solo per espormi al pubblico dileggio!
Teoricamente non faceva una piega. La prima istruzione del metodo acceduto è la registrazione dell'accesso del Thread che accede e l'ultima è la registrazione dell'uscita da parte dello stesso Thread. Quando è registrato un accesso, il registro controlla se il tempo di "entrata" si sovrappone con un periodo di accesso già registrato per un Thread diverso. Quello che ho scritto, però, non coincide. Un chiaro caso di cervello distribuito: metà in testa e metà nelle dita Codice:
public class CriticalAccess {
private final long acquireTime;
private final int threadId;
private volatile long releaseTime;
public CriticalAccess(int threadId, long acquireTime) {
this.acquireTime = acquireTime;
this.threadId = threadId;
}
public boolean hasThreadId(int id) {
return threadId == id;
}
public long getAcquireTime() {
return acquireTime;
}
public long getReleaseTime() {
return releaseTime;
}
public void setReleaseTime(long t) {
releaseTime = t;
}
public boolean overlaps(long time) {
if(releaseTime != 0) {
return time > acquireTime && time < releaseTime;
} else {
return time >= acquireTime;
}
}
public String toString() {
return "ID:"+threadId+"...STARTED:"+acquireTime+"...ENDED:"+releaseTime;
}
}
Codice:
import java.util.*;
public class CriticalAccessRegistry {
private final long time0 = System.nanoTime();
private ArrayList<CriticalAccess> registry = new ArrayList<CriticalAccess>();
private static final CriticalAccessRegistry sharedInstance =
new CriticalAccessRegistry();
public static CriticalAccessRegistry getSharedInstance() {
return sharedInstance;
}
public void registerAcquire(int threadId) {
long timeStamp = getTime();
putAcquire(threadId, timeStamp);
}
public void registerRelease(int threadId) {
long timeStamp = getTime();
putRelease(threadId, timeStamp);
}
private synchronized void putAcquire(int threadId, long timeStamp) {
CriticalAccess overlapped = checkOverlaps(threadId, timeStamp);
CriticalAccess ca = new CriticalAccess(threadId, timeStamp);
if(overlapped != null) {
System.out.println(ca + " conflicts with " + overlapped);
}
registry.add(ca);
}
private synchronized void putRelease(int threadId, long timeStamp) {
CriticalAccess ca = getAccess(threadId);
if(ca != null) ca.setReleaseTime(timeStamp);
}
private CriticalAccess checkOverlaps(int id, long time) {
for(int i = 0; i < registry.size(); i++) {
CriticalAccess c = registry.get(i);
if(!c.hasThreadId(id) && c.overlaps(time)) {
return c;
}
}
return null;
}
private CriticalAccess getAccess(int threadId) {
for(int i = 0; i < registry.size(); i++) {
CriticalAccess c = registry.get(i);
if(c.hasThreadId(threadId)) return c;
}
return null;
}
private long getTime() {
return (System.nanoTime() - time0) / 1000000L;
}
public void dump() {
for(CriticalAccess c : registry) {
System.out.println(c);
}
}
}
Codice:
import java.util.*;
public class SharedResource implements Runnable {
private final Random random = new Random();
public void run() {
while(true) {
CriticalAccessRegistry.getSharedInstance().
registerAcquire(Thread.currentThread().hashCode());
sharedAccess();
CriticalAccessRegistry.getSharedInstance().
registerRelease(Thread.currentThread().hashCode());
if(Thread.currentThread().isInterrupted()) {
System.out.println("Threadus interruptus!!!");
return;
}
}
}
public void sharedAccess() {
try {
Thread.sleep(100);
} catch(InterruptedException swallow) {
Thread.currentThread().interrupt();
}
}
}
Codice:
public class Test {
public static void main(String[] args) {
long testTime = Long.parseLong(args[0]);
while(CriticalAccessRegistry.getSharedInstance() == null);
SharedResource res = new SharedResource();
Thread a = new Thread(res);
Thread b = new Thread(res);
a.start();
b.start();
long startTime = System.currentTimeMillis();
while(System.currentTimeMillis() - startTime < testTime);
a.interrupt();
b.interrupt();
}
}
Comunque, quella roba lì sopra "la va no", non funziona. |
|
|
|
|
|
#13 | |
|
Senior Member
Iscritto dal: Dec 2000
Città: bologna
Messaggi: 1309
|
Quote:
ps.sto ipotizzando a livello astratto, non so quanto sia fattibile in generale, e nel tuo caso.. |
|
|
|
|
|
|
#14 |
|
Senior Member
Iscritto dal: Dec 2000
Città: bologna
Messaggi: 1309
|
http://en.wikipedia.org/wiki/Monkey_test
qua ipotizza l'utilizzo del monkey test per i thread. non approfondisce molto, ma magari qualcosa sul web si trova |
|
|
|
|
|
#15 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
Cioe' si tratterebbe di fare avanzare N thread casualmente in modo di produrre con alta probabilita' un errore ?
Mi sembra una situazione un po' limite, cioe' significherebbe comunque un test che, in caso di codice errato, comunque ha una certa probabilita' di risultare verde... Pero' magari con un numero elevato di thread questa probabilita' potrebbe essere molto piccola...boh... |
|
|
|
|
|
#16 | |
|
Senior Member
Iscritto dal: Dec 2001
Città: Milano
Messaggi: 545
|
Quote:
__________________
Angus the Hunter @ Realm of magic | Angus Young @ Batracer °SetiEmperor°| Ninja Technologies { qualunque cosa sia, è veloce e fa male (cit.) } |
|
|
|
|
|
|
#17 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
Mmmmhhh...sinceramente ancora non mi convince... Cercherò di approcciare la cosa in modo diverso
|
|
|
|
|
|
#18 | |
|
Senior Member
Iscritto dal: Dec 2001
Città: Milano
Messaggi: 545
|
Quote:
__________________
Angus the Hunter @ Realm of magic | Angus Young @ Batracer °SetiEmperor°| Ninja Technologies { qualunque cosa sia, è veloce e fa male (cit.) } |
|
|
|
|
|
|
#19 |
|
Senior Member
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
|
Sì, l'ho letto qualche giorno fa...
|
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 19:17.











|







