Форум программистов, компьютерный форум, киберфорум
Наши страницы
С++ для начинающих
Войти
Регистрация
Восстановить пароль
 
 
Рейтинг 4.69/13: Рейтинг темы: голосов - 13, средняя оценка - 4.69
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
1

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

27.06.2014, 12:29. Просмотров 2385. Ответов 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* и выводится адрес, а не строка.

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

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

Наследование ostream
Я наследовал от ostream класс, перегрузил operator&lt;&lt;(const char *) для одного...

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

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

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

24
Ilot
Эксперт С++
1832 / 1190 / 342
Регистрация: 16.05.2013
Сообщений: 3,139
Записей в блоге: 5
Завершенные тесты: 1
27.06.2014, 13:06 2
Забавно. На моем компиляторе (i686-w64-mingw32) в обоих случаях вывод один и тот же hello42.
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
27.06.2014, 13:10  [ТС] 3
Ilot, на 2010 студии тоже выводит нормально, но меня это, к сожалению, не устраивает. Да и C++11 включить не могу по некоторым причинам.
0
Eva Rosalene
T for Trans-
4050 / 1589 / 303
Регистрация: 06.01.2013
Сообщений: 4,182
Завершенные тесты: 2
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 секунды
Судя по-всему, какая-то бага
0
0x10
2571 / 1751 / 288
Регистрация: 24.11.2012
Сообщений: 4,380
27.06.2014, 13:40 5
Лучший ответ Сообщение было отмечено Eva Rosalene как решение

Решение

Цитата Сообщение от FraidZZ Посмотреть сообщение
Судя по-всему, какая-то бага
Оператор << в классе ostream перегружен только для void*. Для const char* перегружен бинарный оператор, первым аргументом принимающий ostream& - неконстантную ссылку. Следовательно, для временного объекта может быть вызван только оператор внутри класса, принимающий void*. Но это не объясняет почему в С++11 работает иначе.

Добавлено через 2 минуты
Ах, а в С++11 есть rvalue insertion. http://www.cplusplus.com/reference/ostream/ostream/operator-free/
4
Avazart
Эксперт С++
7725 / 5634 / 549
Регистрация: 10.12.2010
Сообщений: 25,416
Записей в блоге: 17
27.06.2014, 13:41 6
Кстати а законно ли вообще наследоваться ?
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
27.06.2014, 13:49  [ТС] 7
0x10, т.е. для достижения желаемого эффекта в C++03 без реализации оператора для производного класса не обойтись, я правильно понимаю?

Добавлено через 3 минуты
Цитата Сообщение от Avazart Посмотреть сообщение
Кстати а законно ли вообще наследоваться ?
А почему нет? std::ostream и так производный класс (на самом деле typedef, но сути дела не меняет), можно и расширить. Деструктор виртуальный, так что ок.
0
0x10
2571 / 1751 / 288
Регистрация: 24.11.2012
Сообщений: 4,380
27.06.2014, 13:57 8
Цитата Сообщение от Tulosba Посмотреть сообщение
т.е. для достижения желаемого эффекта в C++03 без реализации оператора для производного класса не обойтись, я правильно понимаю?
Похоже, что так. Только я не очень представляю как это реализовать грамотно. И возникает желание просто написать обычный логгер, который не будет мимикрировать под потоки stl.
1
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
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
1
Eva Rosalene
T for Trans-
4050 / 1589 / 303
Регистрация: 06.01.2013
Сообщений: 4,182
Завершенные тесты: 2
27.06.2014, 14:46 10
По сути, у нас получается класс-обертка над std::ostream
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
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;
}
1
Eva Rosalene
T for Trans-
4050 / 1589 / 303
Регистрация: 06.01.2013
Сообщений: 4,182
Завершенные тесты: 2
27.06.2014, 15:19 12
Tulosba, можете ответить как эксперт? Почему не перегружается (я пробовал) как-то так:
C++
1
2
3
std::ostream& operator << (const std::ostream& t, const char * n){ 
   //Здесь const_cast, иначе метод write() не работает
}
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
27.06.2014, 15:21  [ТС] 13
Цитата Сообщение от FraidZZ Посмотреть сообщение
Почему не перегружается (я пробовал) как-то так:
Это для какого варианта исполнения класса?
0
Eva Rosalene
T for Trans-
4050 / 1589 / 303
Регистрация: 06.01.2013
Сообщений: 4,182
Завершенные тесты: 2
27.06.2014, 15:23 14
Хотя, я догадываюсь, что для const ostream& это должен быть метод с модификатором const, объявленный внутри класса.

Добавлено через 1 минуту
Tulosba, это не как обертка, а именно для вызова
C++
1
ostream(cout.rdbuf()) << "Hello";
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
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).
1
0x10
2571 / 1751 / 288
Регистрация: 24.11.2012
Сообщений: 4,380
27.06.2014, 15:31 16
FraidZZ, если не закапываться в детали, поток по определению не может быть константным в операции записи в него, запись == модификация.
ЗЫ Справедливо и для чтения.
1
Eva Rosalene
T for Trans-
4050 / 1589 / 303
Регистрация: 06.01.2013
Сообщений: 4,182
Завершенные тесты: 2
27.06.2014, 15:38 17
Да, я уже экспереминтируя понял. Спасибо за помощь!
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
27.06.2014, 16:26  [ТС] 18
Ещё один момент. Есть сделать, как я написал в сообщении #11, то стандартные манипуляторы а-ля std::endl уже использовать не получится.

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

Добавлено совсем недавно
Снова эта перегрузка функций ... работает:
C++
1
DebugMessage() << "hello" << static_cast<std::ostream&(*)(std::ostream&)>(std::endl) << 42;
0
Eva Rosalene
T for Trans-
4050 / 1589 / 303
Регистрация: 06.01.2013
Сообщений: 4,182
Завершенные тесты: 2
27.06.2014, 16:32 19
Tulosba, а почему бы не объявлять объект?
И не париться с тем, что временно созданный объект изволит кушать только const?
0
Tulosba
:)
Эксперт С++
4749 / 3243 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
27.06.2014, 16:38  [ТС] 20
Цитата Сообщение от FraidZZ Посмотреть сообщение
а почему бы не объявлять объект?
Хочется сделать как Qt-шный QDebug(). Только тут ещё допить захват(создание)/освобождение(удаление) мютекса блокирующего консоль, чтобы разные потоки гадили по очереди, и не было мешанины из вывода.
0
27.06.2014, 16:38
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
27.06.2014, 16:38

Отсутствует оператор "<<", соответствующий этим операндам. Типы операндов: std::ostream << void
#include &lt;iostream&gt; using namespace std; void tab(int t) { for (int i =...

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

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


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

Или воспользуйтесь поиском по форуму:
20
Ответ Создать тему
Опции темы

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