欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言函數(shù)超詳細講解上篇

 更新時間:2022年04月18日 09:01:42   作者:初學C語言者  
函數(shù)是一組一起執(zhí)行一個任務的語句。每個?C?程序都至少有一個函數(shù),即主函數(shù)?main()?,所有簡單的程序都可以定義其他額外的函數(shù),函數(shù)我們分兩篇來講解,接下來開始第一篇

前言

本文主要學習函數(shù)的相關(guān)內(nèi)容。

1、函數(shù)是什么?

維基百科中對函數(shù)的定義:子程序

  • 在計算機科學中,子程序(英語:Subroutine, procedure, function, routine, method,subprogram, callable unit),是一個大型程序中的某部分代碼, 由一個或多個語句塊組成。它負責完成某項特定任務,而且相較于其他代 碼,具備相對的獨立性。
  • 一般會有輸入?yún)?shù)并有返回值,提供對過程的封裝和細節(jié)的隱藏。這些代碼通常被集成為軟件庫。

2、C語言中函數(shù)的分類

  • 庫函數(shù)
  • 自定義函數(shù)

2.1 庫函數(shù)

  • 在學習C語言編程時,總是在一個代碼編寫完成之后,想把這個結(jié)果打印到屏幕上看看。這時使用一個功能:將信息按照一定的格式打印到屏幕上(printf)。
  • 在編程的過程中我們會頻繁的做一些字符串的拷貝工作(strcpy)
  • 在編程是我們也計算,總是會計算n的k次方這樣的運算(pow)

上面的函數(shù)不用自己編寫,直接可以調(diào)用。為了支持可移植性和提高程序的效率,所以C語言的基礎(chǔ)庫中提供了一系列類似的庫函數(shù),方便程序員進行軟件開發(fā)。

C語言常用的庫函數(shù)都有:

  • IO函數(shù)
  • 字符串操作函數(shù)
  • 字符操作函數(shù)
  • 內(nèi)存操作函數(shù)
  • 時間/日期函數(shù)
  • 數(shù)學函數(shù)
  • 其他庫函數(shù)

使用庫函數(shù),必須包含 #include 對應的頭文件。

2.1.1 如何學會使用庫函數(shù)

推薦查詢工具官網(wǎng):

MSDN(Microsoft Developer Network)

www.cplusplus.com

http://en.cppreference.com(英文版)

http://zh.cppreference.com(中文版)

2.1.2 自定義函數(shù)

自定義函數(shù)和庫函數(shù)一樣,有函數(shù)名,返回值類型和函數(shù)參數(shù)。這些都是我們自己來設(shè)計,函數(shù)的組成:

ret_type fun_name(para1, * )
{
	statement;//語句項
}
//ret_type 返回類型
//fun_name 函數(shù)名
//para1 函數(shù)參數(shù)
//舉例:寫一個函數(shù)可以找出兩個整數(shù)中的最大值
//get_max函數(shù)的設(shè)計
int get_max(int x, int y)
{
	return (x>y)?(x):(y);
}
int main()
{
	int num1 = 10;
	int num2 = 20;
	int max = get_max(num1, num2);
	printf("max = %d\n", max);
	return 0;
}

3、函數(shù)的參數(shù)

3.1 實際參數(shù)(實參)

  • 真實傳給函數(shù)的參數(shù),叫實參
  • 實參可以是:常量、變量、表達式、函數(shù)等
  • 無論實參是何種類型的量,在進行函數(shù)調(diào)用時,它們都必須有確定的值,以便把這些值傳送給形參

3.2 形式參數(shù)(形參)

  • 形式參數(shù)是指函數(shù)名后括號中的變量,因為形式參數(shù)只有在函數(shù)被調(diào)用的過程中才實例化(分配內(nèi)存單元),所以叫形式參數(shù)
  • 形式參數(shù)當函數(shù)調(diào)用完成之后就自動銷毀了。因此形式參數(shù)只在函數(shù)中有效
