PDA

View Full Version : [C++] friend function


tuccio`
15-01-2011, 14:29
Ho un problemino, probabilmente dovuto al fatto che ho "imparato" il C++ come uno straniero impara una lingua vivendo nel paese in cui si parla e mi manca sempre qualcosa :asd:

Allora, ho una classe Thread (che devo scrivere da me perché devo fare un esame in cui devo fare un progetto che giri sia su windows che su linux)

Ovviamente pthread e windows thread usano tipi di "routine" diversi.. le routine di windows ritornano un DWORD, mentre i pthread ritornano un void*

Nella mia "astrazione", allora, l'exit code del thread sarà sempre un intero (che è semplicemente un campo della classe) e invece il return del thread vero e proprio (quello dell'os insomma) lo uso per distinguere "come" un thread è terminato

Ora, quest'introduzione non c'entra niente, ma era solo per far capire il motivo per cui ho bisogno di una funzione "wrapper" da passare a pthread_create e CreateThread


Thread::exitcode_t Wrapper(void *arg)
{
struct Thread::WrapperArg *warg = (struct Thread::WrapperArg*)arg;
Thread *t = warg->t;
void *rarg = warg->arg;

// Segnala che il thread è partito
try
{
t->OnThreadStart();
}
catch (int ex)
{
// Valore di ritorno che provocherà un errore nella funzione Start
// così che il chiamante possa sapere che il thread non è partito.
return Thread::TERMINATED;
}

// Semplicemente esegue la routine del thread e ne salva l'exitcode
t->exitcode = t->routine(rarg);

// Alla fine rimette il thread in stato inactive
t->GoInactive();
t->ranonce = true;

// Ritorna 0 se il thread termina normalmente
return (Thread::exitcode_t)Thread::OK;
}


La funzione deve essere friend per vedere i campi privati della classe, quindi nella classe Thread ho:


friend Thread::exitcode_t Wrapper(void *arg);


Il problema è: se questa implementazione la metto nel file Thread.h è tutto ok, se la metto nel file .cpp visual studio dice che non posso vedere i campi privati

Non posso implementarla in un file .cpp esterno? La questione sarebbe che ci sono parti legate al sistema operativo (che ora ho "incapsulato" nei metodi GoInactive() e OnThreadStart(), però comunque mi sembra un po' una porcata lasciare l'implementazione nello header)

wingman87
15-01-2011, 14:50
La funzione friend non deve specificare la classe cui appartiene, quindi invece di:
Thread::exitcode_t Wrapper(void *arg)
dovrebbe essere:
exitcode_t Wrapper(void *arg)
In questo modo la stessa funzione può anche essere amica di più classi.

tuccio`
15-01-2011, 14:54
no no quello c'è perché il tipo exitcode_t è un typedef che sta nella classe

è void* su linux e DWORD su win

wingman87
15-01-2011, 15:04
Hai ragione, che scemo :doh:
Ecco cosa succede a studiare un linguaggio senza scrivere neanche una linea di codice.

Shinnok.Exor
15-01-2011, 15:59
non ho capito bene il tuo problema.

Cioè le funzioni friend le usi dichiarandone il prototipo nella classe, e poi la puoi scrivere dove vuoi basta che la implementi.


#include <iostream>

...

class Classe {
friend int amica ();
private:
int num;
};

...

int amica() {
Classe c;
std::cout << "Var privata: " << c.num;
}

