day07 指针和字符串

本文详细探讨了C语言中指针与函数的交互,包括栈帧概念、传值与传址的区别、指针和数组作为函数参数的处理、函数返回值的指针与数组形式,以及字符串操作的相关练习。同时,介绍了带参数的main函数的使用和测试,最后提供了若干巩固练习题,帮助读者深入理解指针和函数的应用。
摘要由CSDN通过智能技术生成

一、指针和函数:

1.栈 帧:

	当函数调用时,系统会在 stack 空间上申请一块内存区域,用来供函数调用,主要存放 形参 和 局部变量(定义在函数内部)。

	当函数调用结束,这块内存区域自动被释放(消失)。

  这个就像是工地里请了批个工人过来干活,那么要腾出一片空间给这个工人使用,搭棚放工具啥的。如果工期结束了这片空间上的东西就销毁了

在这里插入图片描述

2.传值和传址:

2.1传值:

  函数调用期间,实参将自己的值,拷贝一份给形参。 不会影响原来实参的值。为什么不会影响呢?因为被调用的函数中,操作的形参和实参是不同的变量,或者说,处于不同的内存空间中,大家是独立的,当然影响不了。

  咱们来举个例子,有一次回老家,我一共买了两块肉。一块分给大哥家,一块自己留,这个相当于拷贝。然后大哥家当晚就煮了,他们采用红烧。难道大哥家的肉红烧了,我家也变成红烧了吗。不会的,因为这是两块肉,不会影响到。

  (1)下面我就用代码,来测试,并且将实参的地址和形参的地址打了出来。

int swap(int a, int b);

int main()
{
        int m ,n;
        
        m = 32;
        n = 12;

        printf("m 的地址为:%p\n",&m);
        printf("n 的地址为:%p\n",&n);

        swap(m,n);

        printf("m=%d,n=%d\n",m,n);
        
}

int swap(int a, int b)
{
        int tmp = 0;

        tmp = a;
        a   = b;
        b   = tmp;

        printf("a 的地址为:%p\n",&a);
        printf("b 的地址为:%p\n",&b);

        return 0;
}

  (2)运行结果,可以看到地址不一样,操作的值放在不同的变量里面,当然不会有影响。
在这里插入图片描述

2.2传址:

  函数调用期间,实参将地址值,拷贝一份给形参。 相当于直接操作实参了,因为被调用的函数操作的就是实参所在的内存。【重点】

  这个就相当于我叫大哥直接来我家煮我家的肉了,也是红烧的。大哥都直接来到我家操作我家的肉了,肯定会变呀。

		(地址值 --》 在swap函数栈帧内部,修改了main函数栈帧内部的局部变量值)

(1)测试代码:

int swap2(int *a, int *b);

int main()
{
        int m ,n;
        

        m = 32;
        n = 12;

        printf("m 的地址为:%p\n",&m);
        printf("n 的地址为:%p\n",&n);

        //swap(m,n);
        swap2(&m,&n);

        printf("m=%d,n=%d\n",m,n);
        
}

int swap2(int *a, int *b)
{
        int tmp = 0;
 
        tmp = *a;
        *a = *b;
        *b = tmp;

        printf("a 的地址为:%p\n",a);
        printf("b 的地址为:%p\n",b);
 
        return 0;
}

(2)测试结果,被调用函数直接操作实参,因为地址是一样的。
在这里插入图片描述

3.指针做函数参数:

	int swap2(int *a, int *b);

	int swap2(char *a, char *b);

	调用时,传有效的地址值。

4.数组做函数参数:

  数组传递的时候,我们通常是传递首元素的地址过去就行。因为数组是连续的,利用首元素就可以很方便地操作其他的元素。

  如果是讲整个数组传递过去,这样处理很不方便,当数组很大的时候。就相当于你给一个人传信息,你告诉他我传的是100个连续的数字,相邻的数相差1。那你只需要将第一个数传给他就行了,然后说一共有多少个数,其他的让他自己写。这样就很方便,就不用你写完100个数字出来。

void BubbleSort(int arr[10]) == void BubbleSort(int arr[])  == void BubbleSort(int *arr) 

  重点:传递不再是整个数组,而是数组的首地址(一个指针)

  所以,当整型数组做函数参数时,我们通常在函数定义中,封装2个参数。一个表数组首地址,一个表元素个数。

