C++ 类的静态成员及静态成员函数总结及C语言中的__FILE__、__LINE__和__func__等预定义跟踪调试小记录

本文主要是介绍C++ 类的静态成员及静态成员函数总结及C语言中的__FILE__、__LINE__和__func__等预定义跟踪调试小记录,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  •   对象与对象之间的成员变量是相互独立的。要想共用数据,则需要使用静态成员和静态方法。
  •   只要在类中声明静态成员变量,即使不定义对象,也可以为静态成员变量分配空间,进而可以使用静态成员变量。(因为静态成员变量在对象创建之前就已经被分配了内存空间)
  •   静态成员变量虽然在类中,但它并不是随对象的建立而分配空间的,也不是随对象的撤销而释放(一般的成员在对象建立时会分配空间,在对象撤销时会释放)。静态成员变量是在程序编译时分配空间,而在程序结束时释放空间。
  •   静态成员的定义和声明要加个关键static。静态成员可以通过双冒号来使用,即<类名>::<静态成员名>。
  •   初始化静态成员变量要在类的外面进行。初始化的格式如下:数据类型  类名::静态成员变量名 = 初值;
  •   不能用参数初始化表,对静态成员变量进行初始化。
  •   既可以通过类名来对静态成员变量进行引用,也可以通过对象名来对静态成员变量进行引用。
  •   普通成员函数和静态成员函数的区别是:普通成员函数在参数传递时编译器会隐藏地传递一个this指针.通过this指针来确定调用类产生的哪个对象;但是静态成员函数没有this指针,不知道应该访问哪个对象中的数据,所以在程序中不可以用静态成员函数访问类中的普通变量.

下面通过几个例子来总结静态成员变量和静态成员函数的使用规则。

  一、通过类名调用静态成员函数和非静态成员函数

//例子一:通过类名调用静态成员函数和非静态成员函数
class Point{
public:void init(){}static void output(){}
};void main()
{Point::init();Point::output();
}

 编译出错:错误 1 error C2352: “Point::init”: 非静态成员函数的非法调用

  结论一:不能通过类名来调用类的非静态成员函数

  二、通过类的对象调用静态成员函数和非静态成员函数

//例子二:通过类的对象调用静态成员函数和非静态成员函数
class Point{
public:void init(){}static void output(){}
};void main()
{Point pt;pt.init();pt.output();
}

  编译通过。

  结论二:类的对象可以使用静态成员函数和非静态成员函数。

  三、在类的静态成员函数中使用类的非静态成员

//例子三:在类的静态成员函数中使用类的非静态成员
#include <iostream>
using namespace std;class Point{
public:void init(){}static void output(){cout << "m_x=" << m_x << endl;}
private:int m_x;
};
void main()
{Point pt;pt.output();
}

编译出错:IntelliSense: 非静态成员引用必须与特定对象相对

  因为静态成员函数属于整个类,在类实例化对象之前就已经分配空间了,而类的非静态成员必须在类实例化对象后才有内存空间,所以这个调用就会出错,就好比没有声明一个变量却提前使用它一样。

  结论三:静态成员函数中不能引用非静态成员。

  四、在类的非静态成员函数中使用类的静态成员

//例子四:在类的非静态成员函数中使用类的静态成员
#include <iostream>
using namespace std;class Point{
public:void init(){output();}static void output(){}
private:int m_x;
};
void main()
{Point pt;pt.init();
}

 编译通过。

  结论四:类的非静态成员可以调用静态成员函数,但反之不能。

  五、使用类的静态成员变量

//例子五:使用类的静态成员变量
#include <iostream>
using namespace std;class Point{
public:Point(){m_nPointCount++;}~Point(){m_nPointCount++;}static void output(){cout << "m_nPointCount=" << m_nPointCount << endl;}
private:static  int m_nPointCount;
};void main()
{Point pt;pt.output();
}

链接出错:error LNK2001: 无法解析的外部符号 "private: static int Point::m_nPointCount" (?m_nPointCount@Point@@0HA)

  这是因为类的成员变量在使用前必须先初始化。

  改成如下代码即可:

#include <iostream>
using namespace std;class Point{
public:Point(){m_nPointCount++;}~Point(){m_nPointCount++;}static void output(){cout << "m_nPointCount=" << m_nPointCount << endl;}
private:static  int m_nPointCount;
};//类外初始化静态成员变量时,不用带static关键字
int Point::m_nPointCount = 0;
void main()
{Point pt;pt.output();
}

 运行结果:

  

  结论五:类的静态成员变量必须先初始化再使用。

 

 

