字符函数和字符串函数

字符转换函数

C语言提供了两个字符转换函数:

int tolower(int c);//将参数传进去的大写字母转小写

int toupper(int c);//将参数传进去的小写字母转大写

使用函数需要使用头文件 <ctype.h>

这样,我们将大小写字母转换的方式就不仅仅只有将ASCII码±32了,我们还可以直接使用转换函数。
下面我们通过一个将小写字母转换为大写字母,其他字符不变的代码练习来理解toupper

#include <stdio.h>
#include <ctype.h>
int main()
{
	int i = 0;
	char str[] = "Onward and Upward!";
	char c = 0;
	while (str[i])
	{
		c = str[i];
		if (islower(c))//islower是字符分类函数,判断c是不是小写字母
						//之前的文章我们学习过
		{
			c = toupper(c);
		}
		putchar(c);
		i++;
	}
	return 0;
}

在这里插入图片描述
下面我们通过一个将小写字母转换为大写字母,其他字符不变的代码练习来理解tolower

#include <stdio.h>
#include <ctype.h>
int main()
{
	int i = 0;
	char str[] = "LIFE AND LEARN!";
	char c = 0;
	while (str[i])
	{
		c = str[i];
		if (isupper(c))//isupper是字符分类函数,判断c是不是大写字母
			//之前的文章我们学习过
		{
			c = tolower(c);
		}
		putchar(c);
		i++;
	}
	return 0;
}

在这里插入图片描述

strlen的使用和模拟实现

size_t strlen( const char *string );
  • 字符串以 ’ \0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数(不包含 ‘\0’ )
  • 参数指向的字符串必须要以 ‘\0’ 结束。
  • 注意函数的返回值为size_t,是无符号的(易错)
  • strlen的使用需要包含头文件 <string.h>

为什么说函数返回值是size_t,是无符号的整数类型,我们容易用错呢?
接下来我们给出一段代码:

#include <stdio.h>
#include <string.h>
int main()
{
	const char* str1 = "onward and";
	const char* str2 = "upward";
	if (strlen(str2) - strlen(str1) > 0)
	{
		printf("str2 > str1\n");
	}
	else
	{
		printf("str1 > str2\n");
	}
	return 0;
}

我们可以看出str2应该是小于str1,结果应该是str1 > str2,但是结果却不是这样。

在这里插入图片描述

那为什么会出现这个结果呢,问题就出现在这个函数返回值是无符号整数类型,str1的长度为10,str2的长度为6,按照我们数学的算法6 - 10= -4, 结果应该是-4,输出str1 > str2,但是在C语言中无符号整型 - 无符号整型,结果应该也是无符号整型。
我们都知道整型是4个字节,也就是32个比特位
我们知道数据在内存是二进制补码存储和表示的,便于硬件(加法、减法统一处理)设计和符号位和数值位的统一处理。
所以CPU只有加法器没有减法器,所以6 - 10 = 6 + (-10)
6 的补码: 00000000 00000000 00000000 00000110(正数的原码,反码,补码都是一样的)
-10的原码:10000000 00000000 00000000 00001010 (负数的补码是原码取成反码再+1)
-10的反码:11111111 11111111 11111111 11110101 (负数的反码是符号位不变,其他位取反)
-10的补码:11111111 11111111 11111111 11110110
所以6的补码和-10的补码相加为:
11111111 11111111 11111111 11111100
此时会将其视为一个无符号整数类型,算出来就是4,294,967,292,是一个大于0的数,所以输出str2 > str1

#include <stdio.h>
#include <string.h>
int main()
{
	const char* str1 = "onward and";
	const char* str2 = "upward";
	printf("%u\n", strlen(str2) - strlen(str1));//%u是结果以无符号整数类型打印
	return 0;
}

在这里插入图片描述
在这里插入图片描述
所以我们有以下几种解决方法

