1.sizeof用法总结

①与strlen比较

      strlen  计算字符数组的字符数,以"\0"为结束判断,但不包括。
    sizeof  计算数据(数组、变量、类型、结构体等)所占内存空间,用字节数表示。

②指针与静态数组的sizeof操作

  指针均可看为变量类型的一种。因此:
      :int *p;
                    sizeof(p)=4;
                    sizeof(*p)=4;//相当于sizeof(int)

  对于静态数组,sizeof可直接计算数组大小:
      例:int a[10];char b[]="hello";                    sizeof(a)=40;//4*10=40;                    sizeof(b)=6;
  数组做型参时,数组名称当作指针使用:
             例: void  fun(char p[]);
                     sizeof(p)=4;

③格式的写法

sizeof为操作符而非函数,对变量或对象可以不加括号,但若是类型,须加括号。

④操作string的注意事项

        string   str[]={"hello", "world", "CHB","\n"};
        cout<<sizeof(str)<<endl;//输出128
        cout<<sizeof(str[0])<<endl;//输出32,即对象的大小
        cout<<sizeof(str[0].c_str())<<endl;//输出4,c_str()返回 const char*指针,指向str[0]
        cout<<strlen(str[0].c_str())<<endl;//输出5,str[0]字符串的长度

⑤经典问题:

        double* (*a)[3][6];
        cout<<sizeof(a)<<endl; // 4,a为指针
        cout<<sizeof(*a)<<endl; // 72 ,*a为一个有3*6个指针元素的数组
        cout<<sizeof(**a)<<endl; // 24, **a为行指针
        cout<<sizeof(***a)<<endl; // 4 ,***a为一维的第一个指针
        cout<<sizeof(****a)<<endl; // 8 ,****a为一个double变量

       解析:a为指向double*[3][6]类型二维指针数组(数组元素为double* 指针类型)的指针。既然是指针,所以sizeof(a)就是4。*a就表示二维指针数组double*[3][6],因此sizeof(*a)=3*6*sizeof(double*)=72。**a为行指针,指向一维指针数组double*[6],sizeof(**a)=6*sizeof  (double*)=24。***a就表示行指针数组的第一个指针元素,也就是double*了,所以sizeof(***a)=4。至于****a,则是一个double类型,所以sizeof(****a)=sizeof(double)=8。
     看看以下情况:
 double**b[3][6];
        cout<<sizeof(b)<<endl; // 72,b为数组类型,数组元素为double**;
        cout<<sizeof(*b)<<endl; // 24,行指针
        cout<<sizeof(**b)<<endl; // 4,b[0][0]的值,double**类型
        cout<<sizeof(***b)<<endl; // 4,double*
        cout<<sizeof(****b)<<endl; // 8 double
 double (*c)[3][6];
 cout<<sizeof(c)<<endl;//4,指针变量,指向double[3][6]
 cout<<sizeof(*c)<<endl;//144,double[3][6]的数组类型
 cout<<sizeof(**c)<<endl;//48,行指针,指向c[0],实际指向为&c[0][0]
 cout<<sizeof(***c)<<endl;//8,c[0][0]的值
 double* d[3][6];
        cout<<sizeof(d)<<endl; // 72,指针数组类型,元素为double*的[3][6]数组
        cout<<sizeof(*d)<<endl; // 24,行指针
        cout<<sizeof(**d)<<endl; // 4,d[0][0]的元素值,即double*
        cout<<sizeof(***d)<<endl; // 8

⑥操作struct的内存对齐问题

     (1)整体空间是占用空间最大的成员(的类型)所占字节数的整倍数    (2)内存按结构成员的先后顺序排列,当排到该成员变量时,其前面已摆放的空间大小必须是该成员类型大小的整倍数,如果不够则补齐,以此向后类推。    (3)数组按照单个变量一个一个的摆放,而不是看成整体。如果成员中有自定义的类、结构体,也要注意数组问题。

例子1

struct s1{char a;double b;int c;char d;};struct s2{char a;char b;int c;double d;};cout<<sizeof(s1)<<endl; // 24
cout<<sizeof(s2)<<endl; // 16

