不可或缺 Windows Native (6) - C 语言: 函数
作者:webabcd
介绍
不可或缺 Windows Native 之 C 语言
- 函数
示例
cFunction.h
#ifndef _MYHEAD_FUNCTION_
#define _MYHEAD_FUNCTION_ #ifdef __cplusplus
extern "C"
#endif // 函数声明
// 像这种在 .h 中声明的函数,如果想被外部文件调用的话,则外部文件不用再声明即可调用
char *demo_cFunction(); long recursion(int n); void update(int ary[], int aryLength);
// 也可以将上面的函数声明写成下面这种形式,像这种只有参数类型,没有参数名称的函数声明称之为函数原型(function prototype)
// void update(int[], int); void demo_variable1(); void demo_variable2(); void demo_params(); // 加上 static 关键字就是内部函数,仅能在当前文件使用;在不同的文件中如果有同名的内部函数,则他们互不干扰
static void cFunction_demo1(); // 加上 extern 关键字就是外部函数,可被其他文件调用
extern void cFunction_demo2(); // 如果既无 static 也无 extern,则默认为 extern
void cFunction_demo3(); #endif
cFunction.c
/*
* 函数
*
*
* 1、从变量的作用域(scope)角度来分,可以分为全局变量(外部变量)和局部变量
* 2、从变量的生存期角度来分,可以分为静态存储方式和动态存储方式
*
*
* argument - 实参
* parameter - 形参
*/ #include "pch.h"
#include "cFunction.h"
#include "cHelper.h" // 全局变量(亦称外部变量),作用域(scope)是整个源程序
int global01 = ; // 像这种定义在 .c 中的函数(无论在 .c 文件中有无函数声明),如果想被外部文件调用的话,则外部文件在调用之前需要先声明
void function0()
{
;
} char *demo_cFunction()
{
// 函数定义在之前,则不用声明也可以调用(同一文件);函数定义在之后,则必须先声明才可以调用
function0(); // 以下为无参数无返回值的函数声明(declare),其同 void demo_cFunction1();
void function1(void); // 以下为有参数有返回值的函数声明
int function2(int a, int b); // 也可以将上面的函数声明写成下面这种形式,像这种只有参数类型,没有参数名称的函数声明称之为函数原型(function prototype)
// int demo_cFunction2(int, int); // 无参数,无返回值函数演示
function1(); // 有参数,有返回值函数演示
int a = , b = ; // 这里的 a 和 b 是实参
int x = function2(a, b); // 函数递归
long y = recursion(); // 结果:5050 // 向函数传递数组类型的演示
// 之前说过 ary 是数组的首地址,也就是说传递过去的是指针,也就是说实参和形参其实指向的是一个地方,相当于引用类型
int ary[] = {, , , , };
update(ary, ); // 数组 ary 被更新了 // 局部变量和全局变量的演示
demo_variable1(); // 变量存储方式的演示
demo_variable2(); // 参数不定的函数(可变参数)
demo_params(); return "看代码及注释吧";
} // 无参数,无返回值函数演示
void function1()
{
// 空参数时,写 void 和不写 void 的区别(C 语言建议没有参数时使用 void) /*
void fun(); // 代表可以传递任意参数(C++ 则代表不能传递任何参数)
fun(1); // 正常编译(在本例中这么做其实是会编译错误的,因为用的是 C++ 编译器)
fun(1, 2, 3); // 正常编译(在本例中这么做其实是会编译错误的,因为用的是 C++ 编译器)
*/ /*
void fun(void); // 代表不能传递任何参数
fun(1); // 编译错误
fun(1, 2, 3); // 编译错误
*/
} // 有参数,有返回值函数演示
int function2(int a, int b)
{
// 这里的 a 和 b 是形参,形参就是实参的副本
// 形参变量只在被调用期间才分配内存单元,调用结束立即释放,也就是说形参变量只有在函数内才是有效的,离开该函数就不能再使用了,即形参的作用于在函数内
return a + b;
} // 演示函数的递归调用
// return: 0 + 1 + 2 + 3 + ... + n
long recursion(int n)
{
long result = ; if (n > )
{
result += n;
result += recursion(n - );
} return result;
} // 参数为数组类型的演示(之所以需要传递 aryLength 数组长度,是因为传递过来的 ary 是个指针,无法计算其指向的数组长度。就好像 main 函数似的,需要传递参数个数)
void update(int ary[], int aryLength)
{
// 此处实参和形参其实指向的是一个地方
for (int i = ; i < aryLength; i++)
{
ary[i] = ;
} } // 局部变量和全局变量的演示
void demo_variable1()
{
// 我是局部变量,作用域在函数内
int i = ;
{
// 我是复合语句内的局部变量,作用域在复合语句内,会屏蔽外面的同名变量
int i = ;
} // 我是局部变量,如果全局变量有与我同名的,则“屏蔽”全局变量
int global01 = ; // 上面“屏蔽”了全局变量,那我要怎么引用全局变量呢?像下面这样用“::”
int global01Outside = ::global01; // c++
} // 变量存储方式的演示
void demo_variable2()
{
// 从变量的生存期角度来分,可以分为静态存储方式和动态存储方式
// 1、静态存储方式:是指在程序运行期间分配固定的存储空间的方式
// 2、动态存储方式:是在程序运行期间根据需要进行动态的分配存储空间的方式 // 全局变量在程序运行期间都会被分配固定的存储空间(静态存储方式) // 局部变量默认就是 auto 的,即动态分配存储空间。下面这句写全了就是 auto int i = 0;(动态存储方式)
int i = ; // 如果希望函数中的局部变量的值在函数调用结束后不消失的话,则应该用 static 指定局部变量为“静态局部变量”(静态存储方式)
static int j = ; // 用 register 指定“寄存器变量”,其会保存在 cpu 中的寄存器中。一个最佳场景是,将其作为循环控制的变量
// 寄存器变量无地址,所以不能用“&”对其取地址(C++ 可以取其地址,但是编译器会自动将其变为内存变量)。另外,寄存器不够用的话会自动变为内存变量
register int x = ; // 用 volatile 保证变量必在内存中,而不会被放入 cpu 寄存器(如果编译器认为某变量无外部修改,则为了优化会将其放入寄存器)
volatile int y = ; // 全局变量是在函数的外部定义的,它的作用域为从变量定义处开始,到本程序文件的末尾
// 如果在定义点之前的函数想引用该全局变量,则应该使用 extern 表明该变量在别的地方已经定义过了,在这里要使用那个变量
extern int global02, global03; // 如果没有这句的话,则编译会报“未声明的标识符”的错误 int p02 = global02;
int p03 = global03;
} int global02 = , global03 = ; // 以下演示如何定义及使用参数不定的函数(可变参数) #include <stdarg.h> // 其定义了 va_start, va_arg, va_end (va 应该是 variable argument(可变参数)的简写) void demo_params()
{
// 计算 n 个整数的平均值,第一个参数代表后面的参数的个数
float average(int paramCount, ...); float result = average(, , , , ); // 2.5
} float average(int paramCount, ...)
{
// 定义一个参数列表
va_list va_params; long sum = ; // 告知不定参数的个数,并准备读取参数
va_start(va_params, paramCount); for (int i = ; i < paramCount; i++)
{
// 一个一个地按照指定的类型读取参数的值,每读取完一个,参数指针就会向后移动继续读取下一个
int value = va_arg(va_params, int);
sum += value;
} // 停止读取参数
va_end(va_params); return sum * 1.0f / paramCount;
}
OK
[源码下载]
不可或缺 Windows Native (6) - C 语言: 函数的更多相关文章
- 不可或缺 Windows Native (1) - C 语言: hello c
[源码下载] 不可或缺 Windows Native (1) - C 语言: hello c 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 在 Windows Sto ...
- 不可或缺 Windows Native (3) - C 语言: 运算符,表达式,条件语句,循环语句,转向语句,空语句等
[源码下载] 不可或缺 Windows Native (3) - C 语言: 运算符,表达式,条件语句,循环语句,转向语句,空语句等 作者:webabcd 介绍不可或缺 Windows Native ...
- 不可或缺 Windows Native (4) - C 语言: 预处理命令,输入,输出
[源码下载] 不可或缺 Windows Native (4) - C 语言: 预处理命令,输入,输出 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 预处理命令 输入 ...
- 不可或缺 Windows Native (7) - C 语言: 指针
[源码下载] 不可或缺 Windows Native (7) - C 语言: 指针 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 指针 示例cPointer.h #i ...
- 不可或缺 Windows Native (9) - C 语言: 动态分配内存,链表,位域
[源码下载] 不可或缺 Windows Native (9) - C 语言: 动态分配内存,链表,位域 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 动态分配内存 链 ...
- 不可或缺 Windows Native (10) - C 语言: 文件
[源码下载] 不可或缺 Windows Native (10) - C 语言: 文件 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 文件 示例cFile.h #ifn ...
- 不可或缺 Windows Native (23) - C++: 虚函数
[源码下载] 不可或缺 Windows Native (23) - C++: 虚函数 作者:webabcd 介绍不可或缺 Windows Native 之 C++ 虚函数 示例1.基类CppHuman ...
- 不可或缺 Windows Native (20) - C++: 友元函数, 友元类
[源码下载] 不可或缺 Windows Native (20) - C++: 友元函数, 友元类 作者:webabcd 介绍不可或缺 Windows Native 之 C++ 友元函数 友元类 示例演 ...
- 不可或缺 Windows Native (2) - C 语言: 常量,变量,基本数据类型
[源码下载] 不可或缺 Windows Native (2) - C 语言: 常量,变量,基本数据类型 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 常量 变量 基本 ...
随机推荐
- [GraphQL] Use GraphQLNonNull for Required Fields
While certain fields in a GraphQL Schema can be optional, there are some fields or arguments that ar ...
- centos 7 中 tomcat 安装
一. 安装 jdk [root@iZm5ei4m ~]# /usr/local/tomcat/bin/startup.sh 一.安装 jdk 二.设置环境变量 #################### ...
- 基情四射的两个css样式
自定义blog样式时,代码段的line-height继承样式post的line-height,间隔太大了,决定再减小点,css都玩了几年了,感觉中这是很容易的事情.然后,就悲剧了好久,原先自定义样式表 ...
- Android开发之Canvas rotate方法释疑
Canvas的rotate()函数本应该是很简单的一个函数,但是由于api手册言之不详,使用中难免有吃不准的地方.下面所记录的几点,都是我在使用中所迷惑过的问题,特此记录. 1,坐标原点在哪里? 如果 ...
- 转载 ACM训练计划
leetcode代码 利用堆栈:http://oj.leetcode.com/problems/evaluate-reverse-polish-notation/http://oj.leetcode. ...
- 转 如何理解 重要性采样(importance sampling)
分类: 我叫学术帖2011-03-25 13:22 3232人阅读 评论(4) 收藏 举报 图形 重要性采样是非常有意 思的一个方法.我们首先需要明确,这个方法是基于采样的,也就是基于所谓的蒙特卡洛法 ...
- SQLite 入门教程 (终端)命令
一.基本简介 SQLite 是一个自持的(self-contained).无服务器的.零配置的.事务型的关系型数据库引擎.因为他很小,所以也可以作为嵌入式数据库内建在你的应用程序中.SQLite 被应 ...
- Java版的Quartz表达式生成器,同时适用于Quartz.net(免费下载)
Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用.Quartz可以用来创建简单或为运行十个,百个, ...
- [转]MySQL与MongoDB的操作对比
MySQL与MongoDB都是开源的常用数据库,但是MySQL是传统的关系型数据库,MongoDB则是非关系型数据库,也叫文档型数据库,是一种NoSQL的数据库.它们各有各的优点,关键是看用在什么地方 ...
- Reveal查看任意app的高级技巧(转)
原文:http://zhuanlan.zhihu.com/iOSRe/19646016 Reveal查看任意app的高级技巧 hangcom · 12 小时前 Reveal是一个很强大的UI分析工具, ...