Модераторы: Daevaorn
  

Поиск:

Ответ в темуСоздание новой темы Создание опроса
> перегрузка ostream& operator <<, такая, чтобы std::endl работал 
V
    Опции темы
marcusmae
Дата 2.4.2008, 23:22 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


stravaganza
**


Профиль
Группа: Участник
Сообщений: 874
Регистрация: 26.3.2006

Репутация: 5
Всего: 39



Всем привет,

Множество определённых мною перегрузок оператора <<

Код

ostream& operator << (const char*);
ostream& operator << (const int&);
ostream& operator << (const char&);
ostream& operator << (const float&);
ostream& operator << (const double&);
ostream& operator << (ostream &);


не охватывает возможности принятия std::endl / std::beginl в качестве аргумента. Пробовал подсмотреть в стандартные исходники :

Код

basic_ostream<_Elem, _Traits> > operator << (basic_ostream<_Elem, _Traits>&);


с подстановкой wchar_t не катит. Как же всё-таки его правильно записать?..

Это сообщение отредактировал(а) marcusmae - 2.4.2008, 23:23


--------------------
ἀπὸ μηχανῆς θεός
PM MAIL ICQ GTalk   Вверх
bsa
Дата 2.4.2008, 23:51 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Эксперт
****


Профиль
Группа: Модератор
Сообщений: 9185
Регистрация: 6.4.2006
Где: Москва, Россия

Репутация: 63
Всего: 196



вопрос, а зачем тебе это вообще делать? чем тебя стандартное поведение не устраивает? Ты что, свой ostream пишешь?

Это сообщение отредактировал(а) bsa - 3.4.2008, 00:07
PM   Вверх
marcusmae
Дата 2.4.2008, 23:58 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


stravaganza
**


Профиль
Группа: Участник
Сообщений: 874
Регистрация: 26.3.2006

Репутация: 5
Всего: 39



bsa, я не претендую на универсальность. Достаточно перехватить стандартный вывод лишь в части пяти типов :

Код

ostream& operator << (const char*);
ostream& operator << (const int&);
ostream& operator << (const char&);
ostream& operator << (const float&);
ostream& operator << (const double&);


Это вроде бы удалось. Единственная проблема - если справа сразу endl стоит :

Код

mycout << endl;


Это приводит к ошибке (MVC++ 9.0) :

Цитата

error C2679: binary '<<' : no operator found which takes a right-hand operand of type 'overload-function' (or there is no acceptable conversion)


Как бы это обойти?.. Или я вообще ничего не понимаю? smile 

Это сообщение отредактировал(а) marcusmae - 3.4.2008, 00:02


--------------------
ἀπὸ μηχανῆς θεός
PM MAIL ICQ GTalk   Вверх
bsa
Дата 3.4.2008, 00:07 (ссылка) |  (голосов:2) Загрузка ... Загрузка ... Быстрая цитата Цитата


Эксперт
****


Профиль
Группа: Модератор
Сообщений: 9185
Регистрация: 6.4.2006
Где: Москва, Россия

Репутация: 63
Всего: 196



Кстати, стандартные типы лучше передавать по значению, разве что long double, имхо, лучше по ссылке. Ты забыл long double, short int, long int и long long int.

std::endl и прочие - это функции, которые объявлены так:
Код
extern std::basic_ostream& endl(std::basic_ostream &stream);
Поэтому для их "вывода" нужно написать соответствующие методы, которые принимают указатели на функцию:
Код
inline __ostream_type& operator<<(__ostream_type& (*__pf)(__ostream_type&));

inline __ostream_type& operator<<(__ios_type& (*__pf)(__ios_type&));

inline __ostream_type& operator<<(ios_base& (*__pf) (ios_base&));
С манипуляторами там все хитрее. Они возвращают структуру. Поэтому чтобы их "выводить", приходится писать соответствующий оператор <<:
Код
struct _Setw { int _M_n; };

inline _Setw setw(int __n)
{
    _Setw __x;
    __x._M_n = __n;
    return __x;
}

template<typename _CharT, typename _Traits>
inline basic_ostream<_CharT,_Traits>& operator<<(basic_ostream<_CharT,_Traits>& __os, _Setw __f)
{
    __os.width(__f._M_n);
    return __os;
}

PM   Вверх
marcusmae
Дата 3.4.2008, 00:24 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


stravaganza
**


Профиль
Группа: Участник
Сообщений: 874
Регистрация: 26.3.2006

Репутация: 5
Всего: 39



Код

inline ostream& operator<< (ostream& (*__pf)(__ostream_type&));

inline ostream& operator<< (ios& (*__pf)(ios&));

inline ostream& operator<<(ios_base& (*__pf) (ios_base&));


круто smile то что нужно, вроде (если правильно понял).

Манипуляторы - фиг с ними. Пока, думаю, они не понадобятся.

Спасибо! smile

Добавлено через 2 минуты и 3 секунды
Цитата(bsa @  3.4.2008,  00:07 Найти цитируемый пост)
Кстати, стандартные типы лучше передавать по значению


Ок, сделал.


--------------------
ἀπὸ μηχανῆς θεός
PM MAIL ICQ GTalk   Вверх
student0511
Дата 22.4.2008, 00:47 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Новичок



Профиль
Группа: Участник
Сообщений: 42
Регистрация: 26.3.2006

Репутация: нет
Всего: нет



У меня возникла проблема: никак не могу перегрузить операцию << для собственного класса строк Tstring.
Код

