静态数据成员和静态成员函数

末蓝、 2022-08-27 12:59 330阅读 0赞

c++中的static静态数据成员和静态成员函数应该是让大家比较头疼的东西,好像也是找工作公司面试中常常问到的东西。我自己也深有体会,在学习c++的过程中,总感觉static很烦人,但是又是一个必须懂的东西,所以今天就对静态数据成员和静态成员函数坐下小结哈!

一、静态数据成员

1.静态数据成员怎么去定义?

在类中声明静态数据成员很简单,是以static关键字表明即可,如下所示

  1. class Test{
  2. private:
  3. //静态数据成员
  4. static int a;
  5. };

a就是静态数据成员了,在类中只能声明可是不能定义哈!

要对静态数据成员定义和初始化必须在类的外面也就是在全局作用域中定义,如果定义不给出初值,则默认初值为0

  1. class Test{
  2. public:
  3. int GetA() const{return a;}
  4. private:
  5. //静态数据成员
  6. static int a;
  7. };
  8. //int Test::a;如果这样定义不赋予初值则初值为零
  9. int Test::a = 1;
  10. #include <iostream>
  11. int main()
  12. {
  13. Test T;
  14. std::cout << T.GetA() << std::endl;
  15. }

定义时一定要在全局作用域中定义,一定不要在类中定义!

静态数据成员甚至在类没有任何对象的时候都可以访问,静态成员可以独立访问,无需依赖任何对象的建立

另外,不要试图在头文件中定义(初始化)静态数据成员。在大多数的情况下,这样做会引起重复定义这样的错误。即使加上#ifndef #define #endif或者#pragma once也不行。

2.静态数据成员被类的所有对象共享,包括该类的派生类对象,基类对象和派生类对象共享基类的静态数据成员

答:静态数据成员不属于任何对象,类的静态数据成员的存在不依赖于任何类对象的存在,静态数据成员是由类的所有对象共享的。例子如下:

  1. class Base{
  2. public:
  3. //静态数据成员
  4. static int a;
  5. };
  6. class Derived : public Base{
  7. };
  8. //int Test::a;如果这样定义不赋予初值则初值为零
  9. int Base::a;
  10. #include <iostream>
  11. int main()
  12. {
  13. Base B;
  14. Derived D;
  15. B.a++;
  16. std::cout << B.a << std::endl;
  17. D.a++;
  18. std::cout << D.a << std::endl;
  19. }

打印结果如下:

20140607213146281

由打印结果看出来,派生类对象和基类对象都是共享基类的静态数据成员,而基类的所有对象也是共享该静态数据成员,且该静态数据成员应该在全局作用域中定义,可以赋予初值也可以不赋予初值,如果不赋予初值,静态数据成员有其默认值。

3.静态数据成员可以作为成员函数的默认形参,而普通数据成员则不可以

答:不多说,直接看例子马上就明白了哈!

  1. class Test{
  2. public:
  3. //静态数据成员
  4. static int a;
  5. int b;
  6. void fun_1(int i = a);//正确
  7. void fun_2(int i = b);//报错
  8. };

4.静态数据成员的类型可以是所属类的类型,而普通数据成员则不可以。普通数据成员的只能声明为 所属类类型的 指针或引用

答:这个也不多说,直接看例子就可以懂什么意思哈!

  1. class Test{
  2. public:
  3. //静态数据成员
  4. static Test a;//正确
  5. Test b;//报错
  6. Test *pTest;//正确
  7. Test &m_Test;//正确
  8. static Test *pStaticObject;//正确
  9. };

5.静态数据成员在const函数中可以修改,而普通的数据成员是万万不能修改的哈!

答:看个例子

  1. class Test{
  2. public:
  3. Test():b(0){}
  4. //静态数据成员
  5. static int a;//正确
  6. int b;
  7. void test() const
  8. {
  9. a++;
  10. b++;//const指的不能修改当前调用该函数对象的数据成员
  11. }
  12. };
  13. int Test::a;

const修饰的时当前this指针所指向的对象是const,但是静态数据成员不属于任何类的对象,它被类的所有对象修改,所以this指针不修饰静态的数据成员,所以可以更改。

二、静态成员函数

静态成员函数的声明也很简单,就是在类的成员函数前加上static关键字即可,和静态成员一样,静态成员函数也是属于类的,它并不属于任何对象,当调用静态成员函数时应该使用类名和域运算符“∷”,当然也可以使用对象调用操作,但是这样的操作并不意味着静态成员函数属于这个对象,它只是被这个对象共享而已,这样也就决定了静态成员函数中是不能访问本类中的非静态数据成员的,它是不能访问非静态数据成员的,在c++中静态成员函数主要用来访问静态数据成员而不访问非静态数据成员

1.静态成员函数不能调用非静态成员函数,但是反过来是可以的

2.静态成员函数没有this指针,也就是说静态成员函数不能使用修饰符(也就是函数后面的const关键字)

