郴州网站建设维护,杭州百度公司在哪里,西安h5建站,吉林沈阳网站建设目录 1.构造函数深入
2.类型转换
3.static成员
4.友元
5.内部类
6.匿名对象
7.对象拷⻉时的编译器优化 1.构造函数深入
#xff08;1#xff09;前我们实现构造函数时#xff0c;初始化成员变量主要使⽤函数体内赋值#xff0c;构造函数初始化还有⼀种⽅式#x…目录 1.构造函数深入
2.类型转换
3.static成员
4.友元
5.内部类
6.匿名对象
7.对象拷⻉时的编译器优化 1.构造函数深入
1前我们实现构造函数时初始化成员变量主要使⽤函数体内赋值构造函数初始化还有⼀种⽅式就是初始化列表初始化列表的使⽤⽅式是以⼀个冒号开始接着是⼀个以逗号分隔的数据成员列表每个成员变量后⾯跟⼀个放在括号中的初始值或表达式。
#define _CRT_SECURE_NO_WARNINGS 1
#includeiostream
using namespace std;
class Data
{Data(int year1900, int month1, int day1):_year(year),_month(month),_day(day){}
private:int _year;int _month;int _day;
};
int main()
{return 0;
}
2每个成员变量在初始化列表中只能出现⼀次语法理解上初始化列表可以认为是每个成员变量定义初始化的地⽅。
3引⽤成员变量const成员变量没有默认构造的类类型变量必须放在初始化列表位置进⾏初始化否则会编译报错。
4C11⽀持在成员变量声明的位置给缺省值这个缺省值主要是给没有显⽰在初始化列表初始化的成员使⽤的。
5尽量使⽤初始化列表初始化因为那些你不在初始化列表初始化的成员也会⾛初始化列表如果这个成员在声明位置给了缺省值初始化列表会⽤这个缺省值初始化。如果你没有给缺省值对于没有显⽰在初始化列表初始化的内置类型成员是否初始化取决于编译器C并没有规定。对于没有显⽰在初始化列表初始化的⾃定义类型成员会调⽤这个成员类型的默认构造函数如果没有默认构造会编译错误。
6初始化列表中按照成员变量在类中声明顺序进⾏初始化跟成员在初始化列表出现的的先后顺序⽆关。建议声明顺序和初始化列表顺序保持⼀致。
总结
每个成员都要走初始化列表
1.在初始化列表初始化的成员
2.没有在初始化列表的成员 1声明的地方有缺省值用缺省值 2没有缺省值 a.内置类型不确定看编译器大概率是随机值。 b.自定义类型调用默认构造没有默认构造就编译报错。
3.引用 const 没有默认构造自定义必须在初始化列表初始化。
2.类型转换
1C⽀持内置类型隐式类型转换为类类型对象需要有相关内置类型为参数的构造函数
2构造函数前⾯加explicit就不再⽀持隐式类型转换
#includeiostream
using namespace std;
class A
{
public:
// 构造函数explicit就不再⽀持隐式类型转换
// explicit A(int a1)
A(int a1)
:_a1(a1)
{}
//explicit A(int a1, int a2)
A(int a1, int a2)
:_a1(a1)
,_a2(a2)
{}
void Print()
{
cout _a1 _a2 endl;
}
private:
int _a1 1;
int _a2 2;
};
int main()
{
// 1构造⼀个A的临时对象再⽤这个临时对象拷⻉构造aa3
// 编译器遇到连续构造拷⻉构造-优化为直接构造
A aa1 1;
aa1.Print();
const A aa2 1;
// C11之后才⽀持多参数转化
A aa3 { 2,2 };
return 0;
}
3.static成员
1⽤static修饰的成员变量称之为静态成员变量静态成员变量⼀定要在类外进⾏初始化。
2静态成员变量为所有类对象所共享不属于某个具体的对象不存在对象中存放在静态区。 3⽤static修饰的成员函数称之为静态成员函数静态成员函数没有this指针。
4⾮静态的成员函数可以访问任意的静态成员变量和静态成员函数。
5突破类域就可以访问静态成员可以通过类名::静态成员 或者 对象.静态成员 来访问静态成员变量和静态成员函数。
6突破类域就可以访问静态成员可以通过类名::静态成员 或者 对象.静态成员 来访问静态成员变量和静态成员函数。
7静态成员变量不能在声明位置给缺省值初始化因为缺省值是个构造函数初始化列表的静态成员变量不属于某个对象不⾛构造函数初始化列表。
4.友元
1友元提供了⼀种突破类访问限定符封装的⽅式友元分为友元函数和友元类在函数声明或者类声明的前⾯加friend并且把友元声明放到⼀个类的⾥⾯。
2外部友元函数可访问类的私有和保护成员友元函数仅仅是⼀种声明他不是类的成员函数。
3友元函数可以在类定义的任何地⽅声明不受类访问限定符限制。
4⼀个函数可以是多个类的友元函数。
5友元类中的成员函数都可以是另⼀个类的友元函数都可以访问另⼀个类中的私有和保护成员
6友元类中的成员函数都可以是另⼀个类的友元函数都可以访问另⼀个类中的私有和保护成员
7友元类关系不能传递如果A是B的友元 B是C的友元但是A不是C的友元。
8有时提供了便利。但是友元会增加耦合度破坏了封装所以友元不宜多⽤。
#includeiostream
using namespace std;
class A
{
// 友元声明
friend class B;
private:
int _a1 1;
int _a2 2;
};
class B
{
public:
void func1(const A aa)
{
cout aa._a1 endl;
cout _b1 endl;
}
void func2(const A aa)
{
cout aa._a2 endl;
cout _b2 endl;
}
private:
int _b1 3;
int _b2 4;
};
int main()
{
A aa;
B bb;
bb.func1(aa);
bb.func1(aa);
return 0;
}
5.内部类
1如果⼀个类定义在另⼀个类的内部这个内部类就叫做内部类。内部类是⼀个独⽴的类跟定义在全局相⽐他只是受外部类类域限制和访问限定符限制所以外部类定义的对象中不包含内部类。
2内部类默认是外部类的友元类。
3)内部类本质也是⼀种封装当A类跟B类紧密关联A类实现出来主要就是给B类使⽤那么可以考虑把A类设计为B的内部类如果放到private/protected位置那么A类就是B类的专属内部类其他地⽅都⽤不了。
#includeiostream
using namespace std;
class A
{
private:
static int _k;
int _h 1;
public:
class B // B默认就是A的友元
{
public:
void foo(const A a)
{
cout _k endl;
//OK
cout a._h endl;
//OK
}
};
};
int A::_k 1;
int main()
{
cout sizeof(A) endl;
A::B b;
A aa;
b.foo(aa);
return 0;
}
6.匿名对象
1⽤ 类型(实参) 定义出来的对象叫做匿名对象相⽐之前我们定义的 类型 对象名(实参) 定义出来的叫有名对象
2匿名对象⽣命周期只在当前⼀⾏⼀般临时定义⼀个对象当前⽤⼀下即可就可以定义匿名对象。
class A
{
public:
A(int a 0)
:_a(a)
{
cout A(int a) endl;
}
~A()
{
cout ~A() endl;
}
private:
int _a;
};
class Solution {
public:
int Sum_Solution(int n) {
//...
return n;
}
};
int main()
{
A aa1;
// 不能这么定义对象因为编译器⽆法识别下⾯是⼀个函数声明还是对象定义
//A aa1();
// 但是我们可以这么定义匿名对象匿名对象的特点不⽤取名字
// 但是他的⽣命周期只有这⼀⾏我们可以看到下⼀⾏他就会⾃动调⽤析构函数
A();
A(1);
A aa2(2);
// 匿名对象在这样场景下就很好⽤当然还有⼀些其他使⽤场景这个我们以后遇到了再说
Solution().Sum_Solution(10);
return 0;
}
7.对象拷⻉时的编译器优化
1现代编译器会为了尽可能提⾼程序的效率在不影响正确性的情况下会尽可能减少⼀些传参和传参过程中可以省略的拷⻉。
2如何优化C标准并没有严格规定各个编译器会根据情况⾃⾏处理。当前主流的相对新⼀点的编译器对于连续⼀个表达式步骤中的连续拷⻉会进⾏合并优化有些更新更激进的编译还会进⾏跨⾏跨表达式的合并优化。
#includeiostream
using namespace std;
class A
{
public:
A(int a 0)
:_a1(a)
{
cout A(int a) endl;
}
A(const A aa)
:_a1(aa._a1)
{
cout A(const A aa) endl;
}
A operator(const A aa)
{
cout A operator(const A aa) endl;
if (this ! aa)
_a1 aa._a1;
}
return *this;
}
~A()
{
cout ~A() endl;
}
private:
int _a1 1;
};
void f1(A aa)
{}
A f2()
{
A aa;
return aa;
}
int main()
{
// 传值传参
A aa1;
f1(aa1);
cout endl;
// 隐式类型连续构造拷⻉构造-优化为直接构造
f1(1);
// ⼀个表达式中连续构造拷⻉构造-优化为⼀个构造
f1(A(2));
cout endl;
cout *********************************************** endl;
// 传值返回
// 返回时⼀个表达式中连续拷⻉构造拷⻉构造-优化⼀个拷⻉构造 vs2019
// ⼀些编译器会优化得更厉害进⾏跨⾏合并优化直接变为构造。vs2022
f2();
cout endl;
// 返回时⼀个表达式中连续拷⻉构造拷⻉构造-优化⼀个拷⻉构造 vs2019
// ⼀些编译器会优化得更厉害进⾏跨⾏合并优化直接变为构造。vs2022
A aa2 f2();
cout endl;
// ⼀个表达式中连续拷⻉构造赋值重载-⽆法优化
aa1 f2();
cout endl;
return 0;
}