C ++ global extern "C" друг не может получить доступ к приватному члену в классе пространства имен - PullRequest
10 голосов
/ 25 июля 2011

Пожалуйста, рассмотрите код:

#include    <iostream>

using namespace std;

extern  "C"
void    foo( void );

namespace   A
{
    template< int No >
    class   Bar
    {
    private:
        friend  void    ::foo( void );

        static void private_func( int n );
    };

    template< int No >
    void    Bar< No >::private_func( int n )
    {
        cout << "A:Bar< " << No << ">::private_func( " << n << " )" << endl;
    }
}

extern  "C"
void    foo( void )
{
    A::Bar< 0 >::private_func( 1 );
}

int main( )
{
    cout << " ---- " << endl;
    foo( );
}

G ++ дает:

> g++ -Wall -o extern_c extern_c.cpp
extern_c.cpp: In function ‘void foo()’:
extern_c.cpp:20:7: error: ‘static void A::Bar<No>::private_func(int) [with int No = 0]’ is private
extern_c.cpp:29:31: error: within this context

Если я прокомментирую namspace A, он будет правильно скомпилирован и запущен.

Чего мне не хватает?

Я посмотрел похожие темы, но не смог найти ни одной, подходящей для моей проблемы.

Спасибо людям.


EDIT:

Теперь я убежден, что extern "C" не имеет никакого отношения к проблеме. Пожалуйста, игнорируйте это.

Ответы [ 3 ]

4 голосов
/ 25 июля 2011

Это ошибка g ++.Существует в 4.4, исправлено в 4.6.

UPD: кажется, что это вызвано комбинацией template и namespace.extern "C" не имеет значения, так как может быть закомментировано и ошибка остается.

2 голосов
/ 26 июля 2011

Я не знаю объяснения, но если вы поместите foo () в пространство имен, это сработает.

#include    <iostream>

using namespace std;

namespace C
{
    extern  "C"
    void    foo( void );
}

namespace   A
{
    template< int No >
    class   Bar
    {
    private:
        friend  void    C::foo( void );

        static void private_func( int n );
    };

    template< int No >
    void    Bar< No >::private_func( int n )
    {
        cout << "A::Bar< " << No << ">::private_func( " << n << " )" << endl;
    }
}


namespace C
{
    extern  "C"
    void    foo( void )
    {
        A::Bar< 0 >::private_func( 1 );
    }
}

int main( )
{
    cout << " ---- " << endl;
    C::foo( );
}

И результат:

bbcaponi@bbcaponi friends]$ g++ -Wall namespace_friend.cpp -o namespace_friend
[bbcaponi@bbcaponi friends]$ ./namespace_friend
 ----
A::Bar< 0>::private_func( 1 )
0 голосов
/ 25 июля 2011

Вам нужно объявить друга как extern "C". Ваше текущее объявление друга находит друга foo в глобальном пространстве имен с искажением имен в C ++.

...