4.1 注意调用函数与别调用函数中的sizeof(arr);

  要注意这个区别,比如说数组是在主函数中定义的,数组名为arr。此时我们使用 sizeof(arr) 求的是数组的大小。如果我们将数组名传给被调用的函数,那么sizeof(arr)是求指针的大小,或者说是求数组类型指针的大小。比如说数组是 int 型,那么在被调用函数中使用sizeof(arr)就等于 sizeof(int *);

(1)测试:

int main() // 主函数
{
 int arr[7] = {1,2,3,4,5,6,7};
        printf(" 在主函数中 sizeof(arr)=%d\n",sizeof(arr));  
}

int PrintfArr(int *arr, int size) // 被调用函数
{
        printf(" 在被调用函数中 sizeof(arr)=%d,sizeof(int *)=%d\n",sizeof(arr),sizeof(int *));
}

(2)测试结果:
在这里插入图片描述
在这里插入图片描述

5.指针做函数返回值:

	int *test_func(int a, int b);

	指针做函数返回值,不能返回【局部变量的地址值】。因为局部变量会被释放,如果不及时使用,就会被释放,得不到预期的值。所以为了安全考虑,不用局部变量的地址。

6.数组做函数返回值:

	C语言,不允许!!!!  只能写成指针形式。

二、指针和字符串:

  通常,在c中我们操作字符串的方式有两种,一种是数组的方式,另一种是指针的方式。

  数组就是 :

str[i];
i++;

  指针的方式就是:

1*(str1+i);
	i++;2*str1;
	 str1++;
	1char str1[] = {'h', 'i', '\0'};			变量,可读可写

		char str2[] = "hi";				变量,可读可写

		char *str3 = "hi";				常量,只读

			str3变量中,存储的是字符串常量“hi”中首个字符‘h’的地址值。


			str3[1] = 'H';	// 错误!!

		char *str4 = {'h', 'i', '\0'};  // 错误!!!

	

  当字符串(字符数组), 做函数参数时, 不需要提供2个参数。 因为每个字符串都有 ‘\0’。

1.练习:比较两个字符串: strcmp();实现

  比较 str1 和 str2, 如果相同返回0, 不同则依次比较ASCII码,str1 > str2 返回1,否则返回-1

1.1数组方式:

int mystrcmp(char *str1, char *str2)
{
	int i = 0;

	while (str1[i] == str2[i])   // *(str1+i) == *(str2+i)
	{
		if (str1[i] == '\0')
		{
			return 0;			// 2字符串一样。
		}
		i++;
	}
	return str1[i] > str2[i] ? 1 : -1;
}

1.2指针方式:

int mystrcmp2(char *str1, char *str2)
{
	while (*str1 == *str2)   // *(str1+i) == *(str2+i)
	{
		if (*str1 == '\0')
		{
			return 0;			// 2字符串一样。
		}
		str1++;
		str2++;
	}
	return *str1 > *str2 ? 1 : -1;
}

2.练习:字符串拷贝:

2.1数组版本

void mystrcpy(char *src, char *dst)
{
	int i = 0;
	while (src[i] != 0)  // src[i] == *(src+i)
	{
		dst[i] = src[i];
		i++;
	}
	dst[i] = '\0';
}

2.2指针版

void mystrcpy2(char *src, char *dst)
{
	while (*src != '\0')  // src[i] == *(src+i)
	{
		*dst = *src;
		src++;
		dst++;
	}
	*dst = '\0';
}

3.练习:在字符串中查找字符出现的位置:


char *myStrch(char *str, char ch)
{
	while (*str)
	{
		if (*str == ch)
		{
			return str;
		}
		str++;
	}
	return NULL;
}
// hellowrld --- 'o'
char *myStrch2(char *str, char ch)
{
	int i = 0;
	while (str[i])
	{
		if (str[i] == ch)
		{
			return &str[i];  
		}
		i++;
	}
	return NULL;
}

4.练 习:字符串去空格。

void str_no_space(char *src, char *dst)
{
	int i = 0;   // 遍历字符串src
	int j = 0;	 // 记录dst存储位置
	while (src[i] != 0)
	{
		if (src[i] != ' ')
		{
			dst[j] = src[i];
			j++;
		}
		i++;
	}
	dst[j] = '\0';
}
// 指针版
void str_no_space2(char *src, char *dst)
{
	while (*src != 0)
	{
		if (*src != ' ')
		{
			*dst = *src;
			dst++;
		}
		src++;
	}
	*dst = '\0';
}