占空间的最大成员是double类型变量,故对齐空间大小为8。s1中的c和d,s2中a,b,c可以放进一个“8”位空间中。

例子2

struct s1
{char a[8];};struct s2
{double d;};struct s3{  s1 s;char a;};struct s4{s2 s;char a; };cout<<sizeof(s1)<<endl; // 8
cout<<sizeof(s2)<<endl; // 8
cout<<sizeof(s3)<<endl; // 9
cout<<sizeof(s4)<<endl; // 16

虽然s1和s2大小都是8,但是s1的对齐空间大小是1(char),s2是8(double)。所以定义结构体的时候,如果空间紧张的话,最好考虑对齐因素来排列结构体里的元素。这里结构体中定义的数组可以当做多个同类数据顺序排列,以此确定对齐的空间大小。

例子3

struct s1 { int i: 8; int j: 4; double b; int a:3;}; struct s2 { int i; int j; double b; int a;}; struct s3 { int i; int j; int a; double b;}; struct s4 { int i: 8; int j: 4; int a:3; double b;}; cout<<sizeof(s1)<<endl; // 24
cout<<sizeof(s2)<<endl; // 24
cout<<sizeof(s3)<<endl; // 24
cout<<sizeof(s4)<<endl; // 16 

在结构体和类中,可以使用位域来规定某个成员所能占用的空间,所以使用位域能在一定程度上节省结构体占用的空间。double存在会干涉到位域。所以使用位域的的时候,最好把float类型和double类型放在程序的开始或者最后。

    如上。以double--8字节为单位,分配位,以s1为例。i,j占据一个double长度,b自己占据一个double长度,a占据另外一个double。

⑦基本操作结果

sizeof  int:4
     sizeof  short:2
     sizeof  long:4
     sizeof  float:4
     sizeof  double:8
     sizeof  char:1
     sizeof  *p:4
     sizeof  WORD:2
     sizeof  DWORD:4

2.同类对象之间和结构体变量之间的相互赋值

struct student
{int number;char name[8];float score;
}a[3],temp;
temp=a[0];
a[0]=a[1];
a[1]=temp;

同类型的对象之间或结构体变量之间可以相互赋值。而字符串或者字符数组不行。

  • 对象的赋值只对其中的数据成员赋值,而不对成员函数赋值。数据成员是占存储空间的,不同对象的数据成员占有不同的存储空间,赋值的过程是将一个对象的数据成员在存储空间的状态复制给另一对象的数据成员的存储空间。而不同对象的成员函数是同一个函数代码段,不需要、也无法对它们赋值。
  • 类的数据成员中不能包括动态分配的数据,否则在赋值时可能出现严重后果。

3.setw()和setfill()的使用

   set(int n),在输出时分配了n个字符的输出宽度,然后默认的是在n个字符宽度中右对齐setiosflags(ios::rigth)输出,可以使用setiosflags(ios::left)设置为左对齐输出可以使用。配套使用setfill(char x)使用x来填充空下的空格。VS2008中编译时,必须包括头文件【#include "iomanip"】。
#include "stdafx.h"
#include"iostream"
#include "iomanip"
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{cout<<"方法一:    "<<endl;cout << setw(8) << setiosflags(ios::left) <<setfill('0') << 8 << endl;//左对齐,即从右边填充cout << setw(8) << setiosflags(ios::right) <<setfill('0') << 9 << endl;//右对齐,从左边填充cout << setw(8) << setfill('0') << 10 << endl;//默认右对齐,即从左边填充cout<<"使用以下这种形式来调用显得更加整齐并简洁:    "<<endl;cout.setf(ios::right,ios::adjustfield);//标志位设置cout.fill('0');cout << setw(8) << 11 <<endl;cout << setw(8) << 12 << endl;return 0;
}

结果如下:
分析:1)倘若用于用cout输出的话,被填充的仅仅应该是int值;
      2)第一种方法三个函数setw(8),setiosflags(ios::right),setfill('0')的位置可以任意调换。

4.字符串流stringstream

