最近学习了C++2.0版本的一些新的特性,利用Variadic Templates(可变参数模板)实现printf函数。
语言环境
Dev-C++ 5.11
并需要自己的环境是支持C++11的,例如:Dev-C++ 5.11可以通过以下步骤进行修改:
源码
#include <iostream>
#include <stdexcept>
using namespace std;
namespace myPrintf {
void printf(const char *s) {
while(*s) {
if(*s=='%'&&*(++s)!='%')
throw std::runtime_error("invalid format string:missing arguments");
std::cout<<*s++;
}
}
template<typename T,typename... Args>
void printf(const char *s,T value,Args... args) {
while(*s) {
if(*s=='%'&&*(++s)!='%') {
std::cout<<value<<" ";
printf(++s,args...);
return;
}
std::cout<<*s++;
}
throw std::logic_error("extra arguments provided to printf");
}
}
int main() {
int *pi=new int;
myPrintf::printf("%d%s%p%f\n",15,"this is a test",pi,3.14);
system("pause");
return 0;
}
原理
- 在调用自己实现的printf时,也就是
void printf(const char *s,T value,Args... args)
,就会分为第一个参数15,和之后的一堆; - 经过while的判断,打印出第一个参数,然后将s指针加一,即指向下一个%,再次递归调用printf;
- 以此类推,当最后只剩余一个参数的时候,就会调用
void printf(const char *s)
;
运行结果展示
注意:
- 一定要在自己的命名空间下,否则会和标准库中的printf冲突;
- 一定要对最后一个只有一个参数的函数,进行处理,否则编译器不知道该如何处理最后一个参数;
- 要保证编译器支持C++11