4.4数据语义学-数据成员存取

news/2024/7/20 14:58:39 标签: c++, 内存管理

本文C++类的static的说明,参考C++类中的static

类内static变量

static变量不像普通的变量,static变量独立于一切类对象处在。static修饰的变量先于对象存在,所以static修饰的变量要在类外初始化。因为static是所有对象共享的东西嘛,必须要比对象先存在的。
静态成员是可以独立访问的,也就是说,无须创建任何对象实例就可以访问;静态成员函数可不建立对象就可以被使用。或者说静态函数与一般函数没有太大的区别,只是访问有限制,静态变量跟一般的全局变量的区别就是访问有限制。
好处:用static修饰的成员变量在对象中是不占内存的,因为他不是跟对象一起在堆或者栈中生成,用static修饰的变量在静态存储区生成的,所以用static修饰一方面的好处是可以节省对象的内存空间。所以一般类const变量一般改为static const变量,可以节省一些空间。

class test
{
private:
public:
    static int i;
};
int test::i = 100;//此句包含了声明和赋值,初始化不受private和protected访问限制,但是若是priivate,下面main函数就无法访问
int main()
{
    cout << test::i << endl;
    return 0;
}

static成员函数

class test
{
public:
    static int i;//此处只是声明变量
    static void f();//声明
};
int test::i;//此处是定义,少了此处会链接错误
void test::f()//定义
{
    i = 1000;
    cout << i << endl;
}
int main()
{
    //test a;
    test::f();
    return 0;
}

static函数也是类函数,所以在写定义时也要写明属于哪个类。与不同类函数不同的是,它没有传入this指针,正因为没有this指针,所以static类成员函数不能访问非static的类成员,只能访问 static修饰的类成员。静态成员函数不可以同时声明为 virtual、const、volatile函数。

静态成员变量的存取

#include <iostream>
#include <time.h >
#include <stdio.h>
using namespace std;

class MYACLS
{
public:
	int m_i;
	int m_j;
	static int m_si;//一定要类外初始化
	void myfunc() {
		m_i = 5;
		m_j = 6;
	}
};

int MYACLS::m_si = 10; //定义

int main()
{
	//一:静态成员变量的存取
	//静态成员变量,可以当做一个全局量,但是他只在类的空间内可见;引用时用 类名::静态成员变量名
	//静态成员变量只有一个实体,保存在可执行文件的数据段的;

	MYACLS myobj;
	MYACLS* pmyobj = new MYACLS();

	/*
		访问类内static的3种办法(static不属于对象而是属于类,后面两种是语法的允许)
	*/
	cout << MYACLS::m_si << endl;//10
	cout << myobj.m_si << endl;//10
	cout << pmyobj->m_si << endl;//10

	cout << "类内普通成员变量地址:" << endl;
	printf("myobj.m_i = %p\n", &myobj.m_i);
	printf("pmyobj->m_i = %p\n", &pmyobj->m_i);

	//编译时确定地址,不变的
	cout << "类内静态成员变量地址(3行输出一样):" << endl;
	printf("MYACLS::m_si = %p\n", &MYACLS::m_si);
	printf("myobj.m_si = %p\n", &myobj.m_si);
	printf("pmyobj->m_si = %p\n", &pmyobj->m_si);
	
	return 1;
}


结果

在这里插入图片描述

非静态成员变量的存取

#include <iostream>
using namespace std;

class FAC
{
public:
	int m_fai;
	int m_faj;
};
//继承
class MYACLS : public FAC
{
public:
	int m_i;
	int m_j;
	static int m_si ;
	void myfunc() {
		m_i = 5;
		m_j = 6;
	}
};
int MYACLS::m_si = 10; 

