目录
模板
- 使用模板我们就可以写一个函数,而它的参数可以接收多种类型。比如:
// 普通函数, 只能接受 int 类型参数,如果传入其他类型需要类型转换
int sum(int a, int b) {
return (a + b);
}
double sum(double a, double b) {
return (a + b);
}
- 由于不管是 int 参数、double参数,我们的 sum 函数求和内部代码都是一样的 (a + b),那么写多遍就变得麻烦,因此我们可以用模板的代替,只写一次代码,然后编译器会帮我们生成多个版本出来:
// 模板
template<typename T>
T sum(T a, T b) {
return (a + b);
}
// 编译会生成:
int sum(int a, int b) {
return (a + b);
}
double sum(double a, double b) {
return (a + b);
}
- c++的模板是编译期就会展开的,编译期会根据你使用到的类型,生成 template <int> ... 版本、template<double> ... 版本,而原本写的 template<T> ... 其实就是一个配置一样,运行时是不存在的,只是为了告诉编译期应该怎么生成代码。
可变参数模板
- 上面的求和只能2个数求和,那我如果有3个、4个、5个 ... 呢,难道要都写一遍吗:
int sum(int a, int b, int c) {
return (a + b + c);
}
int sum(int a, int b, int c, int d) {
return (a + b + c + d);
}
// ......
- 这时就可以请出可变参数模板来了,我们可以用三个点 来表示参数个数是不确定的:
template<typename T>
T sum(T... args) {
// 展开求和
}
- 但展开这个参数包 args 并不是当成数组遍历的,而是需要通过递归来实现:
template<typename T>
T mysum(T item) {
return item;
}
template<typename T>
T mysum(T item, T... args) {
return mysum(item) + mysum(args);
}
/// 入口
template<typename T>
T sum(T... args) {
return mysum(args);
}
- 最后一个函数 T sum(T... args) 才是我们暴露给用户使用的函数,而它其实是把参数包 args 转给了mysum(T item, T... args),可见原本在 sum 的参数包被拆出第一个参数 item,和剩下的其他参数 args 调用 mysum,假如 args 不存在,则是直接调用了 mysum(T item)
- 而 mysum(T item, T... args) 干嘛了呢, 它将 item 传给 mysum(T item) 处理,并将剩下的传给了"自己"!是的,此时其实就是递归,但请注意,以往我们递归调用的是函数本身,但这次递归不一样,举个例子来看看:
- 假如我们现在要调用 sum 了:sum(1, 2, 3, 4, 5),编译器就会生成接受5个参数的sum:
int sum(int arg0, int arg1, int arg2, int arg3, int arg4);
- 然后sum里面又调用了mysum,编译器接着按照mysum的声明,拆分sum传过来的5个参数为1 + 4生成:
int mysum(int item, int arg1, int arg2, int arg3, int arg4);
- 那接下来就看这个参数列表是 1+4 的mysum里面了,编译器先生成 mysum(T) 对应的 mysum(int):
int mysum(int item);
- 然后1+4的mysum里还把剩下的4个参数递归传给了自己这个模板,那就得把 4 再拆分成 1 + 3 生成mysum
int mysum(int item, int arg2, int arg3, int arg4);
- 到这里你可能已经理解它是怎么拆参数包的了,虽然叫递归,但其实每一次拆包调用的并不是函数本身,而是源于同一个模板生成的少一个参数的函数。
- 再往下就是接着生成:
int mysum(int item, int arg3, int arg4);
int mysum(int item, int arg4);
- 最终参数包 args 只有一个参数时就不用再生成了,它会直接去调用之前生成的 mysum(T item)
不定类型数量
- 前面的可变参数模板还不够彻底,它生成的函数接受的类型只有一个 T,那我能不能每一个参数类型都不一样呢?那么就需要对 template 那一行动手脚了:
template<typename T1, typename T2, typename ...Args>
T1 mysum(T2 item, Args... args);
- 可以看到,template内,我们单独声明了 返回值类型是 T1,第一个参数类型是 T2,而后面的参数类型则不确定,因此用 typename...Args 声明,在 mysum 的参数列表中 (T2 item, Args... args),Args...就表示args的数量是不确定的,由于Args本身类型数量也不确定,因此整体就表示 这里可以接受多个相同或不同类型的参数。
实现printf
- 有了上面的知识,实现printf就简单了,我们可以让它接收 第一个是字符串类型,然后加上多个任意类型的参数,并递归解析输出即可:
template<typename T>
int printNum(const std::string& str, int index, T&& item) {
auto doShift = true;
while (index < str.size()) {
if (index == (str.size() - 1) || str[index] != '%') {
cout << str[index];
++index;
}
else if(doShift) {
doShift = false;
++index;
switch (str[index]) {
case 's':
case 'd':
cout << item;
break;
}
++index;
}
else {
return index - 1;
}
}
return index - 1;
}
template<typename T, typename... Args>
int printNum(const std::string& str, int index, T&& item, Args&&... args) {
index = printNum(str, index, item);
return printNum(str, index + 1, std::forward<Args>(args)...);
}
template<typename ...Args>
void myPrintNum(const std::string& str, Args&&... args) {
int index = 0;
while (index < str.size()) {
if (str[index] != '%') {
cout << str[index];
++index;
}
else {
break;
}
}
index = printNum(str, index, std::forward<Args>(args)...);
++index;
while (index < str.size()) {
cout << str[index];
++index;
}
}
int main() {
// 调用,参数数量允许和字符串内数量不同,不会出错
myPrintNum("output: %d, %d, %d, %d ---", 123, 100.123, true);
return 0;
}
Excellent blog you’ve got here.. It’s hard to find excellent writing like yours these days. I really appreciate people like you! Take care!!
My spouse and I stumbled over here different page and thought I might check things out. I like what I see so now i am following you. Look forward to looking into your web page yet again.
cytotec pills buy online https://lipitor.guru/# lipitor prescription drug
lasix tablet
https://lipitor.guru/# lipitor 40mg
buy cytotec pills online cheap https://tamoxifen.bid/# tamoxifen for gynecomastia reviews
generic lasix
You have made some good points there. I checked on the internet for more info about the issue and found most people will go along with your views on this site.
http://cytotec.pro/# buy cytotec over the counter
tamoxifen side effects forum: buy tamoxifen online – tamoxifen and depression
buy furosemide online: furosemide online – lasix 40 mg
buy cytotec in usa http://tamoxifen.bid/# tamoxifen and weight loss
lasix dosage
Misoprostol 200 mg buy online buy cytotec online buy cytotec pills
Misoprostol 200 mg buy online: Misoprostol price in pharmacy – Abortion pills online
http://furosemide.win/# lasix 20 mg
@StephenJonge can i buy ventolin online: ventolin sale uk – ventolin tablets
ventolin medication
http://lipitor.guru/# lipitor 4 mg
Misoprostol 200 mg buy online https://furosemide.win/# furosemide 100 mg
lasix pills
This website was… how do I say it? Relevant!! Finally I have found something which helped me. Kudos.
lasix for sale: cheap lasix – buy furosemide online
nolvadex for sale: how does tamoxifen work – tamoxifen side effects forum
buy cytotec in usa: purchase cytotec – buy cytotec
Very good post. I will be dealing with some of these issues as well..
tamoxifen and uterine thickening: Purchase Nolvadex Online – tamoxifen and weight loss
п»їcytotec pills online http://lisinopril.guru/# how to buy lisinopril
lasix 100mg
Hi there! I could have sworn I’ve been to this site before but after looking at many of the posts I realized it’s new to me. Anyhow, I’m definitely happy I came across it and I’ll be bookmarking it and checking back regularly.
https://lipitor.guru/# lipitor drug prices
buy cytotec in usa https://lipitor.guru/# generic lipitor for sale
buy lasix online
May I simply say what a comfort to discover somebody who actually understands what they’re discussing online. You certainly understand how to bring an issue to light and make it important. More people must check this out and understand this side of the story. I was surprised you aren’t more popular since you definitely possess the gift.
The minimum tick is 10 cents.
While you cannot predict the future, you can make sure that you have an understanding of the risks involved in your investment, by employing due diligence, and researching the property market thoroughly.
The tires also contain technology called raindrop grooves — sipes situated deep inside the tire that are designed to channel water on the road surface to the left and to the right, away from the bottom of the tire.
Its geographical area covers most of the urban development agency the Hyderabad Metropolitan Development Authority (HMDA).
Very good write-up. I absolutely appreciate this website. Keep it up!
Fees can reduce an investor’s profits and potentially reduce the total return below what could be achieved through a less expensive mutual fund or exchange-traded fund (ETF).
http://lipitor.guru/# lipitor prescription cost
http://furosemide.win/# lasix uses
lisinopril 5mg tablets buy lisinopril lisinopril 80 mg daily
lisinopril 20mg india cheap lisinopril lisinopril 49 mg
order cytotec online https://tamoxifen.bid/# tamoxifen hip pain
furosemida
buy cytotec online https://furosemide.win/# lasix 40mg
lasix 100mg
https://tamoxifen.bid/# tamoxifen 20 mg
buy cytotec pills online cheap https://tamoxifen.bid/# cost of tamoxifen
lasix generic
buy lasix online buy furosemide lasix 40mg
https://cytotec.pro/# п»їcytotec pills online
Great article. I’m going through many of these issues as well..
Cytotec 200mcg price https://tamoxifen.bid/# tamoxifen
furosemida 40 mg
http://lipitor.guru/# price canada lipitor 20mg
https://furosemide.win/# furosemida
lisinopril 20 mg price: Buy Lisinopril 20 mg online – lisinopril 1.25 mg
The very next time I read a blog, Hopefully it won’t fail me as much as this particular one. After all, I know it was my choice to read, but I genuinely thought you would probably have something helpful to say. All I hear is a bunch of whining about something that you could fix if you weren’t too busy seeking attention.
order cytotec online http://tamoxifen.bid/# tamoxifen adverse effects
lasix medication
cytotec online http://lisinopril.guru/# lisinopril 5 mg daily
lasix