C++中cout<

恋喵大鲤鱼
博客专家认证
2016-05-03 06:54:52
我知道cout<<endl;的作用是刷新输出缓冲流并换行。其实质是cout.operator(endl);endl在VC++中可以找到其定时,本质上是一个内联函数。cout对象的operator<<()输出运算符函数有一个重载版本,用于接收endl类型的函数指针。具体可参见:http://blog.csdn.net/k346k346/article/details/49981695。

但是对于cout<<setw(8)有点疑问。setw()确实一个函数调用,查找其定义,setw()的函数返回值类型是_Smanip<streamsize>。再找_Smanip<streamsiz>的定义可以得到如下:
template<class _Arg>
struct _Smanip
{ // store function pointer and argument value
_Smanip(void (__cdecl *_Left)(ios_base&, _Arg), _Arg _Val)
: _Pfun(_Left), _Manarg(_Val)
{ // construct from function pointer and argument value
}

void (__cdecl *_Pfun)(ios_base&, _Arg); // the function pointer
_Arg _Manarg; // the argument value
};


streamsiz的定义是:
typedef _Longlong streamsize;


但是我查找了cout.operator()函数的定义,并没有发现有接收_Smanip<streamsiz>类型的重载版本,那么为什么cout<<setw
(8);可以正常运行呢?其本质又是什么呢?请大牛解答,谢谢!
...全文
714 5 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
赵4老师 2016-05-04
  • 打赏
  • 举报
回复
理解讨论之前请先学会如何观察! 计算机组成原理→DOS命令→汇编语言→C语言(不包括C++)、代码书写规范→数据结构、编译原理、操作系统→计算机网络、数据库原理、正则表达式→其它语言(包括C++)、架构…… 对学习编程者的忠告: 多用小脑和手,少用大脑、眼睛和嘴,会更快地学会编程! 眼过千遍不如手过一遍! 书看千行不如手敲一行! 手敲千行不如单步一行! 单步源代码千行不如单步Debug版对应汇编一行! 单步Debug版对应汇编千行不如单步Release版对应汇编一行! 不会单步Release版对应汇编?在你想单步Release版C/C++代码片断的前面临时加一句DebugBreak();重建所有,然后在IDE中运行。(一般人我不告诉他!单步类的实例“构造”或“复制”或“作为函数参数”或“作为函数返回值返回”或“参加各种运算”或“退出作用域”的语句对应的汇编代码几步后,就会来到该类的“构造函数”或“复制构造函数”或“运算符重载”或“析构函数”对应的C/C++源代码处。 VC调试时按Alt+8、Alt+7、Alt+6和Alt+5,打开汇编窗口、堆栈窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应堆栈、内存和寄存器变化,这样过一遍不就啥都明白了吗。 对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。
赵4老师 2016-05-04
  • 打赏
  • 举报
回复
具体实现才是真正的真理,标准只是神马浮云。个人意见。 知道折半查找法从论文中出现到无错误的代码实现之间隔了多少年吗? Talk is cheap, show me the code. 不要迷信书、考题、老师、回帖; 要迷信CPU、编译器、调试器、运行结果。 并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。 任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实! 有人说一套做一套,你相信他说的还是相信他做的? 其实严格来说这个世界上古往今来所有人都是说一套做一套,不是吗? 不要写连自己也预测不了结果的代码!
fefe82 2016-05-04
  • 打赏
  • 举报
回复

  struct _Setw { int _M_n; };

  /**
   *  @brief  Manipulator for @c width.
   *  @param  __n  The new width.
   *
   *  Sent to a stream object, this manipulator calls @c width(__n) for
   *  that object.
  */
  inline _Setw
  setw(int __n)
  { return { __n }; }

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

  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;
    }

gcc 5.2.1 的,仅供参考。
恋喵大鲤鱼 2016-05-04
  • 打赏
  • 举报
回复
引用 楼主 K346K346 的回复:
我知道cout<<endl;的作用是刷新输出缓冲流并换行。其实质是cout.operator(endl);endl在VC++中可以找到其定时,本质上是一个内联函数。cout对象的operator<<()输出运算符函数有一个重载版本,用于接收endl类型的函数指针。具体可参见:http://blog.csdn.net/k346k346/article/details/49981695。 但是对于cout<<setw(8)有点疑问。setw()确实一个函数调用,查找其定义,setw()的函数返回值类型是_Smanip<streamsize>。再找_Smanip<streamsiz>的定义可以得到如下:
template<class _Arg>
	struct _Smanip
	{	// store function pointer and argument value
	_Smanip(void (__cdecl *_Left)(ios_base&, _Arg), _Arg _Val)
		: _Pfun(_Left), _Manarg(_Val)
		{	// construct from function pointer and argument value
		}

	void (__cdecl *_Pfun)(ios_base&, _Arg);	// the function pointer
	_Arg _Manarg;	// the argument value
	};
streamsiz的定义是:
typedef _Longlong streamsize;
。 但是我查找了cout.operator()函数的定义,并没有发现有接收_Smanip<streamsiz>类型的重载版本,那么为什么cout<<setw (8);可以正常运行呢?其本质又是什么呢?请大牛解答,谢谢!
所给的链接文章并没有真正的说明cout<<setw(8);实现的原理。
dustpg 2016-05-03
  • 打赏
  • 举报
回复
http://en.cppreference.com/w/cpp/io/manip/setw C++标准并未表明其具体实现方式,具体实现依赖于编译器/stl实现者,不是标准规定的不要拿具体的实现当作真理

65,186

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