C++标准库中的“sstream”提供了比ANSIC的<stdio.h>更高级的一些功能,即单纯性、类型安全和可扩展性。为什么要花额外的精力来学习基于“sstream”的类型转换呢?也许对下面一个简单的例子的回顾能够说服你。假设你想用sprintf()函数将一个变量从int类型转换到字符串类型。为了正确地完成这个任务,你必须确保证目标缓冲区有足够大空间以容纳转换完的字符串。此外,还必须使用正确的格式化符。但是,对上面代码的一个微小的改变就会使程序崩溃,如下:

int n=10000;
char s[10];
sprintf(s,”%d”,n);// s中的内容为“10000”
sprintf(s,”%f”,n);// 注意!错误的格式化符

由于错误地使用了%f格式化符来替代了%d。因此,s在调用完sprintf()后包含了一个不确定的字符串。

由于n和s的类型在编译期就确定了,所以编译器拥有足够的信息来判断需要哪些转换。“sstream”库中声明的标准类就利用了这一点,自动选择所必需的转换。而且,转换结果保存在stringstream对象的内部缓冲中。你不必担心缓冲区溢出,因为这些对象会根据需要自动分配存储空间。

“sstream”库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本。

<sstream>使用string对象来代替字符数组。这样可以避免缓冲区溢出的危险。而且,传入参数和目标对象的类型被自动推导出来,即使使用了不正确的格式化符也没有危险。

例子1 基本类型转换

#include "stdafx.h"
#include"iostream"
#include"sstream"
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{int a=10;double b=11.12;string c;char d[10];stringstream ss;ss<<a;ss>>c;//输入到流并抽取ss.clear();//多次转换的时候,必须调用stringstream的成员函数clear().ss<<b;ss>>d;//输入到流并抽取cout<<c<<"\n"<<d<<endl;return 0;
}

 注意:1.不仅支持普通类型也支持char*;

2.多次转换使用一个sstream对象时,使用成员函数clear()。这里sstream对象的构造和析构函数一般比较耗时,因此一般重复使用一个;

3.转换更加安全,自动,直接。

例子2 模板转换函数

#include "stdafx.h"
#include"iostream"
#include"sstream"
using namespace std;
template<class T>
void to_string(string & result,const T& t)
{ostringstream oss;//创建一个流oss<<t;//把值传递如流中result=oss.str();//获取转换后的字符转并将其写入result
}
template<class T1,class T2>
T2 typechange(T1& a)
{stringstream ss;T2 b;ss<<a;ss>>b;return b;
}
int _tmain(int argc, _TCHAR* argv[])
{int a=10;double b=11.12;string c="123.45";string d;to_string(d,a);cout<<d<<endl;//int to stringto_string(d,b);cout<<d<<endl;//double to stringa=typechange<string,int>(c);cout<<a<<endl;// string to int,注意123.45->123c=typechange<double,string>(b);cout<<c<<endl;//double to stringreturn 0;
}

注意:1.使用成员函数str()获取流内部缓冲的拷贝;

2.转换遵循从输入流中给变量赋值的规律,如string to int:“123.45”->123;

3.可以配合使用上文的setw(),setfill()进行到string的转换,如下所示:

stringstream ss;
ss<<"example:"<<setw(8)<<setfill('0')<<12;
ss>>d;
cout<<ss.str()<<endl;

部分参考自:sizeof()用法汇总

转载于:https://www.cnblogs.com/engineerLF/p/5393152.html

