C++:x&(x-1)的作用、sizeof计算多种情况详解、宏误区

一、x &(x-1)作用

x &(x-1)作用:将二进制的最后一个1变为0.
通过下图观察,我们可以发现规律:在这里插入图片描述

实例1: 统计1的个数

1.求下面函数的返回值

int func(int x) 
{ 
    int countx = 0; 
    while(x) 
    { 
          countx ++; 
          x = x&(x-1); //判断二进制有几个1
     } 
    return countx; 
} 
int main()
{
	int rt = func(9999);
	cout << rt << endl;
	return 0;
}

功能:将x转化为2进制,看含有的1的个数。
         每执行一次x = x&(x-1),会将x用二进制表示时最右边的一个1变为0,因为x-1将会将该位(x用二进制表示时最右边的一个1)变为0。
         因此打印结果:8

实例2:判断一个数(x)是否是2的n次方

判断一个数(x)是否是2的n次方

#include <stdio.h>
int func(int x)
{
    if( (x&(x-1)) == 0 )
        return 1;
    else
        return 0;
}
 
int main()
{
    int x = 8;
    printf("%d\n", func(x));
}

功能:判断一个数(x)是否是2的n次方
如果一个数是2的n次方,那么这个数用二进制表示时其最高位为1,其余位为0。返回值类型为size_t
 

二、sizeof计算大小问题

sizeof:判断数据类型长度符的关键字。
作用:sizeof本质是C/C++中的一个操作符(operator),简单的说其作用就是返回一个对象或者类型所占的内存字节数。
返回值:unsigend int
为什么返回值为unsigend int类型?
当一个unsigned int和int类型运算时,编译器会将运算类型都转为unsigned int类型。如图:

//当一个unsigned int和int类型运算时,编译器会将运算类型都转为unsigned int类型
void test()
{
	unsigned int a = 10;
	if (a - 20 > 0)
	{
		printf("大于0\n");
	}
	else
	{
		printf("小于0\n");
	}
}

输出为:大于0,因此:当sizeof(int)-5时

int main()
{
	if(sizeof(int) - 5 > 0)
	{
		printf("大于0\n值为%u\n",sizeof(int)-5);
	}
	else
	{
		printf("小于0\n");
	}
	return 0;
}

在这里插入图片描述
验证了其返回值大于0;为unsigend int类型。

实例1:sizeof计算基本类型的长度
类型大小 (32位)大小(64位)
char11
unsigned char11
short22
unsigned char22
int44
unsigned int44
long48
unsigned long48
long long88
指针48
float44
double88

 

实例2:sizeof计算结构体、联合体、位段长度

      正确计算sizeof(结构体)的大小,需要理解和掌握数据对齐的概念。数据对齐的概念在前面内存中的数据对齐 和 自然对齐和强制对齐有介绍。关键是记住自然对齐和强制对齐的对齐规则,计算sizeof(结构体)的大小就很简单了。
参见:自定义数据类型
 

实例3:sizeof计算数组长度

普通情况: str是字符数组,由字符串"http://www.ibegroup.com/"初始化,
“http://www.ibegroup.com/”,共24个字符,又字符串以 ‘\0’ 结尾,所以str数组总长度是25。

char str[] = “http://www.ibegroup.com/

   sizeof(s)为25个字节。

str数组做函数参数:
  在函数里,数组str作为参数传进来,在函数内部str是指针的形式。所以:例如:实际上计算的为指针大小,这里为32位平台,4个字节

void Foo ( char str[100])
{
  sizeof( str ) = ?//4个字节,数组作为参数,数组名退化为指针
}

 

实例4:sizeof计算与类相关的大小

4.1sizeof计算空类时,空类大小默认为1。
      因为类的实例化就是给每个实例在内存中分配一块内存地址。空类被实例化时,会由编译器隐含得添加一个字节,所以为1。

class A
{
public:
    A();
    ~A();
private:
};

4.2sizeof计算类时内存对齐问题,static不影响类的大小。
      sizeof(b) = 8。类B中最大类型是int。a长度1字节,b长度1字节,int型c内存地址必须可以被4整除,static变量d不影响类的大小。

class B
{
public:
	
protected:
    char a;
    char b;
    int c;
    static int d;
};

