Форум программистов, компьютерный форум CyberForum.ru

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 10, средняя оценка - 4.90
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
#1

Наследование от std::ostream - C++

27.06.2014, 12:29. Просмотров 1654. Ответов 24
Метки нет (Все метки)

Имеем код:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <iostream>
 
class DebugMessage : public std::ostream
{
public:
    DebugMessage() : std::ostream( std::cout.rdbuf() ) { }
    virtual ~DebugMessage() { *this << std::endl; }
};
 
int main(int argc, char *argv[])
{
    DebugMessage() << "hello" << 42;
 
    return 0;
}
На C++11 выводит
hello42
Без С++11 выводит
0x8048e2842
т.е. аргумент const char* интерпретируется как const void* и выводится адрес, а не строка.

Кто-нибудь может объяснить, почему?
Лучшие ответы (1)
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
27.06.2014, 12:29
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Наследование от std::ostream (C++):

отсутствует оператор "<<" соответствующий этим операндам (std::ostream << const std::string) - C++
В 20 строке подсвечиваются красным знаки &lt;&lt;. Пишет, что &quot;отсутствует оператор &quot;&lt;&lt;&quot; соответствующий этим операндам (std::ostream &lt;&lt; const...

Наследование ostream - C++
Я наследовал от ostream класс, перегрузил operator&lt;&lt;(const char *) для одного типа данных, и создал объект newcout, однако он почему-то...

Ошибка undefined reference to `operator<<(std::ostream&, Account)' - C++
Всем здравствуйте! На днях столкнулся с ошибкой, с которой еще не раз не сталкивался, и не знаю как ее исправить. Ошибка выглядит так: ...

Ошибка: отсутствует оператор >>,соответствующий этим операндам типы операндов: std::ostream>>const double - C++
Помогите или объясните в чем моя ошибка?? Вот программный код! заголовочный файл #ifndef _ZAGFILE_H_ #define _ZAGFILE_H_ #include...

Что может выступать в качестве аргумента функции, если тип параметра указан как std::ostream? - C++
#include &lt;iostream&gt; void func1(std::ostream so); void main() { func1(аргумент); std::cin.get(); } void...

C2678 бинарный ">>": не найден оператор, принимающий левый операнд типа "std::ostream" - C++
Код: #include &lt;iostream&gt; #include &lt;memory&gt; #include &lt;typeinfo&gt; #include &lt;ctime&gt; #include &lt;windows.h&gt; #include &lt;ostream&gt; ...

Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Ilot
Модератор
Эксперт С++
1809 / 1166 / 226
Регистрация: 16.05.2013
Сообщений: 3,070
Записей в блоге: 5
Завершенные тесты: 1
27.06.2014, 13:06 #2
Забавно. На моем компиляторе (i686-w64-mingw32) в обоих случаях вывод один и тот же hello42.
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 13:10  [ТС] #3
Ilot, на 2010 студии тоже выводит нормально, но меня это, к сожалению, не устраивает. Да и C++11 включить не могу по некоторым причинам.
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 13:33 #4
У меня те же проблемы. Выводит адрес. У меня вопрос - как пергружен оператор << для std::ostream ?

Добавлено через 19 минут
Вот вам на раздумья:
C++
1
2
3
4
5
6
7
8
9
10
11
12
#include <iostream>
 
class mycout : public std::ostream {
public:
  mycout() : std::ostream(std::cout.rdbuf()) { }
};
 
int main() {
  mycout z;
  mycout() << "Hello, World !!!" << std::endl;
  z<<"Heelo, World!!!"<<std::endl;
}
Вывод:
Код
0x4afe0
Heelo, World!!!
Добавлено через 54 секунды
Судя по-всему, какая-то бага
0x10
2460 / 1632 / 238
Регистрация: 24.11.2012
Сообщений: 4,015
27.06.2014, 13:40 #5
Сообщение было отмечено автором темы, экспертом или модератором как ответ
Цитата Сообщение от FraidZZ Посмотреть сообщение
Судя по-всему, какая-то бага
Оператор << в классе ostream перегружен только для void*. Для const char* перегружен бинарный оператор, первым аргументом принимающий ostream& - неконстантную ссылку. Следовательно, для временного объекта может быть вызван только оператор внутри класса, принимающий void*. Но это не объясняет почему в С++11 работает иначе.

Добавлено через 2 минуты
Ах, а в С++11 есть rvalue insertion. http://www.cplusplus.com/reference/o...operator-free/
Avazart
Эксперт С++
7148 / 5325 / 276
Регистрация: 10.12.2010
Сообщений: 23,566
Записей в блоге: 17
27.06.2014, 13:41 #6
Кстати а законно ли вообще наследоваться ?
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 13:49  [ТС] #7
0x10, т.е. для достижения желаемого эффекта в C++03 без реализации оператора для производного класса не обойтись, я правильно понимаю?

Добавлено через 3 минуты
Цитата Сообщение от Avazart Посмотреть сообщение
Кстати а законно ли вообще наследоваться ?
А почему нет? std::ostream и так производный класс (на самом деле typedef, но сути дела не меняет), можно и расширить. Деструктор виртуальный, так что ок.
0x10
2460 / 1632 / 238
Регистрация: 24.11.2012
Сообщений: 4,015
27.06.2014, 13:57 #8
Цитата Сообщение от Tulosba Посмотреть сообщение
т.е. для достижения желаемого эффекта в C++03 без реализации оператора для производного класса не обойтись, я правильно понимаю?
Похоже, что так. Только я не очень представляю как это реализовать грамотно. И возникает желание просто написать обычный логгер, который не будет мимикрировать под потоки stl.
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 14:14  [ТС] #9
Кривое решение:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#include <iostream>
     
class DebugMessage
{
private:
    std::ostream& stream;
public:
    DebugMessage() : stream( std::cout ) { }
    virtual ~DebugMessage() { stream << std::endl; }
    operator std::ostream&() { return stream; }
};
     
int main(int argc, char *argv[])
{
    static_cast<std::ostream&>(DebugMessage()) << "hello" << 42;
    return 0;
}
http://ideone.com/Ic7QCh

Добавлено через 8 минут
Или всё же лучше так:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
#include <iostream>
 
class DebugMessage 
{
private: 
    std::ostream& stream;
public:
    DebugMessage() : stream( std::cout ) { }
    virtual ~DebugMessage() { stream << std::endl; }
    
    template<typename T>
    std::ostream& operator<<( const T& t ) const
    {
        return stream << t;
    }
};
 
int main(int argc, char *argv[])
{
    DebugMessage() << "hello" << 42;
    return 0;
}
http://ideone.com/lmcSD3
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 14:46 #10
По сути, у нас получается класс-обертка над std::ostream
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 15:14  [ТС] #11
Цитата Сообщение от FraidZZ Посмотреть сообщение
По сути, у нас получается класс-обертка над std::ostream
Да. Переходим от публичного наследования к приватному со всеми вытекающими.

Добавлено через 14 минут
Кстати, вероятно предыдущий вариант следует ещё немного преобразовать, чтобы каждый последующий вызов operator<< в цепочке всё-таки срабатывал на нашем объекте, а не на std::ostream:
C++
1
2
3
4
5
6
template<typename T>
const DebugMessage& operator<<( const T& t ) const
{
   stream << t;
   return *this;
}
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 15:19 #12
Tulosba, можете ответить как эксперт? Почему не перегружается (я пробовал) как-то так:
C++
1
2
3
std::ostream& operator << (const std::ostream& t, const char * n){ 
   //Здесь const_cast, иначе метод write() не работает
}
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 15:21  [ТС] #13
Цитата Сообщение от FraidZZ Посмотреть сообщение
Почему не перегружается (я пробовал) как-то так:
Это для какого варианта исполнения класса?
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 15:23 #14
Хотя, я догадываюсь, что для const ostream& это должен быть метод с модификатором const, объявленный внутри класса.

Добавлено через 1 минуту
Tulosba, это не как обертка, а именно для вызова
C++
1
ostream(cout.rdbuf()) << "Hello";
Tulosba
:)
Эксперт С++
4393 / 3236 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 15:30  [ТС] #15
FraidZZ,
Такой вариант ответа устроит?
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#include <iostream>
#include <cstring>
 
std::ostream& operator << ( std::ostream& t, const char * n)
{ 
    for( int i=0; i<3; ++i )
    {
        t.write( n, strlen(n) );
    }
    return t;
}
 
int main() {
    
    std::cout << "ho" << std::endl;
    
    return 0;
}
http://ideone.com/u8hjnQ

Я думаю, у Вас проблема из-за несогласованности возвращаемого и принимаемого типов. (const/не-const).
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
27.06.2014, 15:30
Привет! Вот еще темы с ответами:

Наследование от std::basic_string - C++
Суть задачки: надо с текста считать слова через istream_iterator откидывая знаки пунктуации в конце слов. Написал вот так, отнаследовав от...

ostream &operator<< (ostream &output, const Array &obj) - что означает эта строка? - C++
void Array::getArray() // вывод массива { for (int ix = 0; ix &lt; size; ix++) cout &lt;&lt; setw(5) &lt;&lt; ptr; // вывод элементов...

friend ostream &operator<<(ostream &stream, MyClass o); - C++
Что означает данная строчка которую обычно пишут в конце класса? friend ostream &amp;operator&lt;&lt;(ostream &amp;stream, MyClass o);

Создание своего класса исключений, наследование std::exception - C++
хотел создать свой класс исключений пронаследовав от std::exception, но немного запутался в конструкторах... конструктор же не наследуется,...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
Yandex
Объявления
27.06.2014, 15:30
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru