Chinaunix首页 | 论坛 | 博客
  • 博客访问: 393663
  • 博文数量: 273
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 1430
  • 用 户 组: 普通用户
  • 注册时间: 2018-02-02 15:57
文章分类

全部博文(273)

文章存档

2018年(273)

我的朋友

分类: C/C++

2018-07-26 15:38:51

还是邓俊辉老师数据结构中List那一章的例子。
List的遍历问题。

main.cpp里调用
PRINT ( La ); // La是一个自定义的List对象

PRINT这个宏的定义就在main.cpp里
#define PRINT(x) { print(x); crc(x); checkOrder(x); }

print的声明在UniPrint/print.h里

template <typename T> static void print ( T& x ) {  UniPrint::p ( x );  } #include "print_implementation.h"

注意C++模板类的定义和实现必须要在同一个文件中,通常是头文件,因为编译器要看到模板实现才能展开模板。
但是print.h里的模板类UniPrint只有方法的声明,没有方法的实现。
所以print.h的末尾引入了print_implementation.h这个头文件。UniPrint::p的实现就在这个头文件里。
这也是C++模板类的常用写法。

print_implementation.h里又引入了Print_traversable.h,UniPrint::p的真正实现在Print_traversable.h里。(windows上C++头文件不分大小写)

print_traversable.h

template <typename T> //元素类型 void UniPrint::p ( T& s ) { //引用 printf ( "%s[%d]*%d:\n", typeid ( s ).name(), &s, s.size() ); //基本信息 s.traverse ( print ); //通过print()遍历输出所有元素 printf ( "\n" );
}

运行到s.traverse( print );这一句的时候会跳到traverse方法里去。

list.h

template <typename T> void List::traverse ( void ( *visit ) ( T& ) )//借助函数指针机制遍历 { for (ListNodePosi(T) p = header->succ; p != trailer; p = p->succ) { printf("%s", "sss"); visit(p->data); } // 因为T已经被替换成了int。所以这里的visit其实是UnitPrint::p的模板实例,在print_basic.cpp里 }

从traverse方法来看,它接收的是一个函数指针,这个函数接收一个T型引用的参数,且没返回值。
所以s.traverse ( print );中print方法也应该接收一个T型引用的参数。
纵观print.h中只有这一句符合条件:template static void print ( T& x ) { UniPrint::p ( x ); }

咦?不对呀,这怎么又回来了?怕不是死循环?

是的,这个地方卡了我好久。后来我想通了。PRINT里调用print的时候,会先走到print_traversable.h里去执行UniPrint::p的实现,


阅读(1075) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~