c语言进阶3-有参函数
一、 有参函数的定义
有参函数的定义格式如下:
类型标识符 函数名(形式参数表列)
{
语句;
}
如
void fun(int a,int b)
{
printf(“a+b=%d”,a+b);
}
当然类型标识符也可以是int或float或char,但那是有返回值函数的范畴,我们在下一章节再详细讲解。
二、 形式参数和实际参数
在上面的案例大家也看到了在函数头后的括号里有内容,我们把它称为参数,在使用函数时,经常会用到形式参数和实际参数。两者都叫参数,那么二者有什么关系?二者之间的区别是什么?两种参数各自又起到什么作用?
- 1. 通过名称理解
形式参数:按照名称进行理解就是形式上存在的参数。
实际参数:按照名称进行理解就是实际存在的参数。
- 2. 通过作用理解
形式参数:在定义函数时,函数名后面括号中的变量名称为“形式参数”。在函数调用之前传递给函数的值将被复制到这些形式参数中。
实际参数:在调用一个函数时,也就是真正使用一个函数时,函数名后面括号中的参数为“实际参数”。函数的调用者提供给函数的参数叫实际参数。实际参数是表达式计算的结果,并且被复制给函数的形式参数。
|
void fun(int num) { ----- } |
如定义或声明函数,此时函数参数num为形式参数,简称形参 |
|
void main() { int number; fun(number); fun(99); } |
调用函数,此时的函数参数中的99或number为实际参数,简称实参 |
- 3. 通过一个比喻来理解形式参数和实际参数
母亲拿来了一袋牛奶,将牛奶倒入一个空奶瓶中,然后喂宝宝喝牛奶。函数的作用就相当于宝宝用奶瓶喝牛奶这个动作,实参相当于母亲拿来的一袋牛奶,而空的奶瓶相当于形参。牛奶放入奶瓶这个动作相当于将实参传递给形参,使用灌好牛奶的奶瓶就相当于函数使用参数进行操作的过程。
下面通过一个实例对形式参数和实际参数进行实际的讲解。
例:形式参数与实际参数的比喻实现
实例中将上面的比喻进行了实际的模拟,希望读者可以一边实际动手操作,一边通过上面的比喻对形式参数和实际参数加深理解,更好地掌握知识点。
#include "stdio.h"
void drinkmilk(char cbottle); /*声明函数*/
void main()
{
char c;
printf("Mother wanna give the baby:");
c=getchar();
drinkmilk(c);
}
void drinkmilk(char cbottle)
{
printf("Theb Baby drink the %c\n",cbottle);
}
函数的声明方式
Void drinkmilk(char cbottle);分号不能少放在函数的开头或函数被调用之前。
也可以转换为
Void drinkmilk(char );
以上程序的执行结果如下:

三、 数组作函数参数
当数组作为函数的实参时,只传递数组的地址,而不是整个数组赋值到函数中。当用数组名作为实参调用函数时,指向该数组的第一个元素的指针就被传递到函数中。
声明函数参数时必须具有相同的类型,根据这一点,下面将对使用数组作为函数参数的各种情况进行详细的讲解。
- 1. 数组元素作为函数参数
由于实参可以是表达式形式,数组元素可以是表达式的组成部分,因此数组元素可以作为函数的实参,与用变量作为函数实参一样,是单向传递。
例:数组元素作为函数参数
#include "stdio.h"
void fun(int a); /* 声明函数*/
void main()
{
int score[]; /* 定义一个整型的数组*/
int i; /* 定义整型变量,用于循环*/
for(i=;i<;i++)/* 进行赋值循环*/
{
score[i]=i; /*为数组中的元素进行赋值操作*/
}
for(i=;i<;i++) /* 循环操作*/
{
fun(score[i]); /*执行输出函数操作*/
}
}
void fun(int a) /*函数定义*/
{
printf("show the member is %d \n",a); /*输出数据*/
}
运行结果:

- 2. 数组名作为函数参数
可以用数组名作为函数参数,此时实参与形参都使用数组名
例:
#include "stdio.h"
void evaluate(int Aarray[]); /* 声明赋值函数*/
void display(int Aarray[]); /*声明显示函数*/
void main()
{
int iarray[]; /* 定义一个具有10个元素的整型数组*/
evaluate(iarray); /*调用函数进行赋值,将数组名作为参数*/
display(iarray); /*调用函数进行输出,将数组名作为参数*/
}
void display(int Aarray[])
{
int i;
for(i=;i<;i++)
{ /*在函数中执行输出操作*/
printf("the member number is %d\n",Aarray[i]);
}
}
void evaluate(int Aarray[])
{
int i;
for(i=;i<;i++)
{ /*在函数中执行赋值操作*/
Aarray[i]=i;
}
}
结果如下:

注:数组作为参数在进行传递时,是以数组名作为参数进行传递的。
- 3. 可变长度数组作为函数参数
可以将函数的参数声明成长度可变的数组,在此基础上利用上面的程序进行修改。声明方式的代码为:
void fun(int array[]);
int iarray[];
fun(iarray);
从上面的代码中可以看到,在定义和声明一个函数时将数组作为函数参数,并且没有指明数组此时的大小,这样就将函数参数声明为数组长度可变的数组。
例 :
#include "stdio.h"
void evaluate(int Aarray[]); /* 声明赋值函数*/
void display(int Aarray[]); /*声明显示函数*/
void main()
{
int iarray[]; /* 定义一个具有10个元素的整型数组*/
evaluate(iarray); /*调用函数进行赋值,将数组名作为参数*/
display(iarray); /*调用函数进行输出,将数组名作为参数*/
}
void display(int Aarray[])
{
int i;
for(i=;i<;i++)
{ /*在函数中执行输出操作*/
printf("the member number is %d\n",Aarray[i]);
}
}
void evaluate(int Aarray[])
{
int i;
for(i=;i<;i++)
{ /*在函数中执行赋值操作*/
Aarray[i]=i;
}
}
结果如下:

五子棋游戏代码
#include "stdio.h"
#include "stdlib.h"
#include "conio.h"
#include "string.h"
#include "windows.h" //控制dos界面
#define MAXIMUS 15 //定义棋盘大小 int p[MAXIMUS][MAXIMUS];//存储对局信息
char buff[MAXIMUS * + ][MAXIMUS * + ]; //输出缓冲器
int Cx, Cy; //当前光标位置
int Now;//当前走子的玩家,1代表黑,2代表白
int wl, wp; //当前写入缓冲器的列数和行数位置
char* showText;//在棋盘中央显示的文字信息
int count;//回合数 char* Copy ( char* strDest, const char* strSrc ) //修改过的字符串复制函数,会忽略末端的\0
{
char* strDestCopy = strDest; while ( *strSrc != '\0' )
{
*strDest++ = *strSrc++;
} return strDestCopy;
}
void Initialize() //初始化一个对局函数
{
int i, j; //循环变量
showText = ""; //重置显示信息
count = ; //回合数归零 for ( i = ; i < MAXIMUS; i++ ) //重置对局数据
{
for ( j = ; j < MAXIMUS; j++ )
{
p[i][j] = ;
}
} Cx = Cy = MAXIMUS / ; //重置光标到中央
Now = ; //重置当前为黑方
}
char* getStyle ( int i, int j ) //获得棋盘中指定坐标交点位置的字符,通过制表符拼成棋盘
{
if ( p[i][j] == ) //1为黑子
return "●";
else if ( p[i][j] == ) //2为白子
return "○";
else if ( i == && j == ) //以下为边缘棋盘样式
return "┏";
else if ( i == MAXIMUS - && j == )
return "┓";
else if ( i == MAXIMUS - && j == MAXIMUS - )
return "┛";
else if ( i == && j == MAXIMUS - )
return "┗";
else if ( i == )
return "┠";
else if ( i == MAXIMUS - )
return "┨";
else if ( j == )
return "┯";
else if ( j == MAXIMUS - )
return "┷"; return "┼";//中间的空位
}
char* getCurse ( int i, int j ) //获得指定坐标交点位置左上格的样式,通过制表符来模拟光标的显示
{
if ( i == Cx )
{
if ( j == Cy )
return "┏";
else if ( j == Cy + )
return "┗";
}
else if ( i == Cx + )
{
if ( j == Cy )
return "┓";
else if ( j == Cy + )
return "┛";
} return " ";//如果不在光标附近则为空
}
void write ( char* c ) //向缓冲器写入字符串
{
Copy ( buff[wl] + wp, c );
wp += strlen ( c );
}
void ln() //缓冲器写入位置提行
{
wl += ;
wp = ;
}
void Display() //将缓冲器内容输出到屏幕
{
int i, l = strlen ( showText ); //循环变量,中间文字信息的长度
int Offset = MAXIMUS * + - l / ; //算出中间文字信息居中显示所在的横坐标位置 if ( Offset % == ) //如果位置为奇数,则移动到偶数,避免混乱
{
Offset--;
} Copy ( buff[MAXIMUS] + Offset, showText ); //讲中间文字信息复制到缓冲器 if ( l % == ) //如果中间文字长度为半角奇数,则补上空格,避免混乱
{
* ( buff[MAXIMUS] + Offset + l ) = 0x20;
} system ( "cls" ); //清理屏幕,准备写入 for ( i = ; i < MAXIMUS * + ; i++ ) //循环写入每一行
{
printf ( "%s", buff[i] ); if ( i < MAXIMUS * ) //写入完每一行需要换行
printf ( "\n" );
}
}
void Print() //将整个棋盘算出并储存到缓冲器,然后调用Display函数显示出来
{
int i, j; //循环变量
wl = ;
wp = ; for ( j = ; j <= MAXIMUS; j++ ) //写入出交点左上角的字符,因为需要打印棋盘右下角,所以很以横纵各多一次循环
{
for ( i = ; i <= MAXIMUS; i++ )
{
write ( getCurse ( i, j ) ); //写入左上角字符 if ( j == || j == MAXIMUS ) //如果是棋上下盘边缘则没有连接的竖线,用空格填充位置
{
if ( i != MAXIMUS )
write ( " " );
}
else//如果在棋盘中间则用竖线承接上下
{
if ( i == || i == MAXIMUS - ) //左右边缘的竖线更粗
write ( "┃" );
else if ( i != MAXIMUS ) //中间的竖线
write ( "│" );
}
} if ( j == MAXIMUS ) //如果是最后一次循环,则只需要处理边侧字符,交点要少一排
{
break;
} ln();//提行开始打印交点内容
write ( " " ); //用空位补齐位置 for ( i = ; i < MAXIMUS; i++ ) //按横坐标循环正常的次数
{
write ( getStyle ( i, j ) ); //写入交点字符 if ( i != MAXIMUS - ) //如果不在最右侧则补充一个横线承接左右
{
if ( j == || j == MAXIMUS - )
{
write ( "━" ); //上下边缘的横线更粗
}
else
{
write ( "—" ); //中间的横线
}
}
} ln();//写完一行后提行
} Display();//将缓冲器内容输出到屏幕
}
int Put() //在当前光标位置走子,如果非空,则返回0表示失败
{
if ( p[Cx][Cy] == )
{
p[Cx][Cy] = Now; //改变该位置数据
return ;//返回1表示成功
}
else
{
return ;
}
}
int Check() //胜负检查,即判断当前走子位置有没有造成五连珠的情况
{
int w = , x = , y = , z = , i; //累计横竖正斜反邪四个方向的连续相同棋子数目 for ( i = ; i < ; i++ ) if ( Cy + i < MAXIMUS && p[Cx][Cy + i] == Now ) w++;
else break;//向下检查 for ( i = ; i < ; i++ ) if ( Cy - i > && p[Cx][Cy - i] == Now ) w++;
else break;//向上检查 if ( w >= ) return Now; //若果达到5个则判断当前走子玩家为赢家 for ( i = ; i < ; i++ ) if ( Cx + i < MAXIMUS && p[Cx + i][Cy] == Now ) x++;
else break;//向右检查 for ( i = ; i < ; i++ ) if ( Cx - i > && p[Cx - i][Cy] == Now ) x++;
else break;//向左检查 if ( x >= ) return Now; //若果达到5个则判断当前走子玩家为赢家 for ( i = ; i < ; i++ ) if ( Cx + i < MAXIMUS && Cy + i < MAXIMUS && p[Cx + i][Cy + i] == Now ) y++;
else break;//向右下检查 for ( i = ; i < ; i++ ) if ( Cx - i > && Cy - i > && p[Cx - i][Cy - i] == Now ) y++;
else break;//向左上检查 if ( y >= ) return Now; //若果达到5个则判断当前走子玩家为赢家 for ( i = ; i < ; i++ ) if ( Cx + i < MAXIMUS && Cy - i > && p[Cx + i][Cy - i] == Now ) z++;
else break;//向右上检查 for ( i = ; i < ; i++ ) if ( Cx - i > && Cy + i < MAXIMUS && p[Cx - i][Cy + i] == Now ) z++;
else break;//向左下检查 if ( z >= ) return Now; //若果达到5个则判断当前走子玩家为赢家 return ;//若没有检查到五连珠,则返回0表示还没有玩家达成胜利
} int RunGame() //进行整个对局,返回赢家信息(虽然有用上)
{
int input;//输入变量
int victor;//赢家信息
Initialize();//初始化对局 while ( ) //开始无限回合的死循环,直到出现胜利跳出
{
Print();//打印棋盘
input = getch(); //等待键盘按下一个字符 if ( input == ) //如果是ESC则退出程序
{
exit ( );
}
else if ( input == 0x20 ) //如果是空格则开始走子
{
if ( Put() ) //如果走子成功则判断胜负
{
victor = Check();
Now = - Now; //轮换当前走子玩家
count++; if ( victor == ) //如果黑方达到胜利,显示提示文字并等待一次按键,返回胜利信息
{
showText = "黑方获得了胜利!";
Print(); if ( getch() == 0xE0 )
{
getch();
} return Now;
}
else if ( victor == ) //如果白方达到胜利,显示提示文字并等待一次按键,返回胜利信息
{
showText = "白方获得了胜利!";
Display(); if ( getch() == 0xE0 )
{
getch();
} return Now;
}
else if ( count == MAXIMUS * MAXIMUS ) //如果回合数达到了棋盘总量,即棋盘充满,即为平局
{
showText = "平局!";
Display(); if ( getch() == 0xE0 )
{
getch();
} return ;
}
}
}
else if ( input == 0xE0 ) //如果按下的是方向键,会填充两次输入,第一次为0xE0表示按下的是控制键
{
input = getch(); //获得第二次输入信息 switch ( input ) //判断方向键方向并移动光标位置
{
case 0x4B://
Cx--;
break; case 0x48:
Cy--;
break; case 0x4D:
Cx++;
break; case 0x50:
Cy++;
break;
} if ( Cx < ) Cx = MAXIMUS - ; //如果光标位置越界则移动到对侧 if ( Cy < ) Cy = MAXIMUS - ; if ( Cx > MAXIMUS - ) Cx = ; if ( Cy > MAXIMUS - ) Cy = ;
}
}
}
int main() //主函数
{
system ( "title 简易五子棋 ——Etsnarl制作" ); //设置标题
system ( "mode con cols=63 lines=32" ); //设置窗口大小
system ( "color E0" ); //设置颜色 while ( ) //循环执行游戏
{
RunGame();
}
}
c语言进阶3-有参函数的更多相关文章
- 【R笔记】R语言进阶之4:数据整形(reshape)
R语言进阶之4:数据整形(reshape) 2013-05-31 10:15 xxx 网易博客 字号:T | T 从不同途径得到的数据的组织方式是多种多样的,很多数据都要经过整理才能进行有效的分析,数 ...
- 《C语言进阶剖析》课程目录
<C语言进阶剖析>学习笔记 本文总结自狄泰软件学院唐佐林老师的<C语言 ...
- 彩虹女神跃长空,Go语言进阶之Go语言高性能Web框架Iris项目实战-项目入口与路由EP01
书接上回,我们已经安装好Iris框架,并且构建好了Iris项目,同时配置了fresh自动监控项目的实时编译,万事俱备,只欠东风,彩虹女神蓄势待发.现在我们来看看Iris的基础功能,如何编写项目入口文件 ...
- R语言进阶
一.初学入门:<R in Action><The Art of_R Programming>入门者可首选两本,前者从统计角度入手,分高中低三部分由浅入深的讲解了如何用R来实现统 ...
- C 语言 进阶
清单狂魔,只挖坑不填坑.. 前言 最近经常被询问 C 语言 相关的问题,突然便也觉得需要思考一下 C 语言的进阶了. 我用 C 语言写过的最大的一个项目,也只是那个贪吃蛇,后来就断断续续地用 Pyth ...
- c语言进阶2-变量的作用域与无参函数
一. 什么是函数 函数是具有特定功能的模块.可以说一个完整的程序其实是由多个函数共同完成的.C语言的全部工作都是由程式各样的函数完成的,所以也把C语言称为函数式语言.使用模块化设计可能 使 ...
- 编程C语言进阶篇——自定义数据类型:共同体
什么是"自定义数据类型"?顾名思义,就是用户可以随时在程序中自行定义新的数据类型.自定义数据类型时需要设置数据类型的名称及其成员.数据类型成员各属性的设置方法等同于变量设置时相应属 ...
- 编程C语言进阶篇——自定义数据类型:结构体
一.结构体 定义方法: 结构名 变量名 特点: 两个同类型的结构变量可以相互赋值,但是结构变量之间不能使用"<","=="等运算符,如果使用则需要对运算符 ...
- 苹果新的编程语言 Swift 语言进阶(六)--函数和闭包
一 .函数 1.1. 函数的定义和调用 函数的定义以funckeyword作为前缀,接着是函数名字,接着跟着一个能够带有參数.也能够不带參数的圆括号.接着用-> 指示函数的返回类型. 函数运行体 ...
- 苹果新的编程语言 Swift 语言进阶(五)--控制流
Swift 语言支持C语言全部的控制语句.包含for 和while循环语句,if和switch条件语句,以及break和continue控制语句等. Swift 语言除了支持以上语句,还添加了一个f ...
随机推荐
- linux log rotate
今天老大提醒我产线kafka自身的log文件积累了好几个月了,我才发现原来kafka的log4j并删除old log. 第一反应是采用linux自带的logrotate功能,在/etc/logrota ...
- [java代码库]-简易计算器(第二种)
[java代码库]-简易计算器(第二种) 第二种方案:在程序中不使用if/switch……case等语句,完成计算器功能. <html> <head> <title> ...
- Windows10 下运行Linux子系统
关于Windows10 下运行Linux子系统: Windows10内置Linux子系统初体验:http://www.jianshu.com/p/bc38ed12da1d Win10运行Ubuntu版 ...
- Python自动化测试 (八)unittest 单元测试
任何一种编程语言, 都会有单元测试框架, 本文介绍Python 自带的unittest模块 # -* - coding: UTF- -* - class Myclass: def sum(self,x ...
- 一步步教你怎么用python写贪吃蛇游戏
目录 0 引言 1 环境 2 需求分析 3 代码实现 4 后记 0 引言 前几天,星球有人提到贪吃蛇,一下子就勾起了我的兴趣,毕竟在那个Nokia称霸的年代,这款游戏可是经典中的经典啊!而用Pytho ...
- Mac上使用brew安装nvm来支持多版本的Nodejs
brew方式 如果机器没有安装过node,那么首先brew install nvm安装nvm. 其次需要在shell的配置文件(~/.bashrc, ~/.profile, or ~/.zshrc)中 ...
- Spring 之Aop实现日志记录
Aop实现见代码,简单demo实现 package com.idcos.automate.config; import com.idcos.automate.dal.auto.dao.dcos.Dco ...
- 六种 主流ETL 工具的比较(DataPipeline,Kettle,Talend,Informatica,Datax ,Oracle Goldengate)
六种 主流ETL 工具的比较(DataPipeline,Kettle,Talend,Informatica,Datax ,Oracle Goldengate) 比较维度\产品 DataPipeline ...
- Sole跟Tomcat的整合 Solr4.4 + Tomcat
1, 下载Solr4.4 Tomcat7 2, 拷贝 solr-4.4.0\dist\solr4.4.war => tomcat\webapps 下面 重命名为solr.war 3, 启动to ...
- cat more less 命令混用
在Linux系统中有三种命令可以用来查阅全部的文件,分别是cat.more和less命令.它们查阅文件的使用方法也比较简单都是 命令 文件名 ,但是三者又有着区别. 1.cat命令可以一次显示整个文件 ...