#include <iostream>
using namespace std;
/////////////////////
class Tstring
{
 private:
     char* str;
     int len;
 public:
     ///////***Designers***///////////
     Tstring(): str(NULL), len(0)
     {}
     ////////////////////////////////
     Tstring(Tstring& s): len(s.len)
     {
      str = new char [len + 1];
      strcpy(str, s.str);
     }
     ////////////////
     Tstring(char* s)
     {
      str = new char[strlen(s)+1];
      len = strlen(s);
      strcpy(str, s);
     }
     ///////***Destructor***//////
     ~Tstring()
     {
      delete[] str;
     }
     //////***Methods***/////////////////
     void show();
     Tstring operator=(const Tstring);
     Tstring operator+(const Tstring&);
     Tstring operator+=(const Tstring);
     Tstring operator-(const Tstring&);
     Tstring operator-=(const Tstring);
     operator char*();
     friend ostream& operator<<(ostream&, const Tstring&);
};

Вот описание класса: как видно operator<< является дружественной функцией, но если в теле описания данной функции обратиться к полю аргумента из private, то вылетает ошибка: error C2248: 'str' : cannot access private member declared in class 'Tstring'.
Код

ostream& operator<<(ostream& os, const Tstring& s)
{
 for(int i = 0; i<strlen(s.str); i++)
     os<<*((s.str)+i);
 return(os);
}

Компилятор: Visual C++ 6 1994 - 1998 Microsoft Corporation - может дело в нём или надо что-то прописать?

Это сообщение отредактировал(а) student0511 - 22.4.2008, 01:29
PM MAIL   Вверх
bronislav
Дата 22.4.2008, 07:37 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Опытный
**


Профиль
Группа: Участник
Сообщений: 334
Регистрация: 29.1.2008
Где: Украина::Донецк

Репутация: нет
Всего: 3



У меня возникала анологичная ошибка именно в MS VC++ 6. Как решал не помню, вроде бы переносли в public, но это конечно не правильное решение.
В более новых компиляторах такой ошибки не возникало.


--------------------
user posted image
иногда проще и быстрей обойти лужу, даже если кажется что она мелкая и путь напрямик короче - ведь она может скрывать открытый люк (с) mes
PM MAIL   Вверх
Fazil6
Дата 22.4.2008, 14:43 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Эксперт
***


Профиль
Группа: Завсегдатай
Сообщений: 1653
Регистрация: 3.5.2006
Где: Минск

Репутация: 35
Всего: 60



странная какая-то ошибка... попробовал в VC++ 6 SP5
Код

#include <iostream>
using namespace std;
/////////////////////
class Tstring;

ostream& operator<<(ostream& os, const Tstring& s);


class Tstring
{
private:
    char* str;
    int len;
public:
    ///////***Designers***///////////
    Tstring(): str(NULL), len(0)
    {}
    ////////////////////////////////
    Tstring(Tstring& s): len(s.len)
    {
        str = new char [len + 1];
        strcpy(str, s.str);
    }
    ////////////////
    Tstring(char* s)
    {
        str = new char[strlen(s)+1];
        len = strlen(s);
        strcpy(str, s);
    }
    ///////***Destructor***//////
    ~Tstring()
    {
        delete[] str;
    }
    //////***Methods***/////////////////
    void show();
    Tstring operator=(const Tstring);
    Tstring operator+(const Tstring&);
    Tstring operator+=(const Tstring);
    Tstring operator-(const Tstring&);
    Tstring operator-=(const Tstring);
    operator char*();
    friend ostream& operator<<(ostream&, const Tstring&);
};

ostream& operator<<(ostream& os, const Tstring& s)
{
    for(int i = 0; i<strlen(s.str); i++)
        os<<*((s.str)+i);
    return(os);
}


int main()
{
    Tstring str("PREVED");
    cout << str;
    return 0;
}

все собралось и работает
PM MAIL   Вверх
student0511
Дата 22.4.2008, 15:11 (ссылка) | (нет голосов) Загрузка ... Загрузка ... Быстрая цитата Цитата


Новичок



Профиль
Группа: Участник
Сообщений: 42
Регистрация: 26.3.2006

Репутация: нет
Всего: нет



Да работает. Когда прототип пишешь функции до описания класса, то работает. Очень даже странно. 
PM MAIL   Вверх
  
Ответ в темуСоздание новой темы Создание опроса
Правила форума "С++:Общие вопросы"
Earnest Daevaorn

Добро пожаловать!

  • Черновик стандарта C++ (за октябрь 2005) можно скачать с этого сайта. Прямая ссылка на файл черновика(4.4мб).
  • Черновик стандарта C (за сентябрь 2005) можно скачать с этого сайта. Прямая ссылка на файл черновика (3.4мб).
  • Прежде чем задать вопрос, прочтите это и/или это!
  • Здесь хранится весь мировой запас ссылок на документы, связанные с C++ :)
  • Не брезгуйте пользоваться тегами [code=cpp][/code].
  • Пожалуйста, не просите написать за вас программы в этом разделе - для этого существует "Центр Помощи".
  • C++ FAQ

Если Вам понравилась атмосфера форума, заходите к нам чаще! С уважением, Earnest Daevaorn

 
1 Пользователей читают эту тему (1 Гостей и 0 Скрытых Пользователей)
0 Пользователей:
« Предыдущая тема | C/C++: Общие вопросы | Следующая тема »


 




[ Время генерации скрипта: 0.1341 ]   [ Использовано запросов: 21 ]   [ GZIP включён ]


Реклама на сайте     Информационное спонсорство

 
По вопросам размещения рекламы пишите на vladimir(sobaka)vingrad.ru
Отказ от ответственности     Powered by Invision Power Board(R) 1.3 © 2003  IPS, Inc.