4.3继承与sizeof计算问题:要包含子类的大小
      sizeof(B) = 12。B继承A,所以B中包含了A中的所有数据,大小为A+B大小= 12 。

class A
{
protected:
    int a;
private:
    int b;
};
class B :public A
{
private:
    int c;
};

4.4单个虚函数与sizeof计算问题:要包含指向虚函数表的指针
       计算大小为4,因为类C中包含一个指向虚函数表的指针。

class C
{
    virtual void fun(){}
};

4.5多个虚函数与sizeof计算问题:多个虚函数只需要一个指向虚函数表的指针
计算大小还是为4。

class D
{
    virtual void fun1()    {}

    virtual void fun2() {}
};

 

实例5:sizeof与strlen对比

      strlen:是求字符串实际长度的函数,strlen所作的仅仅是一个计数器的工作,它从内存的某个位置(可以是字符串开头,中间某个位置,甚至是某个不确定的内存区域)开始扫描,直到碰到第一个字符串结束符’\0’为止,然后返回计数器值(长度不包含’\0’,sizeof包含‘\0’)。

char *p1 = "12345678"; 
char p2[] = "12345678"; 
char p3[1024]="12345678"; 
char p4[] = {'1','2','3','4','5','6','7','8'}; 

int main()
{
	cout<<"sizeof(p1):"<<sizeof(p1)<<endl;
	cout<<"sizeof(p2):"<<sizeof(p2)<<endl;
	cout<<"sizeof(p3):"<<sizeof(p3)<<endl;
	cout<<"sizeof(p4):"<<sizeof(p4)<<endl;

	cout<<"strlen(p1)"<<strlen(p1)<<endl;
	cout<<"strlen(p2)"<<strlen(p2)<<endl;
	cout<<"strlen(p3)"<<strlen(p3)<<endl; 
	cout<<"strlen(p4)"<<strlen(p4)<<endl; 

	return 0;
}

计算结果如图所示:
在这里插入图片描述
p1: 为一个字符指针,指向了静态常量区的一个常量字符串,“12345678”。所以,sizeof(p1)=指针的长度=4;strlen(p1)=字符串"12345678"的长度(不含’\0’),所以是8。
p2: 为一个字符数组,由静态常量区的"12345678"进行初始化。所以sizeof(p2)是计算数组p2的长度,所以结果为"12345678"的所有字符的长度(含’\0’),所以是9。 strlen(p2)也是在计算"12345678"的长度(不含’\0’),所以是8。
p3: 为一个字符数组,由静态常量区的"12345678"进行初始化,但长度为1024。 所以,sizeof(p3)=1024;strlen(p3)为字符串"12345678"的长度(不含’\0’),所以是8。
p4: 为一个字符数组,由字符’1’,‘2’,‘3’,‘4’,‘5’,‘6’,‘7’,‘8’进行初始化,,这样初始化与"12345678"进行初始化的区别是 前者不包含’\0’.因此,sizeof(p4)=8。由于p4作为一个字符数组,并不以’\0’结尾, 所以strlen(p4)在计算字符串长度的时候,找不到结束符’\0’,会发生字符串溢出。 字符串溢出,值不确定,甚至会引起程序崩溃
 

三、宏误区

宏: 是一种批量处理的称谓。计算机科学里的宏是一种抽象,它根据一系列预定义的规则替换一定的文本模式。解释器或编译器在遇到宏时会自动进行这一模式替换。对于编译语言,宏展开在编译时发生,进行宏展开的工具常被称为宏展开器。
实例1:

#define DOUBLE(x) x+x

int main()
{
	int rt = 5*DOUBLE(10);
	cout<< rt << endl;
	return 0;
}

这段代码输出结果为:60

为什么?正如定义所描述的,宏展开在编译时发生,会展开替换为实际的语句。当编译到int rt = 5* DOUBLE(10);时,会自动展开替换为:5*10+10=60
但是如果我们将上述代码改为:

#define DOUBLE(x) x+x//宏在编译时要展开

int main()
{
	int rt = 5*(DOUBLE(10));
	cout<<rt<<endl;
	return 0;
}

此时:我们在int rt = 5*(DOUBLE(10));加了一个括号,输出结果为:100
这就是宏替换时我们没有注意到的小细节,以后还是要认真审题呀!

  • 4
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值