5.注意 arr+i 与 arr++ 的区别

  在写程序的时候,就犯了一个低级的错误,引起了段错误。原因就是不注意 arr+i 与 arr++的区别。 循环的时候,arr 始终指向首元素,arr++ ,arr会往后指向。

(1)测试程序:测试 str1++

void DeleteNbspace1(char *str1,int size)
{
        printf("str1  = %p\n",str1);
        char str2[24];
        int i = 0;
         while( *str1 )
        {       
                if( *str1 != ' ' )
                {       
                        str2[i] = *str1 ;                   
                        i++;
                }
                str1++;     
        }
        str2[i] = '\0';
        
        printf("str1  = %p\n",str1);

查看结果
在这里插入图片描述

(2)测试 str1+j:

void DeleteNbspace1(char *str1,int size)
{
        printf("str1  = %p\n",str1);
        char str2[24];
        int i = 0;
        int j = 0;

        while( *(str1+j) )
        {
                if( *(str1+j) != ' ' )
                {
                        str2[i] = *(str1+j) ;   

                        i++;
                }
                j++;
        }
         str2[i] = '\0';
        
        printf("str1  = %p\n",str1);

查看结果:
在这里插入图片描述

三、带参数的main函数:

1.无参main函数:

	int main(void) == int main()

2.带参数的main函数:

 int main(int argc, char *argv[]) == int main(int argc, char **argv)
	参1:表示给main函数传递的参数的总个数。

	参2:是一个数组!数组的每一个元素都是字符串 char * 

在这里插入图片描述

3.测试1:

(1)在Windows命令行中的中,使用gcc编译生成 可执行文件,如: test.exe

	test.exe abc xyz zhangsan nichousha 

	-->

	argc --- 5
	test.exe -- argv[0]
	abc -- argv[1]
	xyz -- argv[2]
	zhangsan -- argv[3]
	nichousha -- argv[4]

(2)在Linux 中,测试代码

int main(int argc, char *argv[])
{
        printf("用于测试带参数的主函数。\n");

        printf(" argc : 参数的数量。argc = %d\n",argc);

        printf(" argv 参数数组。下面将参数打印出来.\n");

        int i = 0;

        for( i = 0; i<argc; i++ )
        {
                printf("第 %d 个 参数:%s\n",i,argv[i]);
        }

        return 0;
}

测试结果:
在这里插入图片描述

4.测试2:

	在VS中。项目名称上 --》右键--》属性--》调试--》命令行参数 --》将 test.exe abc xyz zhangsan nichousha 写入。

	-->

	argc --- 5
	test.exe -- argv[0]
	abc -- argv[1]
	xyz -- argv[2]
	zhangsan -- argv[3]
	nichousha -- argv[4]

四、巩固练习

1.求非空字符串元素个数:

“ni chou sha chou ni za di”  

2.字符串逆置: str_inverse


	hello -- olleh 

void str_inserse(char *str)
{
	char *start = str;			// 记录首元素地址
	char *end = str + strlen(str) - 1;	// 记录最后一个元素地址。

	while (start < end)			// 首元素地址是否 < 最后一个元素地址
	{
		char tmp = *start;		// 三杯水 char 元素交换
		*start = *end;
		*end = tmp;
		start++;			// 首元素对应指针后移
		end--;				// 尾元素对应指针前移
	}
}

3.判断字符串是回文:

int str_abcbb(char *str)
{
	char *start = str;			// 记录首元素地址
	char *end = str + strlen(str) - 1;	// 记录最后一个元素地址。

	while (start < end)			// 首元素地址是否 < 最后一个元素地址
	{
		if (*start != *end)		// 判断字符是否一致。
		{
			return 0; 		// 0 表示非 回文
		}
		start++;
		end--;
	}
	return 1;				// 1 表示 回文
}

4.字符串处理函数: #include <string.h>

4.1字符串拷贝:


		strcpy: 

			将 src 的内容,拷贝给 dest。 返回 dest。 保证dest空间足够大。【不安全】

			char *strcpy(char *dest, const char *src);

			函数调用结束 返回值和 dest参数结果一致。

		strncpy:

			将 src 的内容,拷贝给 dest。只拷贝 n 个字节。 通常 n 与dest对应的空间一致。 

			默认 不添加 ‘\0char *strncpy(char *dest, const char *src, size_t n);	

			特性: n > src: 只拷贝 src 的大小

			       n < src: 只拷贝 n 字节大小。 不添加 ‘\0

4.2 字符串拼接:

	strcat:

		将 src 的内容,拼接到 dest 后。 返回拼接后的字符串。	保证 dest 空间足够大。

		char *strcat(char *dest, const char *src);

	strncat:

		将 src 的前 n 个字符,拼接到 dest 后。 形成一个新的字符串。保证 dest 空间足够大。

		char *strncat(char *dest, const char *src, size_t n);

		函数调用结束 返回值和 dest 参数结果一致。

4.3 字符串比较: 不能使用 > < >= <= == !=

	strcmp:

		比较s1和s2两个字符串,如果相等 返回0.如果不相等,进一步表 s1 和 s2 对应位 ASCII码值。

		s1 > s2 返回1

		s1 < s2 返回-1

		int strcmp(const char *s1, const char *s2);

	strncmp:

		int strncmp(const char *s1, const char *s2, size_t n);

		比较s1和s2两个字符串的前n个字符,

		如果相等 返回0。如果不相等,进一步表 s1 和 s2 对应位 ASCII码值。(不比字符串ASCII码的和)

		s1 > s2 返回1

		s1 < s2 返回-1

4. 4 字符串格式化输入、输出:

	sprintf():  s -- string

		int sprintf(char *str, const char *format, ...);

		对应printf,将原来写到屏幕的“格式化字符串”,写到 参数1 str中。

		printf("%d+%d=%d\n", 10, 24, 10+24);

		---》 
		char str[100];

		sprintf(str, "%d+%d=%d\n", 10, 24, 10+24);  格式串写入str数组中。

	sscanf():

		int sscanf(const char *str, const char *format, ...);

		对应scanf, 将原来从屏幕获取的“格式化字符串”, 从 参数1 str中 获取。

		scanf("%d+%d=%d", &a, &b, &c);

		---》

		char str[]= "10+24=45";

		sscanf(str, "%d+%d=%d", &a, &b, &c);  a --> 10, b --> 24, c --> 45

4.5 字符串查找字符、子串:

	strchr():

		在字符串str中 找一个字符出现的位置。 返回字符在字符串中的地址。

		char *strchr(const char *s, int c);

		printf("%s\n" strchr("hehehahahoho", 'a'));  --> "ahahoho"

	strrchr():

		自右向左,在字符串str中 找一个字符出现的位置。 返回字符在字符串中的地址。

		char *strrchr(const char *s, int c);

		printf("%s\n" strrchr("hehehahahoho", 'a'));  --> "ahoho"

	strstr():

		在字符串str中,找子串substr第一次出现的位置。返回地址。

		char *strstr(const char *str, const char *substr);

		在字符串中找子串的位置。 

		printf("%s\n" strrchr("hehehahahoho", "ho"));  --> "hoho"

		printf("%s\n" strrchr("hehehahahoho", "xixi"));  --> NULL

scanf("%s", str);

scanf("%[^\n]", str);

4.6 字符串分割:

	strtok(): 按照既定的分割符,来拆分字符串。“www.baidu.com”  --> "www\0baidu.com"

	char *strtok(char *str, const char *delim);

		参1: 待拆分字符串

		参2: 分割符组成的“分割串”

	返回:字符串拆分后的首地址。 “拆分”:将分割字符用 '\0'替换。

特性:
	1)strtok拆分字符串是直接在 原串 上操作,所以要求参1必须,可读可写(char *str = "www.baidu.com" 不行!!!)

	2)第一次拆分,参1 传待拆分的原串。	第1+ 次拆分时,参1传 NULL.


练习: 拆分 ".itcast.cn$This is a strtok$test"

	char str[] = "www.itcast.cn$This is a strtok$test";

	char *p = strtok(str, "$ .");

	while (p != NULL)
	{	
		p = strtok(NULL, " .$");
		printf("p = %s\n", p);
	}

4.7 atoi/atof/atol:

	使用这类函数进行转换,要求,原串必须是可转换的字符串。

	错误使用:"abc123" --> 0;	"12abc345" ---> 12;  "123xyz" -->123
	
	atoi:字符串 转 整数。

	int atoi(const char *nptr);

	atof:字符串 转 浮点数

	atol:字符串 转 长整数

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值