C++零碎知识点(一)相关推荐

  1. 【CSS基础】盒子模型、浮动布局、ps切图、定位及一些零碎知识点

    该系列文章是博主学习前端入门课程的笔记,同时也为了方便查阅,有任何问题都欢迎在评论区提出.本文主要介绍CSS基础知识,包括盒子模型.浮动布局.PS切图.定位.元素的显示与隐藏和一些零碎知识点 思维导图 ...

  2. Java基础零碎知识点,有这一篇就够了!

    Java零碎知识点整理 Java命名规范 数据类型 三元运算符 Random类 switch语句 方法重载 方法的参数传递 ArrayList集合创建 基本数据类型对应的引用数据类型 创建集合的代码实 ...

  3. 从 Option Explicit 开始的零碎知识点(一)

    这两天一直在围着 Visual Basic 转,敲了一些例子,记录了一些零碎的知识点,自己都觉得自己有点小幼稚,不过还好啦,至少没有放弃! <span style="font-fami ...

  4. 从 Option Explicit 开始的零碎知识点

    这两天一直在围着 Visual Basic 转,敲了一些例子,记录了一些零碎的知识点,自己都觉得自己有点小幼稚,不过还好啦,至少没有放弃! Option Explicit'定义变量X Dim xPri ...

  5. 【C语言】C语言零碎知识点

    目录 1.sizeof和strlen ​2.函数形参(很重要) ​2.1传值调用 ​2.2传址调用 ​3.函数的嵌套调用和链式访问 ​3.1函数的嵌套调用 ​3.2函数的链式访问 ​4.隐式类型转换 ...

  6. C#零碎知识点笔记(容易混淆的一些点)

    1:按CW+TAB就可以完成打印命令的快速输入: 2:声明变量的时候 记得在使用的时候给这个变量一个初始化: 3:明白 CPU___内存----硬盘 之间的 相互关系: 4:在增加浮点数的时候要记得为 ...

  7. Linq to sql与EF零碎知识点总结

    ------------------------------第一天(2013-3-25) 1.ado.net实体模型,(Ef) 2.创建上下文对象: 调用相应方法,最后调用.savechanges() ...

  8. j2se学习中的一些零碎知识点2之基础知识

    1.static修饰的变量叫做"静态变量"(静态变量,被存储在方法区中,所有的java对象共享静态变量,所有静态变量是类级别的,使用"类名."的方式访问.),s ...

  9. Android零碎知识点-更新中

    performClick 官方定义 官方API-View.html#performClick() Added in API level 1 public boolean performClick () ...

最新文章

  1. shell启动java_一个通用的java Application启动shell脚本
  2. HTTP1.0,1.1,2.0,HTTPS
  3. 在CentOS6.8下安装Docker
  4. Spring框架的事务管理的基本概念
  5. Java 中判断连接Oracle数据库连接成功
  6. python 字符串转换
  7. 接口测试--apipost参数定位
  8. 如今前端程序员还有前途吗?
  9. python编辑器使用技巧大全_markdown编辑器常用功能汇总
  10. 小熊派·叔BearPi-HM Micro环境搭建(双系统)
  11. 4g 访问App 慢的原因
  12. 加性高斯白噪声 AWGN
  13. 洛谷——P1713 麦当劳叔叔的难题
  14. 南阳oj 215 Sum
  15. [机器学习与scikit-learn-20]:算法-逻辑回归-线性逻辑回归linear_model.LogisticRegression与代码实现
  16. 苹果手机里微信数据删除怎么恢复?
  17. 《红楼梦》中四大家族到底犯了什么罪,导致了「白茫茫大地真干净」的结局?
  18. np.percentile()函数
  19. 探索多层次内存系统的页面管理设计空间Exploring the Design Space of Page Management for Multi-Tiered Memory Systems
  20. 每日词根——sol(完整可靠安慰,太阳严肃)

热门文章

  1. SAP IQ09 可以批量查询序列号数据
  2. 深度 | 学习如何学习的算法:简述元学习研究方向现状
  3. 干货丨计算机视觉必读:目标跟踪、网络压缩、图像分类、人脸识别等
  4. gitzip没有作用以及github如何下载单个文件或目录
  5. 强化学习之gym初战实战案例:悬崖案例CliffWalking-v0。
  6. 谁是深度学习框架一哥?2022 年,PyTorch 和 TensorFlow 再争霸
  7. 格力入局的数控机床,掌握“核心科技”有多难?
  8. 2019年度全球工程前沿研究报告
  9. NASA:首批“太空蜜蜂”将赴空间站上班,助宇航员推进研究!
  10. 你社交网站上的照片,也许已经被用来训练人工智能了