zoukankan      html  css  js  c++  java
  • 2017.11.13 在C语言中是否能用函数实现模块化程序设计

    第七章 用函数实现模块化程序设计

    (1)为什么要用函数?
    @function既是函数也是功能。函数就是用来完成一定功能的的(函数就是功能),函数名就是给这个功能起一个名字,一个C程序可由一个主函数和若干个其他函数构成。

    所有的函数都是平行的,即在函数定义时是分别进行,是互相独立。

    例题://输出以下结果,用过调用函数实现 
    /*
    ***********
    How are you!
    *********** 
    */
     
    #include<stdio.h>
    int main()
    {
    	void print_star();		//声明void print_star()函数 
    	void print_message();	//声明void print_message函数 
    	print_star();			//调用print_star函数 
    	print_message();		//调用print_message函数 
    	print_star();		   //用print_star函数 
    	return 0;
     } 
     void print_star()      //定义void print_star函数,这里后面不能有";"符号 
     {
     	printf("****************
    ");
     }
     void print_message()	//定义void print_message函数		
     {
     	printf("How are you!
    ");   //输出一行文字信息 
     }
    

    (2)怎样定义函数
    @在程序中若用到数学函数【如:sqrt、fabs、sin cos等】必须在文本开头写上(#include<math.h>)

    函数的定义包括以下几个内容
    1.指定函数的名字,以便以后按名调用。
    2.指定函数类型,即函数值返回类型。
    3.指定函数的名字和类型,在调用时向他们传递数据。
    4.指定函数应当完成什么操作,也就是函数应该做什么,就是函数的功能。
    

    定义函数的方法:

    1.定义无参函数:
    一般形式为:   
    	类型名  函数名()         类型名   函数名(void)
    	{                       {
    		函数体       或            函数体     //包括声明部分和语句部分
    	}                       }
    函数名后面括号内的void表示“空”即函数没有参数
    
    
    2.定义有参函数:
    一般形式为:								例如:定义max函数
    类型名     函数名(形式参数表列)				int max(int x,int y);
    {											{   int z;
    	函数体                                      z=x>y?x:y;
    												return (z);
    }                                           }
    
    
    3.定义空函数(先占用一个位置,后面在编写好的程序放上去)
    一般形式为:
    类型名  函数名()
    {}
     例如:  void dummy()
     {}
     
     【merge():合并 ; matproduct():矩阵相乘 ;  concatenate():字符串连接; shell(): 希尔排序】
    

    (3)调用函数

    1.一般形式为:
         函数名(实参表列)//若是无参函数,则实参表列可以没有,但是括号不能省
            例如:  print_star();        //调用无参函数
                   c=max(a,b)//调用有参函数
    
    2.    3种函数调用方式
    	函数调用语句 、函数表达式、函数参数	
    
    3.函数调用时的数据传递
    	形式参数和实际参数、实参和形参间的数据传递。
    	
    4.函数的调用过程
    5函数的返回值
       它是通过return语句获得
       函数值的类型
       再定义函数时指定的函数类型一般应该和return语句中的表达式类型一致
       **(如果函数值的类型和return语句中表达式的值不一样,以函数类型为准)
    

    (4)对被调用函数的声明和函数原型:
    *函数的首行(即函数的首部)成为函数的原型,使用函数原型作声明是C语言的一个重要特点。
    *函数的定义是指对函数功能的确立,包括函数名,函数值类型、形参及其类型以及函数体等,他是一个完整的、独立的函数单位。

    函数声明有两种:
    1.函数类型   函数名(参数类型1   参数名1,参数类型2  参数名2.....);
    2.函数类型   函数名(参数类型1,参数类型2,.....);
    

    (5)函数的嵌套使用:
    @C语言的函数定义是相互平行的、独立的,也就是说,在定义函数时,一个函数不能再定义另一个函数就是不能嵌套定义,但可以嵌套使用。

    例题1://输入4个整数,找出其中最大的数。用函数嵌套调用来处理。
    //方法一:
    #include<stdio.h>
    int main()
    {
    	int max(int r,int x,int y,int z);
    	int a,b,c,d,e;
    	printf("请输入4个整数:");
    	scanf("%d,%d,%d,%d",&a,&b,&c,&d);
    	e=max(a,b,c,d);
    	printf("最大的数为:%d
    ",e);
    	return 0;
     } 
     int max(int r,int x,int y,int z)
     {
     	int m,n,l;
     		m=r>x?r:x;
    	    n=m>y?m:y;
    	    l=n>z?n:z;
    		return (l);
     }
    
    //方法二:
    /* 
    #include<stdio.h>
    int main()
    {
    	int max4(int a,int b,int c,int d);//对max4函数的声明 
    	int a;
    	int b;
    	int c;
    	int d;
    	int max;
    	
    	printf("请输入4个整数:");
    	scanf("%d,%d,%d,%d",&a,&b,&c,&d);
    	
    	max=max4(a,b,c,d);       //调用max4函数,得到4个数中的最大者 
    	printf("最大的数为:%d
    ",max);
    	return 0;
     }
      
     int max4(int a,int b,int c,int d)		//定义max4函数 
     {
     	int max2(int a,int b);		//对max2函数的声明 
     	int m;
    	m=max2(a,b);				//调用max2函数,得到a和b两个数中的大者,放在m中 
    	m=max2(m,c);				//调用max2函数,得到a,b,c三个数中的大者 
    	m=max2(m,d);
    	return (m);
     }
     int max2(int a,int b)
     {
     	if(a>=b)
     	return a;
     	else
     	return b;
     }
     */ 
    

    (6)函数的递归调用
    在调用一个函数的过程中又出现直接或间接地调用该函数本身,称为函数的递归调用。运用递归调用必须有一个递归结束条件

    例如1:直接调用本函数举例
    int f(int x)
    {
    	int y,z;
    	z=f(y);//在执行f函数的过程中又要调用f函数
    	return (2*z);
    }
    

    例题2:/*有五个学生坐在一起,
    问第五个学生多少岁,他说比第四个学生大两岁。
    问第四个学生岁数,他说比第三个大两岁。
    问第三个学生,又说比第二个大两岁。
    问第二个学生,他说比第一个大两岁。
    最后问第一个学生,他说他十岁。
    请问第五个学生多少岁? 
    */
    
    //递归过程 
    /*
    int age(int n)
    {
    	int c;
    	if(n==1)
    		c=10;
    	else
    		c=age(n-1)+2;
    	return(c);
    }
    */
    
    #include <stdio.h>
    int main()
    {
    	int age(int n);
    	printf("NO.5,age:%d
    ",age(5));  //输出 
    	return 0;
    }
    
    int age(int n)		//定义递归函数 
    {
    	int c;
    	if(n==1)		//n=1时,年龄为10 
    		c=10;
    	else
    		c=age(n-1)+2;		//n!=1时,年龄是前一个学生的年龄加2 
    	return (c);
    }
    //递归调用4次,
    
    例题3://利用递归算法求n!
    
    #include<stdio.h> 
    int main()
    {
    	int fac(int n); //fac 函数声明 
    	int n;
    	int y;
    	printf("input integer number:");  //输入要阶乘的数 
    	scanf("%d",&n);
    	y=fac(n);
    	printf("%d!=%d
    ",n,y);
    	return 0; 
    }
    int fac(int n)			//定义fac函数 
    {
    	int f;
    	if(n<0)       //n不能小于0 
    	  printf("n<0,data error!");
    	else if(n==0||n==1) //n=0或者1时, (n!)=1 
    	   f=1;
    	else 
    	f=fac(n-1)*n;
    	return (f);
    }
    

    (7)数组作为函数参数
    调用有参函数,需要提供实参。可以是常量、变量或着表达式。数组元素也可以作为函数实参。
    【数据传递方向是从实参到形参,单向传递。】

    1.数组元素作为函数实参,不能作为形参。
    例题1://输入 10个数,要求输出其中值最大的元素和该数是第几个数。
     
    #include<stdio.h> 
    int main()
    {
    	int max(int x,int y);
    	int a[10];
    	int m;
    	int n;
    	int i;
    	printf("enter 10 integer numbers:");
    	for(i=0;i<10;i++)
    	scanf("%d",&a[i]);
    	printf("
    ");
    	for(i=1,m=a[0],n=0;i<10;i++)
    	{
    		if(max(m,a[i])>m)
    		{
    			m=max(m,a[i]);
    			n=i;
    		}
    	}
    	printf("The largest number is %d
     it is the %dth number.
    ",m,n+1);
    }
    int max(int x,int y)
    {
    	return (x>y?x:y);
    }
    
    
    2.数组名作函数参数
    【用数组元素做实参时,向形参变量传递是数组元素的值,而用数组名作函数实参时,向形参传递的是数组首元素的地址】
    例题2://有一个一维数组score,内放10个学生成绩,求平均成绩
     
    #include<stdio.h>
    int main()
    {
    	float average(float array[10]);   //函数声明 
    	float score[10];
    	float aver;
    	int i;
    	printf("enter 10 scores:
    ");
    	for(i=0;i<10;i++)
    		scanf("%f",&score[i]);
    	printf("
    ");
    	aver=average(score);				//调用average函数 
    	printf("average score is %5.2f
    ",aver);
    	return 0;
     } 
     float average(float array[10])   //定义average函数 
     {
     	int i;
     	float aver,sum=array[0];
     	for(i=1;i<10;i++)
     		sum=sum+array[i];   //累加学生成绩 
     	aver=sum/10;
     	return(aver);
      } 
    
    例题3://有两个班级,分别有35名和30名学生,调用一个average函数,分别求这两个班级的学生的平均成绩。
    //为了简化,分别设着两个班的人数为5和10 
    
    #include<stdio.h>
    int main()
    {
    	float average(float array[],int n);
    	float score1[5]={98.5,97,91,60,55};
    	float score2[10]={67,88,99,69,77,89,76,54,60,99};
    	printf("The average of class A is %6.2f
    ",average(score1,5));
    	printf("The average of class B is %6.2f
    ",average(score2,10));//用数组名score1和5做实参
    	return 0; 
     }
     
     float average(float array[],int n)
     {
     	int i;
     	float aver,sum=array[0];
     	for(i=1;i<n;i++)
     	sum=sum+array[i];
     	aver=sum/n;
     	return(aver);
      }
       
    3.多维数组名做函数参数
    在被调用函数中对形参数组定义时可以制定每一维大小,也可以省略第一维的大小说明
    例题://有一个3×4的矩阵,求所有元素中的最大值
     
    #include<stdio.h>
    int main()
    {
    	int max_value(int array[][4]);	//函数的声明 
    	int a[3][4]={{1,3,5,7},{2,4,6,8},{15,17,34,12}}; //对数组元素赋初值 
    	printf("Max value is %d
    ",max_value(a));// max_value(a)为函数调用 
    	return 0;
     } 
     int max_value(int array[][4])  //函数定义 
     {
     	int i;
     	int j;
     	int max;
     	max=array[0][0];
     	for(i=0;i<3;i++)
     	  for(j=0;j<4;j++)
     	  if(array[i][j]>max)
     	  	max=array[i][j];
     	return (max);
     }
    

    (8)局部变量和全局变量
    从变量的作用域来(从空间)看可分为全局变量和局部变量。

    局部变量:在复合语句内定义的,复和语句不能在外面用的就是局部变量。形式参数也是局部变量。
    全局变量:在函数之外定义的外部变量是全局变量。

    (9)变量的存储方式和生存期
    变量的存储方式分为两类:

    1.动态存储方式:是在程序运行期间需要进行动态的分配存储空间的方式【形式参数、自动变量、现场保护和返回地址】
    2.静态存储方式:在程序运行期间有系统分配固定的存储空间的方式。【全局变量全部放在这里】
    
    3.存储空间又分为三类:程序区、静态存储区和动态存储区【存放数据】。
    

    *变量的存在时间称之为生存期。
    *【每一个C语言都包含的属性有:数据的类型、数据存储类别。】

    1.存储类别指的是数据在内存中存储的方式(如静态存储方式和动态存储方式)
    2.c的存储类别分为四种:自动的auto 、静态的static、寄存器的 register、外部的 extern
    
    局部变量的存储类别(auto static)
    1.在调用该函数时,系统会给这些变量分配存储空间,在函数调用结束时会自动释放这些存储空间,这就是自动变量。auto可以隐藏不写;它存储在动态存储区中
    2.局部变量的值在函数调用结束后不消失而继续保留原值,及其占用的存储单元不释放,就是静态存储变量。
    3.将局部变量的值存储在CPU的寄存器中,需要用时直接从寄存器中取出参加运算
    
  • 相关阅读:
    MySQL与SQLServer的update left join语法区别
    request获取真实IP
    request获取前台表单
    ehcache memcache redis 三大缓存
    StraUML简单教程
    JSP模板
    判断是否为同一天
    eneityManager的merge、Flush、Refresh方法
    javascript
    javax.naming.NameNotFoundException:Name[ XXX] is not bound in this context.
  • 原文地址:https://www.cnblogs.com/qichunlin/p/7828703.html
Copyright © 2011-2022 走看看