//舉例2:寫一個函數(shù)可以交換兩個整形變量的內(nèi)容
//實現(xiàn)成函數(shù),但是不能完成任務
int exchange1(int x, int y)
{//當實參傳給形參時候,形參是實參的一份臨時拷貝,
	//對形參的修改不會影響實參
	int temp = x;
	x = y;
	y = temp;
}
//正確的版本
int exchange2(int* pa, int* pb)//定義指針,接收地址
{
	int temp = *pa;
	*pa = *pb;
	*pb = temp;
}
int main()
{
	int a = 3;
	int b = 5;
	exchange1(a, b);//傳參是值
	printf("exchange1::a = %d b = %d\n", a, b);//交換前

	exchange2(&a, &b);//傳參是地址
	printf("exchange2::a = %d b = %d\n", a, b);//交換后的
	//傳入地址,自定義的形參和實參聯(lián)系更加緊密,能改變地址存儲的數(shù)值
	//此時,形參的地址與實參的地址是一樣的
	return 0;
}

上面代碼中, exchange1和 exchange2函數(shù)中的參數(shù) x,y,pa,pb 都是形式參數(shù)。在main函數(shù)中傳給 exchange1的 a ,b 和傳給 exchange2函數(shù)的 &a ,&b是實際參數(shù)。

運行結(jié)果如下所示,exchange1并沒有起到預想的交換數(shù)值的作用,exchange2可以。

在這里插入圖片描述

在這里插入圖片描述

通過監(jiān)視變量發(fā)現(xiàn):

名稱意義
a5數(shù)值為5
&a0x113f8e4數(shù)值a的地址
x5形參x數(shù)值為5
&x0x113f800形參x的地址
pa0x113f8e4形參pa數(shù)值為a的地址
  • 變量a在內(nèi)存中開辟了空間,地址是0x113f8e4
  • 函數(shù)exchange1將實參a傳遞給形參x,x的數(shù)值也是5,但此時形參x另外重新再內(nèi)存中開辟了空間,地址 0x113f800,形參和實參的地址是不一樣的
  • 函數(shù)exchange2將實參a傳遞給形參pa,pa的值是變量a的地址0x113f8e4,這個地址里面存放變量a的數(shù)值5。

在這里插入圖片描述

這里可以看到 exchange1函數(shù)在調(diào)用的時候, x , y 擁有自己的空間,同時擁有了和實參一模一樣的內(nèi)容。所以我們可以簡單的認為:

形參實例化之后其實相當于實參的一份臨時拷貝。形參x、y的值發(fā)生交換,但是不影響實參a、b的值。一般性的只是使用數(shù)值大小,利用形參傳值就可以了,傳值表明,形參和實參的關(guān)系膚淺,僅限于表面數(shù)值的拷貝。

如果需要對主函數(shù)的實參值進行操作,比如交換,此時形參需傳地址,功能更為強大。傳地址表明,形參和實參的關(guān)系更深一步,直接可以通過地址修改實參的數(shù)值。

4、函數(shù)的調(diào)用

4.1 傳值調(diào)用

函數(shù)的形參和實參分別占有不同內(nèi)存塊,對形參的修改不會影響實參。

4.2 傳址調(diào)用

  • 傳址調(diào)用是把函數(shù)外部創(chuàng)建變量的內(nèi)存地址傳遞給函數(shù)參數(shù)的一種調(diào)用函數(shù)的方式
  • 這種傳參方式可以讓函數(shù)和函數(shù)外邊的變量建立起真正的聯(lián)系,也就是函數(shù)內(nèi)部可以直接操作函數(shù)外部的變量

4.3 練習

4.3.1 判斷一個數(shù)是不是素數(shù)

寫一個函數(shù)可以判斷一個數(shù)是不是素數(shù)

//返回1 表示是素數(shù)
//返回-1 表示不是素數(shù)