3.静态成员函数的地址可用普通函数指针储存,而普通成员函数地址需要用 类成员函数指针来储存。

总结:其实声明为静态,不论是静态数据成员还是静态成员函数,它们都是不依赖于对象而存在的,类在定义后并不分配存储空间,而是在定义类的对象的时候才分配存储空间,相反静态的数据成员和静态的成员函数是已经在内存中开辟了内存空间了,所以静态数据成员可以独立的访问在任何类对象没有建立起来都可以访问,并且静态成员函数不可以调用非静态成员函数,因为非静态成员函数只有在类对象建立以后才可以调用,相反则是可以的。我觉得当对某个判断产生怀疑的时候应该去测试,只有验证了才知道是不是对的哈!

为了能更好阐释静态数据成员和静态成员函数,然后在网上搜了博客,感觉有些例子不错(因找不到最初的出处,所以无法注明请原作者谅解),所以就拿来给大家参考一下哈!

关于玩篮球的例子很能明显解释静态数据成员和静态成员函数到底是怎么回事

你们班里面有10个人(10个比如高一一班的对象),体育老师分给你们一个篮球(静态成员函数),你们每个人都带了一个篮球(非静态成员数),你们都很小气,自己的球只能自己拍,要是5对5打比赛,那就只能用那个静态的篮球了(每个人都可以拿来用,但是带来的影响是对全体的)。因此,我可以说那个篮球是高一一班的成员。所以也就是说:静态成员函数是类的成员函数(因为高一二班就不能拿来玩),但是这个篮球最后还是要还给老师的,任何私人不得占有。希望这样你能明白,其实在内存空间里面说白了静态的成员的内存是唯一的一份,就是当你在类外声明他时开辟的,但是非静态函数的空间分配是在你实例化对象时创建的。

为了使大家更好的理解这两个概念,还是老样子用代码来说明上面文字说明的这一切哈!

关于学生类的例子

  1. //定义Student类
  2. #include <iostream>
  3. class Student
  4. {
  5. public:
  6. //定义构造函数
  7. Student(int n,int a,float s):num(n),age(a),score(s){ }
  8. void total();
  9. //声明静态成员函数
  10. static float average();
  11. private:
  12. int num;
  13. int age;
  14. float score;
  15. //静态数据成员,累计学生的总分
  16. static float sum;
  17. //静态数据成员,累计学生的人数
  18. static int count;
  19. };
  20. //在全局作用域对静态数据成员初始化,如果不赋予初值,则使用其默认值零
  21. float Student::sum;
  22. int Student::count;
  23. //定义非静态成员函数
  24. void Student::total()
  25. {
  26. //累加总分
  27. sum+=score;
  28. //累计已统计的人数
  29. count++;
  30. }
  31. //定义静态成员函数
  32. float Student::average()
  33. {
  34. return(sum/count);
  35. }
  36. int main()
  37. {
  38. Student stud[3]={
  39. //定义对象数组并初始化
  40. Student(1001,18,70),
  41. Student(1002,19,78),
  42. Student(1005,20,98)
  43. };
  44. int n;
  45. std::cout<<"please input the number of students: ";
  46. //输入需要求前面多少名学生的平均成绩
  47. std::cin>>n;
  48. //调用3次total函数
  49. for(int i=0;i<n;i++)
  50. {
  51. stud[i].total();
  52. }
  53. //调用静态成员函数
  54. std::cout<<"the average score of "<<n<<" students is "<<Student::average( )<<std::endl;
  55. return 0;
  56. }

打印输出如下:

20140608094355906

对上面的代码做以下说明:

首先,在主函数中定义了对象数组,存放每个学生的编号、年龄和成绩,其中sum和count是要累计所有学生的总成绩和总的学生人数,我们定义成了静态数据成员,在学生类的成员函数中,我们定义了普通的total成员函数,用来计算所有学生的总成绩和总的学生人数,另外,定义了静态成员函数average,学生类的设计大概如此

在全局作用域对类中静态数据成员进行了定义,但未赋予初值,这意味着我们采用其默认值。

在类的普通成员函数toal中可以引用静态数据成员sum和count,可见类的所有成员函数都可以引用类的静态数据成员。

在类的静态成员函数average中,只能引用类的静态数据成员,不能引用非静态数据成员。

在主函数中我们调用类的非静态成员函数时只能通过类对象来调用,如stu[i].total,但是对于静态成员函数可以直接通过类名+作用域符号来调用,如

Student::average。

发表评论

表情:
评论列表 (有 0 条评论,330人围观)

还没有评论,来说两句吧...

相关阅读

    相关 静态数据成员静态成员函数

    类中的静态成员真是个让人爱恨交加的特性。我决定好好总结一下静态类成员的知识点,以便自己在以后面试中,在此类问题上不再被动。 静态类成员包括静态数据成员和静态函数成员两部分。