#include <stdio.h>
#include <string.h>
int main()
{
	const char* str1 = "onward and";
	const char* str2 = "upward";
	if ((int)strlen(str2) - (int)strlen(str1) > 0)
	{
		printf("str2 > str1\n");
	}
	else
	{
		printf("str1 > str2\n");
	}
	return 0;
}
#include <stdio.h>
#include <string.h>
int main()
{
	const char* str1 = "onward and";
	const char* str2 = "upward";
	if (strlen(str2) > strlen(str1))
	{
		printf("str2 > str1\n");
	}
	else
	{
		printf("str1 > str2\n");
	}
	return 0;
}

结果都是
在这里插入图片描述

好了,接下来我们来学习如何模拟实现strlen
方式一:

//计数器方式
#include <stdio.h>
#include <assert.h>
size_t my_strlen(const char* str)
{
	int count = 0;
	assert(str != NULL);//断言,宏assert()判断括号中表达式是否为真,若假则终止程序运行,
						//并且给出报错信息提示(在指针那篇文章中我们了解过)
	while (*str != '\0')
	{
		count++;
		str++;
	}
	return count;
}
int main()
{
	const char* str1 = "never give up";
	printf("%d", my_strlen(str1));
	return 0;
}

在这里插入图片描述
方式二:

//不创建临时变量计数器,采用递归方法实现
#include <stdio.h>
#include <assert.h>
size_t my_strlen(const char* str)
{
	assert(str != NULL);//断言,宏assert()判断括号中表达式是否为真,若假则终止程序运行,
						//并且给出报错信息提示(在指针那篇文章中我们了解过)
	if (*str == '\0')
	{
		return 0;
	}
	return 1 + my_strlen(str+1);
}
int main()
{
	const char* str1 = "never give up";
	printf("%d", my_strlen(str1));
	return 0;
}

在这里插入图片描述
方式三:

//采用指针-指针的方式
#include <stdio.h>
#include <assert.h>
size_t my_strlen(const char* str)
{
	assert(str != NULL);//断言,宏assert()判断括号中表达式是否为真,若假则终止程序运行,
						//并且给出报错信息提示(在指针那篇文章中我们了解过)
	char* p = str;
	while (*p != '\0')
		p++;
	return p - str;
}
int main()
{
	const char* str1 = "never give up";
	printf("%d", my_strlen(str1));
	return 0;
}

在这里插入图片描述

strcpy的使用和模拟实现

char *strcpy( char *strDestination, const char *strSource );
  • strcpy 把 strSource 所指向的字符串复制到 strDestination 中。
  • 源字符串 strSource 必须以 ‘\0’ 结束。
  • strcpy 会将源字符串 strSource 中的 ‘\0’ 拷贝到目标空间 strDestination 。
  • 目标空间 strDestination 必须足够大,以确保能存放源字符串 strSource 。
  • 目标空间 strDestination 必须可以修改,不可以是常量字符串(不可以修改),应该是一个字符数组。
char* str1 = "abcdef";//常量字符串,不可以修改,没法复制到这里面
char str2[20] = "abcdef";//字符串放在字符数组中,可以修改,只要空间足够就可以复制到里面去

通过一段代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char str1[] = "onward and upward";
	char str2[20] = "xxxxxxxxxxxxxxxxxxxx";
	strcpy(str2, str1);
	printf("%s", str2);
	return 0;
}

在这里插入图片描述
strcpy的模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	//拷贝\0前面的内容
	while (*src != '\0')
	{
		*dest = *src;
		dest++;
		src++;
	}
	//拷贝\0
	*dest = *src;
}
int main()
{
	char str1[] = "onward and upward";
	char str2[20] = "xxxxxxxxxxxxxxxxxxxx";
	my_strcpy(str2, str1);
	printf("%s", str2);
	return 0;
}

在这里插入图片描述
我们还可以对代码进行优化:

#include <stdio.h>
#include <assert.h>
char* my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while (*dest++ = *src++)
	{
		;
	}
}
int main()
{
	char str1[] = "onward and upward";
	char str2[20] = "xxxxxxxxxxxxxxxxxxxx";
	my_strcpy(str2, str1);
	printf("%s", str2);
	return 0;
}

在这里插入图片描述

strcat的使用和模拟实现

char *strcat( char *strDestination, const char *strSource );
  • strcat 把 strSource 所指向的字符串追加到 strDestination 所指向的字符串的结尾。
  • 源字符串 strSource 中必须以 ‘\0’ 结束。
  • 目标字符串 strDestination 中也得有 ‘\0’, 否则没办法知道追加从哪里开始。
  • 目标空间 strDestination 必须有足够的大,能容纳下源字符串 strSource 的内容。
  • 目标空间 strDestination 必须可以修改。
#include <stdio.h>
#include <string.h>
int main()
{
	char str1[20] = "onward and ";
	char str2[20] = "upward";
	strcat(str1, str2);
	printf("%s", str1);
	return 0;
}

在这里插入图片描述
模拟实现strcat函数

#include <stdio.h>
#include <assert.h>
char* my_strcat(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while (*dest != '\0')
		dest++;
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}
int main()
{
	char str1[20] = "onward and ";
	char str2[20] = "upward";
	my_strcat(str1, str2);
	printf("%s", str1);
	return 0;
}

在这里插入图片描述

如果字符串自己给自己追加,可以吗?
通过代码来看看:

#include <stdio.h>
#include <assert.h>
char* my_strcat(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while (*dest != '\0')
		dest++;
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}
int main()
{
	char str2[20] = "upward";
	my_strcat(str2, str2);
	printf("%s", str2);
	return 0;
}

在这里插入图片描述
可以看出并没有输出内容,程序崩溃了,因为出现了死循环,并且越界访问了
在这里插入图片描述
所以最好不要自己给自己追加。

strcmp的使用和模拟实现

int strcmp( const char *string1, const char *string2 );
  • 两个字符串自左向右逐个字符相比(按 ASCII 值大小相比较),直到出现不同的字符或遇 \0 为止。
  • 标准规定:
    • 第一个字符串大于第二个字符串,则返回大于0的数字
    • 第一个字符串等于第二个字符串,则返回0
    • 第一个字符串小于第二个字符串,则返回小于0的数字

通过代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char* str1 = "abcdef";
	char* str2 = "abq";
	int ret = strcmp(str1, str2);
	if (ret > 0)
	{
		printf("str1 > str2\n");
	}
	else if (ret = 0)
	{
		printf("str1 = str2\n");
	}
	else 
	{
		printf("str1 < str2\n");
	}
	return 0;
}

在这里插入图片描述
模拟实现strcmp函数:
首先先说一下两种错误:

char str1[] = "abcdef";
char str2[] = "abq";
if (str1 == str2)//数组名表示首元素的地址,这是两个地址在比较,两个不同的数组,地址肯定不同
if ("abcdef" == "abq")//也是在比较地址,我们在前面的文章指针中说过这是第一个字符的地址

接下来我们来模拟实现:

#include <stdio.h>
#include <assert.h>
int my_strcmp(const char* str1, const char* str2)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	while (*str1 == *str2)
	{
		if (*str1 == '\0')
			return 0;
		str1++;
		str2++;
	}
	if (*str1 > *str2)
		return 1;
	else
		return -1;
}
int main()
{
	char* str1 = "abcdef";
	char* str2 = "abq";
	int ret = my_strcmp(str1, str2);
	if (ret > 0)
	{
		printf("str1 > str2\n");
	}
	else if (ret = 0)
	{
		printf("str1 = str2\n");
	}
	else
	{
		printf("str1 < str2\n");
	}
	return 0;
}

在这里插入图片描述

我们可以看出上面的模拟实现可以完成任务,但是似乎跟最开始的标准有点不一样,小于只输出-1,大于只输出1,所以我们可以对代码进行优化。

#include <stdio.h>
#include <assert.h>
int my_strcmp(const char* str1, const char* str2)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	while (*str1 == *str2)
	{
		if (*str1 == '\0')
			return 0;
		str1++;
		str2++;
	}
	return *str1 - *str2;
}
int main()
{
	char* str1 = "abcdef";
	char* str2 = "abq";
	int ret = my_strcmp(str1, str2);
	if (ret > 0)
	{
		printf("str1 > str2\n");
	}
	else if (ret = 0)
	{
		printf("str1 = str2\n");
	}
	else
	{
		printf("str1 < str2\n");
	}
	return 0;
}

在这里插入图片描述

strncpy,strncat, strncmp

接下来我们将学习一组strncpy,strncat, strncmp,他们跟我们上面学的strcpy,strcat,strcmp有什么不同呢?
它们都多了一个n,我们将上面strcpy,strcat,strcmp这些函数叫做长度不受限制的字符串函数,它们关注的是 ‘\0’,将接下来的strncpy,strncat, strncmp叫做长度受限制的字符串函数。
我们给出这三个函数的声明,便可以理解为什么是长度受限制。

char *strncpy( char *strDest, const char *strSource, size_t count );

char *strncat( char *strDest, const char *strSource, size_t count );

int strncmp( const char *string1, const char *string2, size_t count );

两者进行比较,长度不受限制的字符串函数并不安全,而长度受限制的字符串函数相对安全,因为程序员在使用函数,由于需要考虑长度参数的传入,就会去检查是否出现数组访问越界。

char str1[10] = "abcdef";
char str2[10] = "abcdefg";
strcpy(str1, str2);//显然str1数组并不能存放两个数组,存在越界
strncpy(str1, str2, 7);//这时程序员已经知道str1中有7个字符(包含\0),再加7个必然越界,
						//这样程序员便可以修改程序,而不会导致后续的问题,所以相对安全

strncpy函数的使用和模拟实现

char *strncpy( char *strDest, const char *strSource, size_t count );
  • 拷贝count个字符从源字符串 strSource 到目标空间 strDest ,不会自动加 ‘\0’
  • 如果源字符串 strSource 的长度小于count, 则拷贝完源字符串 strSource 之后,在目标的后边追加 ‘\0’ ,直到count个。
  • 注意: strncpy 没有自动加上终止符的,需要手动加上不然会出问题的。

通过代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char str1[30] = "xxxxxxxxxxxx";
	char str2[] = "never give up";
	strncpy(str1, str2, 5);
	printf("%s", str1);
	return 0;
}

在这里插入图片描述

#include <stdio.h>
#include <string.h>
int main()
{
	char str1[30] = "xxxxxxxxxxxx";
	char str2[] = "never give up";
	strncpy(str1, str2, 5);
	str1[5] = '\0';
	printf("%s", str1);
	return 0;
}

在这里插入图片描述
strncpy函数的模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strncpy(char* dest, const char* src, size_t count)
{
	assert(dest != NULL);
	assert(src != NULL);
	while (count--)
	{
		if (*src == '\0')
		{
			*dest = '\0';
			dest++;
		}
		else
		{
			*dest = *src;
			dest++;
			src++;
		}
	}
}
int main()
{
	char str1[30] = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx";
	char str2[] = "never give up";
	my_strncpy(str1, str2, 15);
	printf("%s", str1);
	return 0;
}

在这里插入图片描述

strncat函数的使用和模拟实现

char *strncat( char *strDest, const char *strSource, size_t count );
  • 将 strSource 指向的字符串的前count个字符追加到 strDest 指向的字符串结尾,再追加一个 ‘\0’ 字符。
  • 如果 strSource 指向的字符串的长度小于count的时候,只会将字符串中 ‘\0’ 的内容追加到 strSource 指向的字符串结尾。

通过代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char str1[30] = "never ";
	char str2[] = "give up";
	strncat(str1, str2, 10);
	printf("%s", str1);
	return 0;
}

