C++零碎知识点(一)-程序员宅基地

技术标签: c/c++  

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

例子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”库定义了三种类:istringstreamostringstreamstringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本。

   <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 string
	to_string(d,b);cout<<d<<endl;//double to string
	a=typechange<string,int>(c);cout<<a<<endl;// string to int,注意123.45->123
	c=typechange<double,string>(b);cout<<c<<endl;//double to string
        return 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

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_34167819/article/details/93360761

智能推荐

稍微深入分析Ubuntu环境下安装NVIDIA驱动导致黑屏的原因_prime-select nvidia 黑屏-程序员宅基地

文章浏览阅读8.2k次,点赞5次,收藏48次。本文承接之前写的有关如何用正确姿势安装NVIDIA驱动的博文 (https://blog.csdn.net/Edward_ed_liu/article/details/109552761)。首先之所以要更新Linux内核,是因为不更新内核就无法使用笔记本自带的无线网卡。其次,目前NVIDIA官方不建议把Linux内核更新到最新版(5.9),而且这条消息只在英文的官网才有,中文的则是广告。TWICE如果强行更新到5.9版本,之后的Cuda安装表面上会显示成功,但在实际使用Cuda的过程中._prime-select nvidia 黑屏

java中间件有哪些_金九银十期间成功斩获58万架构师Offer!六面字节跳动面经和面试题分享 - 小梦爱Java...-程序员宅基地

文章浏览阅读167次。金九银十期间成功斩获58万Offer!六面字节跳动面经(成功关键:吃透九大核心知识+狂刷大厂面试真题)第一轮:团队面试第一轮基本上是你的团队成员面试你,是和你同级或者高你一个P的师兄来面你,我的话基本没问什么特别的,主要还是讲自己简历上的做的项目,这里需要你很熟悉自己的项目才行,我个人觉得这里你要把项目里你的角色做了什么没做什么讲清楚,然后最好能把自己做的那部分重点展开来讲,然后面试官会从你讲的内..._java架构师 中间件简历

【规则引擎】一、规则引擎简介_规则引擎 风险 功能-程序员宅基地

文章浏览阅读1.9k次。(第一章规则引擎学习入门之规则引擎简介)# 系列文章目录规则引擎简介前言一、为什么要使用规则引擎?1.不使用规则引擎的规则执行现状2. 规则引擎优点二、规则引擎的功能三、规则引擎的分类实现1.事中规则实现2.事后规则实现四、规则引擎调研1.开源规则引擎2.商业规则引擎五、Drools六、Aviator前言规则引擎由推理引擎发展而来,是一种嵌入在应用程序中的组件,实现了将业务决策从应用程序代码中分离出来,并使用预定义的语义模块编写业务决策。接受数据输入,解释业务规则,并根据业务规则做出业务决策。–来自百_规则引擎 风险 功能

oracle设置生成归档日志大小,Oracle 改变归档日志大小-程序员宅基地

文章浏览阅读1k次。该变归档日志大小只有改变日志组的大小!方法:加入新的大的日志文件,然后删掉旧的小的日志文件假设现有三个日志组,每个组内有一个成员,每个成员的大小为1MB,现在想把此三个日志组的成员大小都改为10MB1、创建2个新的日志组alter database add logfile group 4 ('D:\ORACLE\ORADATA\ORADB\REDO04_1.LOG') size 1024k;alt..._oracle归档日志空间大小

this.options在chrome浏览器提示undefined的解决办法_options is undefined-程序员宅基地

文章浏览阅读7.5k次。很早用的一段三级联动下拉菜单最近发现在chrome里不能联动下拉了,ie下正常,很奇怪,这段代码在之前有段时间经常用,没出现过什么问题,后来调试发现在“this.options.value”处提示“this.options is unfioned”,应该是浏览器之间js用法不同的问题,查资料测试后,改成“this.value”就正常了_options is undefined

整合Spring Cloud Bus报错_failed to instantiate [org.springframework.boot.ac-程序员宅基地

文章浏览阅读680次。spring: cloud: refresh: enabled: false2020-09-20 20:41:40.882 ERROR 13292 —[ost-startStop-1] o.s.b.web.embedded.tomcat.TomcatStarter :Error starting Tomcat context. Exception: org.springframework.beans.factory.BeanCreationException.Me_failed to instantiate [org.springframework.boot.actuate.endpoint.web.servlet

随便推点

当下 React 项目该放弃的以及更好用的技术推荐-程序员宅基地

文章浏览阅读210次。React 版本推荐当前 React 都已经发布 18 了,虽然是个 alpha 版本,但是 17 确实也已经有大厂在用了。目前如果你的版本还停留在 v16.8 之前的话还是尽早升了吧。毕..._react 放弃

C++基础-资源管理:堆、栈与 RAII_c++heap内存堆管理-程序员宅基地

文章浏览阅读161次。堆,英文是 heap,在内存管理的语境下,指的是动态内存分配的区域,和数据结构中的“大根堆和小根堆”不是一个概念。同时,这里堆分配的内存需要手工释放,否则会造成内存泄漏。 C++ 标准里有一个和堆相关的概念是自由存储区,英文是 free store,特指使用 new 和 delete 来分配和释放内存的区域。一般而言,这是堆的一个子集。_c++heap内存堆管理

CentOS7中安装MySQL8.0.21爬坑记录:1045-Access denied、Job for firewalld.service failed等异常_error: failed to read file "/proc/sys/net/netfilte-程序员宅基地

文章浏览阅读1.3k次。在CentOS7.3中安装了MySQL8.0.21之后,就开启了一段漫长的爬坑历程,简要回顾如下:一、从Win10中用Navicat连接安装好的MySQL服务器出现如下异常:1045 - Access denied for user ‘root’@‘192.168.101.151’(using password: YES) 于是,在网上查阅了多篇博客,结论可能是3306端口没有加入到防火墙的允许列表。之后,开始研究了CentOS7中的防火墙,发现RHEL6之前版本用的防火墙管理工具都是iptab_error: failed to read file "/proc/sys/net/netfilter/nf_conntrack_helper": [e

Java位运算技巧_java巧用位运算记录用户-程序员宅基地

文章浏览阅读3.2k次,点赞6次,收藏16次。位运算作为底层的基本运算操作,往往是和'高效'二字沾边,适当的运用位运算来优化系统的核心代码,会让你的代码变得十分的精妙。以下是我所遇之的一些简单的位运算技巧作为博文记录。1.获得int型最大值 public static void main(String[] args) { int maxInt = (1 << 31) - 1; ..._java巧用位运算记录用户

HTML的列表标签,表格table和表单标签_html用ul写表格-程序员宅基地

文章浏览阅读710次。名词1名词1解释1...名词2名词2解释1名词2解释2..._html用ul写表格

HDU - 4333 Revolving Digits(扩展KMP)-程序员宅基地

文章浏览阅读187次。题目链接:点击查看题目大意:给出一个由 n 个数位组成的数字,现在可以通过将其不同的后缀移到前面来组成 n 个新的数字,现在要求出 n 个新数字进行去重后,有多少个新数字分别大于、等于、小于原数字如:1234进行上述转移可以得到的四个新数字分别为:1234,4123,3412,2341题目分析:如果暴力的比较虽然看似只需要枚举 n 个新的字符串,但是每个字符串的比较还需要花费O(n)的..._hdu - 4333