标准C语言预处理要求定义某些对象宏,每个预定义宏的名称一两个下划线字符开头和结尾,这些预定义宏不能被取消定义(#undef)或由编程人员重新定义。下面预定义宏表,被我抄了下来。

  • __LINE__  :当前程序行的行号,表示为十进制整型常量
  • __FILE__  :当前源文件名,表示字符串型常量
  • __DATE__ :转换的日历日期,表示为Mmm dd yyyy 形式的字符串常量,Mmm是由asctime产生的。
  • __TIME__  :转换的时间,表示"hh:mm:ss"形式的字符串型常量,是有asctime产生的。(asctime貌似是指的一个函数)
  • __STDC__ :编辑器为ISO兼容实现时位十进制整型常量
  • __STDC_VERSION__  :如何实现复合C89整部1,则这个宏的值为19940SL;如果实现符合C99,则这个宏的值为199901L;否则数值是未定义
  • __STDC_EOBTED__ :(C99)实现为宿主实现时为1,实现为独立实现为0
  • __STDC_IEC_559__ :(C99)浮点数实现复合IBC 60559标准时定义为1,否者数值是未定义
  • __STDC_IEC_559_COMPLEX__: (C99)复数运算实现复合IBC 60559标准时定义为1,否者数值是未定义
  • __STDC_ISO_10646__ :(C99)定义为长整型常量,yyyymmL表示wchar_t值复合ISO 10646标准及其指定年月的修订补充,否则数值未定义

C++中还定义了 __cplusplus

C语言中的__FILE____LINE____DATE__等都在头文件#include<stdio.h>中

其他说明:

如果编译器不是标准的,则可能仅支持以上宏名中的几个,或根本不支持。记住编译程序也许还提供其它预定义的宏名。
__LINE__ 及 __FILE__ 宏指示,#line指令可以改变它的值,简单的讲,编译时,它们包含程序的当前行数和文件名。
__STDC__ 宏指令的意义是编译时定义的。一般来讲,如果__STDC__已经定义,编译器将仅接受不包含任何非标准扩展的标准C/C++代码。如果实现是标准的,则宏__STDC__含有十进制常量1。如果它含有任何其它数,则实现是非标准的。
__cplusplus 与标准c++一致的编译器把它定义为一个包含至少6为的数值。与标准c++不一致的编译器将使用具有5位或更少的数值。


常用宏的具体例子如下:

C语言中的__LINE__用以指示本行语句在源文件中的位置信息,举例如下:

#include <stdio.h>void main(void)
{printf("%d\n",__LINE__);printf("%d\n",__LINE__);printf("%d\n",__LINE__);
}
该程序在linux用gcc编译,在windows的VS2013下编译都可以通过,执行结果都为:

5
6
7

 

还可以通过语句#line来重新设定__LINE__的值,举例如下:

#include <stdio.h>/* 指定下一行的__LINE__为200,重点是line的下一行是200 */
#line 200int main(void)
{printf("%d\n",__LINE__);printf("%d\n",__LINE__);printf("%d\n",__LINE__);return 0;
}
编译执行后输出结果为:

203

204

205

 

C语言中的__FILE__用以指示本行语句所在源文件的文件。

例如,test.c文件内容如下:

#include <stdio.h>int main()
{printf("%s\n",__FILE__);
}

在gcc编译生成a.out,执行后输出结果为:

test.c

 

 

C语言中__DATE__和__TIME__表示时间和日期

#include<stdio.h>void main(void)
{printf("%s\n",__DATE__);printf("%s\n",__TIME__);
}
结果:

Aug 26 2020
23:18:22

 

__STDC__是预定义宏。当它被定义后,编译器将按照ansic标准来编译你的c程序。

__cplusplus用来定义是否是C++编译器

#include <stdio.h>int main(void)
{
#ifdef _cplusplusprintf("C++\n");
#endif#ifdef __STDC__printf("C\n");
#endifreturn 0;
}
输出结果为:

C

 

但是如果在前面定义_cplusplus

#include <stdio.h>#define _cplusplusint main(void)
{
#ifdef _cplusplusprintf("C++\n");
#endif#ifdef __STDC__printf("C\n");
#endifreturn 0;
}

 

那么输出就是:

C++


 

另外 gcc还支持__func__,和__FUNCTION__,它指示所在的函数,但是这个关键字不被windows下的vc6.0支持,举例如下

#include <stdio.h>
void main(void)
{printf("%s\n",__FUNCTION__);printf("%s\n",__func__);
}

其编译后输出结果为

main

main

 

注意: “#line”、 “__LINE__”、 “__FILE__" 及 “__func__" 都是大小写敏感的。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

这篇关于C++ 类的静态成员及静态成员函数总结及C语言中的__FILE__、__LINE__和__func__等预定义跟踪调试小记录的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/975716

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl