一个存储区的地址应该是它自身大小的整数倍(双精度浮点类型存储区的地址只需要是4的整数倍),这个规则叫数据对齐,结构体内部的存储区通常也需要遵守数据对齐的规则,数据对齐有可能导致结构体相邻子存储区之间有空隙

/*
* 数据对齐和补齐演示
* */
#include <stdio.h>
typedef struct {
char buf[2];
int num;
} tmp;
typedef struct {
char ch1;
int num;
char ch2;
} tmp1;
int main() {
printf("sizeof(tmp)是%d\n", sizeof(tmp));
printf("sizeof(tmp1)是%d\n", sizeof(tmp1));
return 0;
}

结构体存储区的大小必须是它所包含的占地最大,基本类型子存储区大小的整数倍(如果这个基本类型子存储区是doule类型的则结构体存储区的大小只需要是4的整数倍),这个规则叫数据补齐,数据补齐可能造成结构体最后会多占用一些无效的字节

枚举也可以用来创建新的数据类型,枚举类型存储区就是整数类型存储区,枚举类型存储区在使用的时候只能存放有限的几个整数,枚举类型也需要先声明然后才能使用,声明枚举类型的时候需要使用enum关键字,声明枚举类型的时候还需要提供一组名称,计算机为每个名称分配一个对应的整数,只有这些整数才能记录到枚举类型的存储区里.不同枚举类型存储区能记录的整数范围不同,计算机把从0开始的非负数按顺序分配给枚举类型里的所有名称,可以在声明枚举类型的时候指定把某个整数分配给某个名称,这个名称后面的所有名称被分配的整数也会随着改变

/*
* 枚举类型演示
* */
#include <stdio.h>
int main() {
enum /*season*/ {CHUN, XIA = 5, QIU, DONG};
printf("QIU是%d\n", QIU);
return 0;
}

联合也可以用来创建数据类型,联合也需要先声明然后才能使用,声明联合的时候需要使用union关键字,联合的所有子存储区所占的内存互相重叠,联合所有成员变量存储区的开始地址都一样,联合存储区的大小就是最大子存储区的大小,联合类型的存储区可以当作多种不同类型的存储区使用,每个成员变量代表了一种可选的类型

/*
* 联合演示
* */
#include <stdio.h>
typedef union {
int num;
float fnum;
} tmp;
int main() {
tmp utmp = {0};
printf("&(utmp.num)是%p\n", &(utmp.num));
printf("&(utmp.fnum)是%p\n", &(utmp.fnum));
printf("sizeof(tmp)是%d\n", sizeof(tmp));
return 0;
}

记录普通类型存储区地址的指针叫做一级指针,记录一级指针存储区地址的指针叫做二级指针,二级指针也需要先声明然后才能使用,声明二级指针的时候需要写两个*

/*
* 二级指针演示
* */
#include <stdio.h>
int main() {
int num = 0;
int *p_num = #
int **pp_num = &p_num; //二级指针声明语句
**pp_num = 10;
printf("num是%d\n", num);
*pp_num = NULL;
printf("p_num是%p\n", p_num);
return 0;
}

二级指针名称前使用**可以表示捆绑的普通类型存储区,二级指针名称前使用*可以表示捆绑的一级指针存储区

如果把指针数组里第一个指针的地址记录到一个二级指针里就可以通过这个二级指针找到指针数组里的每个指针,这个时候可以把二级指针看作指针数组的代表,二级指针不可以代表二维数组

/*
* 二级指针形式参数演示
* */
#include <stdio.h>
typedef struct {
int row;
int col;
} pt;
typedef struct {
pt center;
int radius;
} circle;
void larger(const circle *p_cl1, const circle *p_cl2, circle **pp_cl) {
*pp_cl = (circle *)(p_cl1->radius >= p_cl2->radius ? p_cl1 : p_cl2);
}
int main() {
circle cl1 = {0}, cl2 = {0}, *p_cl = NULL;
printf("请输入一个圆的位置:");
scanf("%d%d%d", &(cl1.center.row), &(cl1.center.col), &(cl1.radius));
printf("请再输入一个圆的位置:");
scanf("%d%d%d", &(cl2.center.row), &(cl2.center.col), &(cl2.radius));
//p_cl = larger(&cl1, &cl2);
larger(&cl1, &cl2, &p_cl);
printf("比较大的圆是((%d, %d), %d)\n", p_cl->center.row, p_cl->center.col, p_cl->radius);
return 0;
}

无类型指针有可能实际记录的是一级指针的地址,这个时候就必须首先把它强制类型转换成二级指针然后才能使用

二级指针通常作为函数的形式参数使用,被调用函数可以通过二级指针形式参数向调用函数传递一个地址数据

C语言里函数也有地址,函数的名称可以表示函数的地址,函数指针可以用来记录函数的地址,函数指针也需要先声明然后才能使用,函数指针的声明可以根据函数声明变化得到

/*
* 函数指针演示
* */
#include <stdio.h>
int add(int num, int num1) {
return num + num1;
}
int main() {
int (*p_func)(int, int) = add; //函数指针声明
printf("add是%p\n", add);
printf("结果是%d\n", p_func(3, 8));
return 0;
}

函数指针也分类型,不同类型的函数指针适合与不同格式的函数捆绑

函数指针可以用来调用函数

标准C语言(12)的更多相关文章

  1. 关于标准C语言的预定义宏【转】

    标准C语言预处理要求定义某些对象宏,每个预定义宏的名称一两个下划线字符开头和结尾,这些预定义宏不能被取消定义(#undef)或由编程人员重新定义.下面预定义宏表,被我抄了下来. __LINE__  当 ...

  2. 【部分原创】标准C语言的优先级、结合性、求值顺序、未定义行为和非确定行为浅析

    零. 优先级    在C++ Primer一书中,对于运算符的优先级是这样描述的:     Precedence specifies how the operands are grouped. It ...

  3. 标准SQL语言的用法

    原文链接:http://www.ifyao.com/2015/05/18/%E6%A0%87%E5%87%86%E7%9A%84sql%E8%AF%AD%E8%A8%80%E4%BD%BF%E7%94 ...

  4. 关于标准C语言的预定义宏

    标准C语言预处理要求定义某些对象宏,每个预定义宏的名称一两个下划线字符开头和结尾,这些预定义宏不能被取消定义(#undef)或由编程人员重新定义.下面预定义宏表,被我抄了下来.__LINE__  当前 ...

  5. UML标准建模语言与应用实例

    一.基本信息 标题:UML标准建模语言与应用实例 时间:2012 出版源:科技创新导报 领域分类:UML标准建模语言 面向对象 系统分析与设计 二.研究背景 问题定义:UML建模语言用图形来表现典型的 ...

  6. UML(统一建模语言)是通用的可视化标准建模语言。由构造块、公共机制、构架三部分组成。

    UML UML(统一建模语言)是通用的可视化标准建模语言.由构造块.公共机制.构架三部分组成. 1.构造块:包括基本的UML建模元素(类.接口.用例等).关系(关联关系.依赖关系.泛化关系.实现关系) ...

  7. 标准C 语言总结

    ***************C语言****************** --day01-- Linux是一个和Windows类似的操作系统 通常通过终端软件使用Linux操作系统 终端软件里只能使用 ...

  8. python标准库介绍——12 time 模块详解

    ==time 模块== ``time`` 模块提供了一些处理日期和一天内时间的函数. 它是建立在 C 运行时库的简单封装. 给定的日期和时间可以被表示为浮点型(从参考时间, 通常是 1970.1.1 ...

  9. 标准C语言的输入输出流(i/o)方法详解

    cppreference.com -> 标准 C I/O ->详细说明 标准 C I/O clearerr 语法: #include <stdio.h> void cleare ...

随机推荐

  1. Eclipse 设置护眼背景色

    Eclipse 设置护眼背景色 1.设置字体大小 Window --> Preferences --> General --> Apprearance --> Colors a ...

  2. Day6 && Day7图论

    并查集 A - How Many Answers Are Wrong 题意:已知区间[1,n],给出m组数据,即[l,r]区间内数据之和为s,求错误数据的数量. 拿到这道题,真的没思路,知道用并查集, ...

  3. 走进异步编程的世界 - 在 GUI 中执行异步操作

    转载:https://www.cnblogs.com/liqingwen/p/5877042.html 走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://w ...

  4. SqlServer:SqlServer(sql,游标,定时作业,行转列,列转行,公用表达式递归,merge合并)

    1.加载驱动: Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver"); DriverManager.getCo ...

  5. python多进程——进程间通信

    (一)进程锁 抢票的例子: # -*- coding:utf-8 -*- from multiprocessing import Process, Lock import time import js ...

  6. swoole前置基础知识1——1.1多进程/多线程的概念

    一.为何需要多进程(或者多线程),为何需要并发? 这个问题或许本身都不是个问题.但是对于没有接触过多进程编程的朋友来说,他们确实无法感受到并发的魅力以及必要性. 我想,只要你不是整天都写那种int m ...

  7. react-redux provider组件

    provider组件概念图   react-redux provider组件概念图 provider组件的作用 provider包裹在根组件外层,使所有的子组件都可以拿到state.示例: impor ...

  8. Ubuntu 16.04安装anaconda3

    首先下载anaconda3镜像,清华大学开源软件镜像站下载地址:https://mirrors.tuna.tsinghua.edu.cn/anaconda/archive/ 因为实验需要用到tenso ...

  9. NOIp2018D1T2 货币系统【分析&完全背包】

    题目传送门 看到题目瞬间想起某凯的疑惑,感觉不会做....然后观察样例可以知道,去掉原来货币系统中能够被其他币值凑出来的数就是答案(样例分析法),然后就完事了(huaji). 简单理解一下吧: 首先, ...

  10. Java严选

    1,假如有两个线程,一个线程A,一个线程B都会访问一个加锁方法,可能存在并发情况,但是线程B访问频繁,线程A访问次数很少,问如何优化.(然后面试官说有了解过重度锁和轻度锁吗) a,竞争资源不激烈,选择 ...