tuccio`
16-01-2011, 02:04
io ho messo nella classe:

friend Thread::exitcode_t Wrapper(void *arg);

poi nel file cpp:


Thread::exitcode_t Wrapper(void *arg)
{
struct Thread::WrapperArg *warg = (struct Thread::WrapperArg*)arg;
Thread *t = warg->t;
void *rarg = warg->arg;
DWORD rvalue;

// Segnala che il thread è partito
if (!t->sem->Signal())
{

// Se c'è un errore, la funzione Start() fallirà
// e questo thread verrà terminato.
rvalue = Thread::TERMINATED;

}
else
{

// Semplicemente esegue la routine del thread e ne salva l'exitcode
t->exitcode = t->routine(rarg);
t->cterm = true;

// Alla fine rimette il thread in stato inactive
t->GoInactive();
rvalue = Thread::OK;
}

return rvalue;
}


(sì è cambiata un po' di roba :X)

e il risultato è:


1>------ Inizio compilazione: Progetto: sxl, Configurazione: Debug Win32 ------
1>C:\Program Files (x86)\MSBuild\Microsoft.Cpp\v4.0\Microsoft.CppBuild.targets(299,5): warning MSB8004: la directory Output non termina con una barra finale. Nell'istanza della compilazione verrà aggiunta la barra poiché è necessaria per consentire la corretta valutazione della directory Output.
1> CommonThread.cpp
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(13): error C2248: 'sxl_mt::Thread::WrapperArg': impossibile accedere a private struct dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(107): vedere la dichiarazione di 'sxl_mt::Thread::WrapperArg'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(13): error C2248: 'sxl_mt::Thread::WrapperArg': impossibile accedere a private struct dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(107): vedere la dichiarazione di 'sxl_mt::Thread::WrapperArg'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(19): error C2248: 'sxl_mt::Thread::sem': impossibile accedere a private membro dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(91): vedere la dichiarazione di 'sxl_mt::Thread::sem'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(24): error C2248: 'TERMINATED': impossibile accedere a private enumeratore dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(80): vedere la dichiarazione di 'TERMINATED'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(31): error C2248: 'sxl_mt::Thread::exitcode': impossibile accedere a private membro dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(69): vedere la dichiarazione di 'sxl_mt::Thread::exitcode'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(31): error C2248: 'sxl_mt::Thread::routine': impossibile accedere a private membro dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(66): vedere la dichiarazione di 'sxl_mt::Thread::routine'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(32): error C2248: 'sxl_mt::Thread::cterm': impossibile accedere a private membro dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(72): vedere la dichiarazione di 'sxl_mt::Thread::cterm'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(35): error C2248: 'sxl_mt::Thread::GoInactive': impossibile accedere a private membro dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(97): vedere la dichiarazione di 'sxl_mt::Thread::GoInactive'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
1>e:\università\programmazione di sistema\progetto\src\commonthread.cpp(36): error C2248: 'OK': impossibile accedere a private enumeratore dichiarato nella classe 'sxl_mt::Thread'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(77): vedere la dichiarazione di 'OK'
1> e:\università\programmazione di sistema\progetto\inc\thread.h(33): vedere la dichiarazione di 'sxl_mt::Thread'
2>------ Inizio compilazione: Progetto: threadtest, Configurazione: Debug Win32 ------
2> test.cpp
2>LINK : fatal error LNK1104: impossibile aprire il file 'sxl.lib'


il problema sparisce se implemento la funzione nella classe

ps: sparisce anche mettendo static al posto di friend :E una funzione public static si comporta come una friend? (su msvc sì, è così anche nello standard iso?)

Shinnok.Exor
17-01-2011, 01:29
Ah eheh.
Ci scommettevo che lavori con i namespaces :)

Ti incasinano il codice, cioè ti devi ricordare che ci sono ed ogni cosa dichiarata nel namespace ne fa parte, anche una funzione friend

quindi devi cambiare ogni cosa aggiungendo il namespace, anche le varie struct ecc...

Es.

sxl_mt::Thread::exitcode_t sxl_mt::Wrapper(void *arg)
{


Se la dichiari nella classe ovviamente funziona perchè è dichiarata nel namespace.

tuccio`
17-01-2011, 01:34
ehm, sì così in effetti va

io pensavo che quella non fosse una "dichiarazione", ma soltanto specificare che la funzione Wrapper (quindi fuori dal namespace) è una funzione friend

grazie mille :v