//寫法1  2-n-1 試除法
int  is_sushu(int a)
{//素數(shù)就是除1和自身外,不能被其他數(shù)整除
	for (int i = 2; i < a-1; i++)
	{//用2-n-1的數(shù)一一試除
		if (a%i==0)
		{//若有能除的,直接返回-1,跳出后面的循環(huán)了
			return -1;//表明不是素數(shù),
		}
	}//break退出循環(huán)會調(diào)到這里
	//return返回時直接退出這個函數(shù)了,返回到主函數(shù)了,級別更高
	return 1;
}

int main()
{
	int a = 0;
	scanf("%d", &a);
	int num = is_sushu(a);
	if (num==1)
	{
		printf("%d: 是素數(shù)", a);
	}
	else
	{
		printf("%d:不是素數(shù)", a);
	}
	return 0;
}

//寫法2  2-sqrt(n) 試除法,速度更快
#include<math.h>
int  is_sushu(int num)
{
	for (int i = 2; i <=sqrt(num); i++)
	{
		if (num%i == 0)
		{
			return -1;
		}
	}
	return 1;
}

int main()
{
	int num = 0;
	for (int num = 100; num < 201; num++)
	{
		if ((is_sushu(num)) == 1)
		{
			printf("%d ", num);
		}
	}
	return 0;
}

4.3.2 判斷一年是不是閏年

寫一個函數(shù)判斷一年是不是閏年

//閏年時是4的倍數(shù)且不是100的倍數(shù),或者是400的倍數(shù)
int  is_run_nian(int y)
{
		if (((num%4 == 0)&& (num%100 != 0))|| (num % 400 == 0))
		{
			return 1;
		}
	//return ((num%4 == 0)&& (num%100 != 0))|| (num % 400 == 0);//簡寫
		
}
int main()
{
	int num = 0;
	for (int num = 1000; num <= 2000; num++)
	{
		if ((is_run_nian(num)) == 1)
		{
			printf("%d ", num);
		}
	}
	return 0;
}

4.3.3 二分查找

寫一個函數(shù),實現(xiàn)一個整形有序數(shù)組的二分查找

//找到了就返回下標
//找不到返回-1
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	int k = 0;
	int i = 0;
	int sz = sizeof(arr) / sizeof(arr[0]);
	int left = 0;//左下標
	int right = sz - 1;//右下標

	scanf("%d", &k);
	while (left<=right)
	{//每次left right mid都要更新
		int mid = left + (right - left) / 2;		   
		if (arr[mid] > k)
		{//尋找的數(shù)值在左半邊,所以左不動,右邊動
			right = mid - 1;
		}
		else if (arr[mid] < k)
		{//尋找的數(shù)值在右半邊,所以左邊動,右不動
			left = mid + 1;
		}
		else 
		{
			printf("找到了:%d ", mid);
			break;
		}
	}
	if (left>right)//循環(huán)結(jié)束了
	{
		printf("找不到");
	}
	return 0;
}

可以將二分法進行改進,寫成獨立的函數(shù),讓函數(shù)功能單一化:

int is_erfen(int arr[],int k, int sz)
{//數(shù)組傳遞參數(shù)進來就是數(shù)組首元素的地址,并不是整個數(shù)組
	//地址的大小是4或8個字節(jié),前面有講到過的
//x86平臺中,地址占4個字節(jié),sizeof(arr)是4, 而不是40了
	//int sz = sizeof(arr) / sizeof(arr[0]);//sz要在主函數(shù)計算,
	int left = 0;
	int right = sz - 1;
	while (left <= right)
	{//每次left right mid都要更新
		int mid = left + (right - left) / 2;
		if (arr[mid] > k)
		{
			right = mid - 1;
		}
		else if (arr[mid] < k)
		{
			left = mid + 1;
		}
		else
		{
			return mid;
		}
	}
	return -1;
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	int k = 0;
	int i = 0;
	scanf("%d", &k);
	//主函數(shù)中  按數(shù)組名找到整個數(shù)組,求出占用字節(jié)是40
	int sz = sizeof(arr) / sizeof(arr[0]);
	int num = is_erfen(arr, k, sz);

	if (num ==-1)
	{
		printf("找不到"); 
	}
	else
	{
		printf("找到了:%d ", num);
	}
	return 0;
}

在這里插入圖片描述

在上面的函數(shù)中需要注意,數(shù)組的大小必須在主函數(shù)中計算,下面代碼可說明在主函數(shù)中和其他函數(shù)中求數(shù)組長度的區(qū)別:

int erfen(int arr[])
{//數(shù)組傳遞進來,只能是首地址,這個arr數(shù)組里只有{1},長度為4
	int sz1 = sizeof(arr) / sizeof(arr[0]);,//4/4 = 1求出只有1個元素
	printf("erfen求數(shù)組大小,sz1:%d\n ", sz1);
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };	
	//數(shù)組長度為40
	int sz = sizeof(arr) / sizeof(arr[0]);//40/4 = 10 求出數(shù)組10個元素
	printf("main中求數(shù)組大小,sz:%d\n ", sz);
	erfen(arr);	
	return 0;
}

在這里插入圖片描述

按F10進入調(diào)試界面,按F11,可觀察到:

  • 主函數(shù)中sizeof(arr)長度是40,數(shù)組中包含10個元素。
  • 而在函數(shù)erfen中,sizeof(arr)長度是4,數(shù)組中包含1個元素,就是首元素{ 1 }。

在這里插入圖片描述

由此,可知道,參數(shù)里傳遞數(shù)組時,實際傳遞的數(shù)組就是數(shù)組的地址,也是數(shù)組首元素的地址。數(shù)組名傳遞進來,只能是首地址,這個arr數(shù)組里只有首元素{1},長度為4。整個數(shù)組是傳遞不了的。因此,必須在主函數(shù)里求取數(shù)組的長度。

因為傳遞的參數(shù)是地址,所以erfen中也可以定義指針來接受數(shù)組:

//int erfen(int arr[])//接受數(shù)組,只能接受一個首元素
int erfen(int* arr)
{//數(shù)組傳遞進來,只能是首地址,這個arr數(shù)組里只有{1},長度為4
	int sz1 = sizeof(arr) / sizeof(arr[0]);,//4/4 = 1求出只有1個元素
	printf("erfen求數(shù)組大小,sz1:%d\n ", sz1);
}
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };	
	//數(shù)組長度為40
	int sz = sizeof(arr) / sizeof(arr[0]);//40/4 = 10 求出數(shù)組10個元素
	printf("main中求數(shù)組大小,sz:%d\n ", sz);
	erfen(arr);	
	return 0;
}

運行結(jié)果一樣

在這里插入圖片描述

4.3.4 數(shù)值自增增加1

寫一個函數(shù)用一次這個函數(shù),就會將 num 的值增加1

int add(int* p)
{
	(*p)++;//參數(shù)傳地址,可以操作實參的值
}
int main()
{
	int a = 10;
	add(&a);//傳地址
	printf("%d\n", a);
	add(&a);
	printf("%d\n", a);
	add(&a);
	printf("%d\n", a);
	add(&a);
	printf("%d\n", a);
	return 0;
}

5、函數(shù)的嵌套調(diào)用和鏈式訪問

函數(shù)和函數(shù)之間可以根據(jù)實際的需求進行組合的,也就是互相調(diào)用的。

5.1 嵌套調(diào)用

//套娃
void newline()
{
	printf("hehe\n");
}
void threeline()
{
	int i = 0;
	for (int i = 0; i < 3; i++)
	{
		newline();
	}
}
int main()
{
	threeline();
	newline;
	return 0;
}

5.2 鏈式訪問

把一個函數(shù)的返回值作為另外一個函數(shù)的參數(shù)。

int main()
{
	char arr[20] = "hello";
	int ret = strlen(strcat(arr, "bit"));
	printf("%d\n", ret);
	return 0;
}
int main()
{	
	printf("%d", printf("%d", printf("%d", 43)));//輸出4321,
	//前面是打印43 2是返回兩個字符(因為4 、3是兩個字符)  
	//1是返回1個字符(2是1個字符)

	return 0;
}