int main()
{
	MYACLS myobj;
	MYACLS* pmyobj = new MYACLS();

	//二:非静态成员变量的存取(普通的成员变量),存放在类的对象中。存取通过类对象(类对象指针)

	//普通成员偏移值;父类继承过来的在前面
	printf("MYACLS::m_fai = %p\n", &MYACLS::m_fai);//00000000
	printf("MYACLS::m_faj = %p\n", &MYACLS::m_faj);//00000004
	printf("MYACLS::m_i = %p\n", &MYACLS::m_i);//00000008
	printf("MYACLS::m_j = %p\n", &MYACLS::m_j);// 0000000C
	
	pmyobj->myfunc();
	//编译器角度:会加上this指针,并且地址会偏移08和0C
	//MYACLS::myfunc(MYACLS *const this)
	//{
		//this->m_i = 5;
		//this->m_j = 5;
	//}

	pmyobj->m_faj = 7;//偏移4字节

	return 1;
}

运行pmyobj->myfunc()时,通过反汇编,能更清楚的看到this指针位置偏移08和0C
在这里插入图片描述
pmyobj->m_faj = 7时,this偏移4字节
在这里插入图片描述


http://www.niftyadmin.cn/n/1530964.html

相关文章

Linux系统编程之信号

本篇博客所回顾的知识&#xff08;学习目标&#xff09;&#xff1a; 一、信号介绍 从图中可以看出&#xff0c;信号的优先级是高于普通操作的&#xff01;出现信号就一定要先执行完才能继续做之前的事情&#xff01; 注意①&#xff1a;如何查看信号都有哪些呢&#xff1f; 答…

插入排序的3种实现方式(包含希尔排序)

插入排序的基本方法是&#xff1a;每一步将一个待排序的元素&#xff0c;按其排序码的大小&#xff0c;插入到前面已经排好序的一组元素的适当位置上去&#xff0c;直到元素全部插入为止。 顺序法定位插入位置——直接插入排序 i表示当前要排序的元素&#xff0c;j是i的前面一…

冒泡排序和快速排序优化方法

冒泡排序 参考【排序】&#xff1a;冒泡排序以及三种优化 //假设排序arr[] { 1, 3, 4, 2, 6, 7, 8, 0 }; void BubbleSort(int arr[],int len) {int i 0;int tmp 0;for (i 0; i < len - 1; i)//确定排序趟数{int j 0;for (j 0; j < len - 1 - i; j)//确定比较次…

leetcode回溯专题

22.括号生成 class Solution { public:vector<string> res;vector<string> generateParenthesis(int n){if (n 0)return {};string track;//可用的左括号和右括号数量backtrack(n, n, track, res);return res;}void backtrack(int left, int right, string &t…

leetcode字符串专题

151.翻转字符串里的单词 class Solution { public:string reverseWords(string s){reverse(s.begin(), s.end());int size (int)s.size();int start 0; //每个单词开始int end 0; //每个单词末尾int index 0; //要返回的字符串下标for (; start < size; start){if (s[…

两种方法实现拓扑排序

本文转载自&#xff1a;https://blog.csdn.net/qq_35644234/article/details/60578189 1.拓扑排序的介绍 对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序&#xff0c;是将G中所有顶点排成一个线性序列&#xff0c;使得图中任意一对顶点u和v&#xff0c;若边(…

leetcode.第184场周赛(2020.4.12)

1.数组中字符串匹配 思路 主要是strstr函数&#xff0c;在头文件string.h里 复杂度 时间&#xff1a;O(Kn2)O(Kn^2)O(Kn2),K为字符串长度 空间&#xff1a;O(1)O(1)O(1) class Solution { public:vector<string> stringMatching(vector<string>& words) {…

leetcode数论专题

367.完全平方数 1 1; 4 1 3; 9 1 3 5; 16 1 3 5 7; N*N 1 3 5 … (2N - 1) 数学规律 时间&#xff1a;O(sqrt(num))O(sqrt(num))O(sqrt(num)) 空间&#xff1a;O(1)O(1)O(1) class Solution { public:bool isPerfectSquare(int num) {int num1 1;while(num…