我基本上总结出以下4部分:

1、问题的足迹大小。

2、字节对齐问题。

3、特别保留位0。

4、这种结构被存储在存储器中的位置。

#include <stdio.h>
#include <stdlib.h>
#include <string.h> //基本概念
/*
struct _M
{
(1) 类型 參数名 : 占位大小。
(2) 类型 : 占位大小;
} (1)
类型 -- int,unsigned(32位),short,char。 參数名 -- 同个结构体里面不能重名參数。 占位大小 -- 不能大于类型最最大位数。 (2) 一般用于寄存器中保留位
*/ //占位大小问题:
/*
err -- int类型是32位。34大于32位,所以编译出错。
struct _A
{
int x1:34;
}; err -- char类型是8位,9大于8位,所以编译出错。
struct _A1
{
char x1:9;
};
*/ //字节对齐
/*
偏移量 -- 类型和占位大小影响(与结构体字节对齐差点儿相同意思)。 */
struct B
{
char x1:1;//1位。字节对齐占1字节,8位。
char x2:8;
char x3:1;
char x4:8;
char x5:1;
}; struct C
{
char x1:4;
char :1;
char x2:3;
}; struct _C
{
char x1:4;
char :8;
char x2:3;
}; //特殊保留位0
struct C1
{
char x1:4;
char :0;//这个0占了4个字节
char x2:3;
}; struct _C1
{
char x1:4;
char :4;//这个0占了4个字节
char x2:3;
}; //内存中存放顺序
struct D
{
char x1:1;//最低位
char x2:1;
char x3:1;
char x4:1;
char x5:1;
char x6:1;
char x7:1;
char x8:1;//最高位
}; int
main()
{
//字节对齐
B b;
printf("sizeof(b) = %d\n", sizeof(b)); C c;
printf("sizeof(c) = %d\n", sizeof(c)); _C _c;
printf("sizeof(_c) = %d\n", sizeof(_c)); //特殊保留位0
C1 c1;
printf("sizeof(c1) = %d\n", sizeof(c1));
memset(&c1, 0, sizeof(c1) );
c1.x1 = 0xf;
c1.x2 = 0x7;
printf("c1 = 0x%x\n", c1); _C1 _c1;
printf("sizeof(_c1) = %d\n", sizeof(_c1));
memset(&_c1, 0, sizeof(_c1) );
_c1.x1 = 0xf;
_c1.x2 = 0x7;
printf("_c1 = 0x%x\n", _c1); //内存中存放顺序
D d;
memset(&d, 0, sizeof(d) );
printf("d = 0x%08x\n", d);
d.x1 = 1;
printf("d = 0x%08x\n", d); } /*
[root@localhost test_class]# gcc quote.cpp ;./a.out
sizeof(b) = 5
sizeof(c) = 1
sizeof(c1) = 2
c1 = 0x70f
sizeof(c2) = 3
d = 0x00000000
d = 0x00000001
[root@localhost test_class]# gcc quote.cpp ;./a.out
sizeof(b) = 5
sizeof(c) = 1
sizeof(_c) = 3
sizeof(c1) = 2
c1 = 0x70f
sizeof(_c1) = 2
_c1 = 0x70f
d = 0x00000700
d = 0x00000701 */

以下是别人的博客:

在使用结构体位制的时候有两点要特别注意:
1.//位段成员的类型仅可以为unsigned或者int
2.
unsigned b:4;
unsigned :0; //定义长度为0的位段时不能指定名字,否则编译只是
unsigned d:1; //定义了0字段后,紧接着的下一个成员从下一个存储单元開始存放; //此样例中,d前面那个存储单元中的余下的27位中被0填充了 /*
DATE : 2010.6.24
关于C中的位端igned或者int
*/
#include <stdio.h> typedef struct _A
{
unsigned int a:4;//位段成员的类型仅可以为unsigned或者int
unsigned b:4;
unsigned c:2;
unsigned d:6;
unsigned E:1;
unsigned D:2;
unsigned T:3;
unsigned A:9;
unsigned h:4; //前面已经为31,故4+31>32已超过一个存储单元,所以4在一个新的存储单元存放
unsigned y:29;//由于前面的4在一个新的存储单元的开头存放,且29+4>32, 故在还有一个新的存储单元存放
}A; //所以最后求出的A的大小是4 + 4 + 4 =12 /*对上面的详细解释: 一个位段必须存储在同一个存储单元中,不能跨两个单元.假设某存储单元空间中不能容纳
下一个位段,则改空间不用,而从下一个存储单元起存放该位段. 结构体A中的h和y就是这样的情况.
在gcc环境下,測试后,一个存储单元为4个字节.
*/ typedef struct _S
{
unsigned a:4;
unsigned b:4;
unsigned c:22;
unsigned q:1;
unsigned h:1;
//unsigned i:33; // 错误:‘i’ 的宽度超过它自身的类型
//unsigned i:1;当多出此行时,该结构体大小由4变为8,由于此行之前正好为32位
} S; typedef struct _T
{ //当没有占满一个存储单元时,结构体的大小对齐为一个存储单元的大小
unsigned a:2;
unsigned b:2;
unsigned j:1;
unsigned : 1;//可以定义无名位段,此例中该无名位段占用1位的空间,该空间将不被使用
} T; typedef struct _V
{
unsigned a:1;
unsigned b:4;
unsigned :0; //定义长度为0的位段时不能指定名字,否则编译只是
unsigned d:1; //定义了0字段后,紧接着的下一个成员从下一个存储单元開始存放;
}V; //此样例中,d前面那个存储单元中的余下的27位中被0填充了 int main()
{
A a; S s; T t; V v;
printf("sizeof(a)=%d\n", sizeof(a));
printf("sizeof(s)=%u\nsizeof(t)=%u\n", sizeof(s), sizeof(t));
printf("sizeof(v)=%d\n", sizeof(v));
return 0;
}