在这里插入图片描述

strncat函数的模拟实现

#include <stdio.h>
#include <assert.h>
char* my_strncat(char* dest, const char* src, size_t count)
{
	assert(dest != NULL);
	assert(src != NULL);
	while (*dest != '\0')
	{
		dest++;//到dest字符串结尾
	}
	while (count-- && ((*dest++ = *src++) != '\0'))//先判断是否拷贝了count个字符,
													//再判断是否拷贝到src字符串结尾,并进行拷贝
													//任何一个不满足都退出循环
	{
		;
	}
	if (*dest != '\0')//如果是因为count=0而结束循环,则末尾加一个\0
		*dest = '\0';
}
int main()
{
	char str1[30] = "never ";
	char str2[] = "give up";
	my_strncat(str1, str2, 7);
	printf("%s", str1);
	return 0;
}

在这里插入图片描述

strncmp函数的使用和模拟实现

int strncmp( const char *string1, const char *string2, size_t count );

比较string1和string2的前count个字符,如果相等就继续往后比较,最多比较count个字母,如果提前发现不一样,就提前结束,大的字符所在的字符串大于另一个。如果count个字符都相等,就是相等返回0。

  • 标准规定:
    • 第一个字符串大于第二个字符串,则返回大于0的数字
    • 第一个字符串等于第二个字符串,则返回0
    • 第一个字符串小于第二个字符串,则返回小于0的数字

通过代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char str1[] = "abcdef";
	char str2[] = "abcq";
	int ret = strncmp(str1, str2, 3);
	if (ret == 0)
	{
		printf("str1 = str2\n");
	}
	else if (ret > 0)
	{
		printf("str1 > str2\n");
	}
	else
	{
		printf("str1 < str2\n");
	}

	return 0;
}

在这里插入图片描述

strncmp函数的模拟实现

#include <stdio.h>
#include <assert.h>
int my_strncmp(const char* str1, const char* str2, size_t count)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	while (count && *str1 == *str2)
	{
		str1++;
		str2++;
		count--;
	}
	if (count == 0)
		return 0;
	else return *str1 - *str2;
}
int main()
{
	char str1[] = "abcdef";
	char str2[] = "abcq";
	int ret = my_strncmp(str1, str2, 4);
	if (ret == 0)
	{
		printf("str1 = str2\n");
	}
	else if (ret > 0)
	{
		printf("str1 > str2\n");
	}
	else
	{
		printf("str1 < str2\n");
	}
	return 0;
}

在这里插入图片描述

strstr的使用和模拟实现

char *strstr( const char *string, const char *strCharSet );
  • 在字符串 string 中查找第一次出现字符串 strCharSet 的位置,不包含终止符 ‘\0’。
  • 函数返回字符串 strCharSet 在字符串 string 中第一次出现的位置,如果未找到则返回NULL。
  • 字符串的比较匹配不包含 ‘\0’ 字符,以 ‘\0’ 作为结束标志。

通过代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char str[] = "we dont can do it";
	char* pch = NULL;
	pch = strstr(str, "dont");//找到dont字符串,并返回第一次出现的地址
	strncpy(pch, "must", 4);//将must拷贝到dont的地址处
	printf("%s", str);
	return 0;
}

在这里插入图片描述

strstr函数的模拟实现
当我们在函数模拟过程中,可能会出现几种情况:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
经过上面的分析,我们可以来模拟实现了

#include <stdio.h>
#include <assert.h>
char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 != NULL);
	assert(str2 != NULL);
	char* cp = (char*)str1;//因为上面是const char*类型不一致,需要强制类型转换
						//记录开始匹配的位置
	char* s1 = NULL;
	char* s2 = NULL;	//记录两个字符串匹配起始位置

	if (*str2 == '\0')
		return ((char*)str1);//str2为空字符串

	while (*cp != '\0')
	{
		s1 = cp;
		s2 = (char*)str2;
		while ((*s1 != '\0') && (*s2 != '\0') && (*s1 == *s2))
		{
			s1++;
			s2++;
		}
		if (*s2 == '\0')
			return cp;		//只有str2到\0才是真正匹配,才算是找到。
		cp++;
	}
	return NULL;
}
int main()
{
	char str[] = "we dont can do it";
	char* pch = NULL;
	pch = my_strstr(str, "dont");
	strncpy(pch, "must", 4);
	printf("%s", str);
	return 0;
}

