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

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

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

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

27.06.2014, 12:29. Просмотров 1560. Ответов 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
Посмотрите здесь:

Наследование 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...

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

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

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

После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Ilot
Модератор
Эксперт С++
1789 / 1164 / 226
Регистрация: 16.05.2013
Сообщений: 3,060
Записей в блоге: 5
Завершенные тесты: 1
27.06.2014, 13:06     Наследование от std::ostream #2
Забавно. На моем компиляторе (i686-w64-mingw32) в обоих случаях вывод один и тот же hello42.
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 13:10  [ТС]     Наследование от std::ostream #3
Ilot, на 2010 студии тоже выводит нормально, но меня это, к сожалению, не устраивает. Да и C++11 включить не могу по некоторым причинам.
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 13:33     Наследование от std::ostream #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
2459 / 1631 / 238
Регистрация: 24.11.2012
Сообщений: 4,009
27.06.2014, 13:40     Наследование от std::ostream #5
Сообщение было отмечено автором темы, экспертом или модератором как ответ
Цитата Сообщение от FraidZZ Посмотреть сообщение
Судя по-всему, какая-то бага
Оператор << в классе ostream перегружен только для void*. Для const char* перегружен бинарный оператор, первым аргументом принимающий ostream& - неконстантную ссылку. Следовательно, для временного объекта может быть вызван только оператор внутри класса, принимающий void*. Но это не объясняет почему в С++11 работает иначе.

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

Добавлено через 3 минуты
Цитата Сообщение от Avazart Посмотреть сообщение
Кстати а законно ли вообще наследоваться ?
А почему нет? std::ostream и так производный класс (на самом деле typedef, но сути дела не меняет), можно и расширить. Деструктор виртуальный, так что ок.
0x10
2459 / 1631 / 238
Регистрация: 24.11.2012
Сообщений: 4,009
27.06.2014, 13:57     Наследование от std::ostream #8
Цитата Сообщение от Tulosba Посмотреть сообщение
т.е. для достижения желаемого эффекта в C++03 без реализации оператора для производного класса не обойтись, я правильно понимаю?
Похоже, что так. Только я не очень представляю как это реализовать грамотно. И возникает желание просто написать обычный логгер, который не будет мимикрировать под потоки stl.
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 14:14  [ТС]     Наследование от std::ostream #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     Наследование от std::ostream #10
По сути, у нас получается класс-обертка над std::ostream
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 15:14  [ТС]     Наследование от std::ostream #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     Наследование от std::ostream #12
Tulosba, можете ответить как эксперт? Почему не перегружается (я пробовал) как-то так:
C++
1
2
3
std::ostream& operator << (const std::ostream& t, const char * n){ 
   //Здесь const_cast, иначе метод write() не работает
}
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 15:21  [ТС]     Наследование от std::ostream #13
Цитата Сообщение от FraidZZ Посмотреть сообщение
Почему не перегружается (я пробовал) как-то так:
Это для какого варианта исполнения класса?
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 15:23     Наследование от std::ostream #14
Хотя, я догадываюсь, что для const ostream& это должен быть метод с модификатором const, объявленный внутри класса.

Добавлено через 1 минуту
Tulosba, это не как обертка, а именно для вызова
C++
1
ostream(cout.rdbuf()) << "Hello";
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 15:30  [ТС]     Наследование от std::ostream #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).
0x10
2459 / 1631 / 238
Регистрация: 24.11.2012
Сообщений: 4,009
27.06.2014, 15:31     Наследование от std::ostream #16
FraidZZ, если не закапываться в детали, поток по определению не может быть константным в операции записи в него, запись == модификация.
ЗЫ Справедливо и для чтения.
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 15:38     Наследование от std::ostream #17
Да, я уже экспереминтируя понял. Спасибо за помощь!
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 16:26  [ТС]     Наследование от std::ostream #18
Ещё один момент. Есть сделать, как я написал в сообщении #11, то стандартные манипуляторы а-ля std::endl уже использовать не получится.

Добавлено через некоторое время
Хм-м. А std::hex кушает.

Добавлено совсем недавно
Снова эта перегрузка функций ... работает:
C++
1
DebugMessage() << "hello" << static_cast<std::ostream&(*)(std::ostream&)>(std::endl) << 42;
FraidZZ
Ex-Member
3897 / 1523 / 229
Регистрация: 06.01.2013
Сообщений: 4,050
Завершенные тесты: 1
27.06.2014, 16:32     Наследование от std::ostream #19
Tulosba, а почему бы не объявлять объект?
И не париться с тем, что временно созданный объект изволит кушать только const?
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
27.06.2014, 16:38     Наследование от std::ostream
Еще ссылки по теме:

ostream - C++
Вот мой класс: class BigInt { int Size; // Size - текущая длина (истинная длинна числа) int SizeMax; //...

Быстродействие std::unordered_map vs. std::unordered_multimap - C++
Привет, скажите пожалуйста, есть какие-либо существенные просадки по скорости исполнения программы, если использовать...

Lambda expressions only available with -std=c++11 or -std=gnu++11 - C++
Нужна помощь. Код: #include &lt;iostream&gt; #include &lt;string&gt; #include &lt;vector&gt; #include &lt;algorithm&gt;

Lambda expressions only available with -std=c++11 or -std=gnu++11 - C++
В студии все работает, в qt попробовал mingw - выводит варнинг lambda expressions only available with -std=c++11 or -std=gnu++11 а...

Класс ostream - C++
Как можно создать класс ostream? Такое ощущение, что кроме cout ничего из этого класса создать невозможно.


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

Или воспользуйтесь поиском по форуму:
Tulosba
:)
Эксперт С++
4392 / 3235 / 297
Регистрация: 19.02.2013
Сообщений: 9,045
27.06.2014, 16:38  [ТС]     Наследование от std::ostream #20
Цитата Сообщение от FraidZZ Посмотреть сообщение
а почему бы не объявлять объект?
Хочется сделать как Qt-шный QDebug(). Только тут ещё допить захват(создание)/освобождение(удаление) мютекса блокирующего консоль, чтобы разные потоки гадили по очереди, и не было мешанины из вывода.
Yandex
Объявления
27.06.2014, 16:38     Наследование от std::ostream
Ответ Создать тему
Опции темы

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