版权声明:本文博主原创文章。博客,未经同意不得转载。

姿势体系结构的详细解释 -- C的更多相关文章

  1. Linux下函数调用堆栈帧的详细解释【转】

    转自:http://blog.chinaunix.net/uid-30339363-id-5116170.html 原文地址:Linux下函数调用堆栈帧的详细解释 作者:cssjtuer http:/ ...

  2. cmd 环境变量设置方法详细解释

    cmd设置环境变量可以方便我们bat脚本的运行,但是要注意的是变量只在当前的cmd窗口有作用(局部生效),如果想要设置持久的环境变量需要我们通过两种手段进行设置:一种是直接修改注册表,另一种是通过我的 ...

  3. .htaccess语法之RewriteCond与RewriteRule指令格式详细解释

    htaccess语法之RewriteCond与RewriteRule指令格式详细解释 (2012-11-09 18:09:08) 转载▼ 标签:  htaccess it 分类: 网络 上文htacc ...

  4. cookie的详细解释

    突然看到网页上中英文切换的效果,不明白怎么弄得查了查 查到了cookie 并且附有详细解释 就copy留作 以后温习 http://blog.csdn.net/xidor/article/detail ...

  5. tar命令的详细解释

    tar命令的详细解释 标签: linuxfileoutputbashinputshell 2010-05-04 12:11 235881人阅读 评论(12) 收藏 举报  分类: linux/unix ...

  6. Linux学习笔记15——GDB 命令详细解释【转】

    GDB 命令详细解释 Linux中包含有一个很有用的调试工具--gdb(GNU Debuger),它可以用来调试C和C++程序,功能不亚于Windows下的许多图形界面的调试工具. 和所有常用的调试工 ...

  7. C语言 - 结构体(struct)比特字段(:) 详细解释

    结构体(struct)比特字段(:) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/26722511 结构体(struc ...

  8. Java - 面向对象(object oriented)计划 详细解释

    面向对象(object oriented)计划 详细解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/24058107 程序包括 ...

  9. 设计模式 - 迭代模式(iterator pattern) Java 迭代器(Iterator) 详细解释

    迭代模式(iterator pattern) Java 迭代器(Iterator) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy 參考迭代器模式(ite ...

随机推荐

  1. C#-循环滚动字幕,timer,从左至右,从右至左,暂停---ShinePans

    Lable的Left属性是能够更改的,可是 Right属性不能够更改,所以我们能够利用 这个特点做自加 自减运算 using System; using System.Collections.Gene ...

  2. C#之任务,线程和同步

    1 概述 对于所有需要等待 的操作,例 如 ,因 为文件 . 数据库或网络访 问都需要一定 的时间,此 时就可以启 动一个新线程,同时完成其他任务,即使是处理密集型的任务,线程也是有帮助的. 2 Pa ...

  3. linux操作提示:“Can&#39;t open file for writing”或“operation not permitted”的解决的方法

    在linux上使用vi命令改动一个文件内容的时候,发现无法保存,每次写完使用":q!"命令能够正常退出可是使用":wq!"命令保存文件并退出时出现一下信息提示: ...

  4. jQuery获取url参数值

    $.extend({ getUrlVars: function () { var vars = [], hash; var hashes = window.location.href.slice(wi ...

  5. spring整合flex

    在常规的开发中只是用flex二不和后台交互是不可能的,为此flex也提供了和后台交互的2种解决方案一种是Data Services另一种是BlazeDs,本篇博客是用的是后一种,我的开发步骤如下: 1 ...

  6. POJ 36666 Making the Grade 简单DP

    题意是: 给出n个数,让你用最小的花费将其改成非递增或非递减的 然后花费就是新序列与原序列各个位置的数的差的绝对值的和 然后可以看到有2000个数,数的范围是10亿 仔细观察可以想象到.其实改变序列中 ...

  7. JQuery日记_5.13 Sizzle选择器(六)选择器的效率

        当选择表达式不符合高速匹配(id,tag,class)和原生QSA不可用或返回错误时,将调用select(selector, context, results, seed)方法,此方法迭代DO ...

  8. 王立平--Program Files (x86)

    window7根据系统.program files(x86) 它是应用程序目录,在64下位系统.为了更好的相容性32位程序,在一些安装32位程序(请注意,有些节目自己是32位),将默认被安装progr ...

  9. HDU 1856 More is better(并查集+离散化)

    题目地址:HDU 1856 水题.因为标号范围太大,而数据数仅仅有10w,所以要先进行离散化.然后就是裸的并查集了. 代码例如以下: #include <iostream> #includ ...

  10. ios 仿新浪微博 UINavigationController 向左滑动时显示上一个控制器的View.

    仿新浪微博 UINavigationController 向左滑动时显示上一个控制器的View. 实现原理,UINavigationController 的 self.view显示时把当前显示的vie ...