1、函数调用过程中的类型转换
书上例子(183):
int fact(int val)
{
int ret=1;
while(val>1)
ret*=val--;
return ret;
}
书上描述为:fact函数只有一个int类型的实参,所以我们每次调用它时,都必须提供一个能转换成int的实参。所以这里是可以给fact传一个float类型的数的,但在传给形参时会隐式转换为int型,c++对隐式转换有较为详细的描述,主要原则为:算数类型之间的隐式转换被设计的尽可能不损失精度。
例如:
int ival=3.541+3;
这里,3.541是double型,3是int型,两个不同类型的值无法相加,所以3首先被转换为double型,再和3.541相加,得到一个double值,但这个值再完成初始化ival的任务时,值被转换成了int型。
2、传引用参数
书上主要提到两点:
1、使用引用形参避免拷贝
2、使用引用形参返回额外信息
书中给出了一个例子:返回字符串中指定的字符第一次出现的位置和这个字符出现的个数,其实这个问题大可以用更好的类来解决,但这里主要为了说明引用形参绑定了实参这个重要特性。此外,这段代码处理第一次的这种方法有一定借鉴的价值
string::size_type find_char(const string&s, char c, string::size_type&occurs)
{
auto ret = s.size();
occurs = 0;
for (decltype(ret) i = 0; i != s.size(); i++) {
if (s[i] == c) {
//这里为了返回c第一次出现的位置,必须进行此判断,因为ret的初始状态是s.size()
if (ret == s.size())
ret = i;
occurs++;
}
}
return ret;
}
3、对象的生命周期
书上主要讲了两种局部对象:自动对象和局部静态对象
自动对象:只存在于块执行期间,当 块 执行结束后后即销毁
局部静态对象:在程序执行路径第一次经过它时执行初始化(没有显示初始化的话,会执行值初始化),
之后对象所在函数执行结束后对它不会产生影响,直到 整个程序终止 才会被销毁。
4、含有可变形参的函数
c++11新标准可以定义可变数量的形参,但形参的类型必须相同,使用l标准库类型initializer_list,值得注意的是,如果向initializer_list中传递一个值得序列,必须把序列放在一对花括号中。
下面是自己编写的一个函数,功能是计算列表中所有元素的和
using namespace std;
int sum(initializer_list<int> i1)
{
int result=0;
for (auto beg = i1.begin(); beg != i1.end(); beg++) {
result += *beg;
}
return result;
}
int main()
{
cout << sum({ 1,2,3,4,5,6,7,8,9,10 });//值的序列放在一对花括号中
cout << endl;
system("pause");
return 0;
}
5、返回数组指针
int arr[10] //arr是一个含有10个整数的数组
int (*p1)[10] // p1是一个指针,指向含有10个整数的数组
int*p2[10] //p2是一个数组,数组中包含10个指向整数的指针
6、函数指针
bool (*pf) (const string&,const string &)//pf是一个指针,指向一个返回bool值且形参时const string的引用的函数
注意:在给pf赋值是,可以直接赋值函数名,但函数的类型必须与定义的函数指针类型严格匹配(返回类型、参数)