在这里插入图片描述

strtok函数的使用

char *strtok( char *strToken, const char *strDelimit );
  • 字符串分割函数
  • strDelimit 参数指向一个字符串,定义了用作分隔符的字符集合
  • 第一个参数指定一个字符串,它包含了0个或者多个由 strDelimit 字符串中的一个或者多个分隔符分割的标记。
  • strtok函数找到str中的下一个标记,并将其用 ‘\0’ 结尾,返回一个指向这个标记的指针。( 注意: strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可以修改。)
  • strtok函数的第一个参数不为NULL,函数将找到str中的第一个标记,strtok函数将保存它在字符串中的位置。
  • strtok函数的第一个参数为NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记。
  • 如果字符串中不存在更多的标记,则返回NULL指针。

我们通过代码来理解:

#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = "192.168.6.112";
	char* sep = ".";
	char* str = NULL;
	str = strtok(arr, sep);
	printf("%s\n", str);
	str = strtok(NULL, sep);
	printf("%s\n", str);
	str = strtok(NULL, sep);
	printf("%s\n", str);
	str = strtok(NULL, sep);
	printf("%s\n", str);
	return 0;
}

在这里插入图片描述
其实上面的代码这样写并不好,因为我们并不知道最后分成几段,到底要输出多少段,所以我们可以通过最后找不到会返回NULL,来实现一个循环输出。

#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = "192.168.6.112";
	char* sep = ".";
	char* str = NULL;
	for (str = strtok(arr, sep); str != NULL; str = strtok(NULL, sep))
	{
		printf("%s\n", str);
	}
	return 0;
}

在这里插入图片描述

strerror函数的使用

char *strerror( int errnum );
  • 库函数在执行的时候,发生了错误会将一个错误码存放在errno这个变量中,errno是C语言提供的一个全局变量
  • strerror函数可以把参数部分错误码对应的错误信息的字符串地址返回来。
  • 在不同的系统和C语言标准库的实现中都规定了一些错误码,一般是放在errno.h这个头文件中说明的,C语言程序启动的时候就会使用一个全局变量errno来记录程序的当前错误码,只不过程序启动的时候errno是0,表示没有错误,存放在errno中,而一个错误码的数字是整数很难理解是什么意思,所以每一个错误码都是有对应的错误信息的。strerror函数就可以将错误对应的错误信息字符串的地址返回。

接下来我们来打印一下0~10这些错误码对应的信息

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
	int i = 0;
	for (i = 0; i <= 10; i++)
	{
		printf("%-2d:%s\n", i, strerror(i));
	}
	return 0;
}

在Window10+VS2022环境下输出的结果如下:
在这里插入图片描述
再来举一个例子:

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
	FILE* pFile = NULL;
	pFile = fopen("abc.txt", "r");
	if (pFile == NULL)
	{
		printf("Error opening file abc.txt: %s\n", strerror(errno));
	}
	return 0;
}

在这里插入图片描述
我们也可以了解一下perror函数

perror函数有能力直接打印错误信息的,打印的时候先打印传给perror的字符串,然后打印冒号,再打印空格,最后打印错误码对应的错误信息。

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
	FILE* pFile = NULL;
	pFile = fopen("abc.txt", "r");
	if (pFile == NULL)
	{
		perror("Error opening file abc.txt");
	}
	return 0;
}

在这里插入图片描述
好了,我们的字符函数和字符串函数就先学习到这里,希望这些知识对大家的学习有帮助!

  • 36
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值