輸出結(jié)果見下圖:

在這里插入圖片描述

在這里插入圖片描述

總結(jié)

本文主要介紹了函數(shù)相關(guān)的知識,下一篇接著介紹函數(shù)的內(nèi)容

到此這篇關(guān)于C語言函數(shù)超詳細講解上篇的文章就介紹到這了,更多相關(guān)C語言 函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言控制臺實現(xiàn)字符飛機大戰(zhàn)

    C語言控制臺實現(xiàn)字符飛機大戰(zhàn)

    這篇文章主要為大家詳細介紹了C語言控制臺實現(xiàn)字符飛機大戰(zhàn),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • C語言數(shù)組和指針,內(nèi)存之間的關(guān)系

    C語言數(shù)組和指針,內(nèi)存之間的關(guān)系

    這篇文章主要介紹了C語言數(shù)組和指針,內(nèi)存之間的關(guān)系,首先論證一維數(shù)組和一級指針之前的關(guān)系,我們常常使用一級指針指針的方式訪問一維數(shù)組,只有對內(nèi)存的理解到位才能理解它們直接的關(guān)系。需要的小伙伴可以參考一下
    2022-02-02
  • C語言位段(位域)機制結(jié)構(gòu)體的特殊實現(xiàn)及解析

    C語言位段(位域)機制結(jié)構(gòu)體的特殊實現(xiàn)及解析

    這篇文章主要為大家介紹了C語言位段位域機制結(jié)構(gòu)體的特殊實現(xiàn)講解有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪
    2022-02-02
  • 關(guān)于C語言多線程pthread庫的相關(guān)函數(shù)說明

    關(guān)于C語言多線程pthread庫的相關(guān)函數(shù)說明

    下面小編就為大家?guī)硪黄P(guān)于C語言多線程pthread庫的相關(guān)函數(shù)說明。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • VC++?2019?"const?char*"類型的實參與"LPCTSTR"類型的形參不兼容解決

    VC++?2019?"const?char*"類型的實參與"LPCTSTR"

    這篇文章主要給大家介紹了關(guān)于VC++?2019?"const?char*"類型的實參與"LPCTSTR"類型的形參不兼容的解決方法,文中通過圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2023-03-03
  • C++詳細實現(xiàn)完整圖書管理功能

    C++詳細實現(xiàn)完整圖書管理功能

    隨著網(wǎng)絡技術(shù)的高速發(fā)展,計算機應用的普及,利用計算機對圖書館的日常工作進行管理勢在必行,本篇文章涵蓋一個圖書管理系統(tǒng)的全部實現(xiàn)代碼,大家可以查缺補漏,提升水平
    2022-05-05
  • C/C++?函數(shù)的存儲位置和占用空間詳解

    C/C++?函數(shù)的存儲位置和占用空間詳解

    Lambda函數(shù)的代碼部分在代碼段中,被捕獲的變量存儲在Lambda函數(shù)對象的內(nèi)部,這些變量的存儲位置取決于Lambda函數(shù)對象的存儲位置,這篇文章主要介紹了C/C++函數(shù)的存儲位置和占用空間,需要的朋友可以參考下
    2023-06-06
  • C++之重載 重定義與重寫用法詳解

    C++之重載 重定義與重寫用法詳解

    這篇文章主要介紹了C++之重載 重定義與重寫用法詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • c++中typename和class的區(qū)別介紹

    c++中typename和class的區(qū)別介紹

    在c++Template中,很多地方都用到了typename與class這兩個關(guān)鍵字,而且好像可以替換,是不是這兩個關(guān)鍵字完全一樣呢?
    2013-03-03
  • 詳解C++中賦值和輸入輸出語句的用法

    詳解C++中賦值和輸入輸出語句的用法

    這篇文章主要介紹了詳解C++中賦值和輸入輸出語句的用法,是C++入門學習中的基礎(chǔ)知識,需要的朋友可以參考下
    2015-09-09

最新評論