求字符串长度
字符串以’\0’作为结束标志,strlen返回的是’\0’之前出现的字符个数,不包含’\0’。
注意函数的返回值是size_t类型的,是无符号的
下面就有这样一个例子
#include
#include
int main() {
if (strlen("abc") - strlen("abcdef") > 0)
printf("大于\n");
else
printf("小于等于\n");
return 0;
}
这个代码的输出结果是大于,就是因为它的返回类型是无符号的,算出来的-3的补码会被当成一个无符号数的原码,而这个无符号数是非常大的。
改进方法是强制类型转换或者直接让两个值进行比较。
模拟实现
size_t my_strlen(const char* str) {
size_t count = 0;
while (*str) {
count++;
str++;
}
return count;
}
size_t my_strlen(const char* str) {//指针相减
char* start = str;
while (*str) {
str++;
}
return str - start;
}
size_t my_strlen(const char* str) {//递归实现
if (*str == '\0')
return 0;
else
return my_strlen(str + 1) + 1;
}
将源头的字符串拷贝到目的地,包括’\0’
注意:1.源头的字符串必须包含’\0’,否则会发生错误。因为只有遇到’\0’时,拷贝才停止。
2.目标空间必须足够大,以确保能存放源字符串。
3.目标空间必须可变,即不能是一个常量字符串。
模拟实现
#include
char* my_strcpy(char* dest, const char* src) {
char* ret = dest;
assert(dest);
assert(src);
while (*dest++ = *src++) {
;
}
return ret;
}
追加源头的字符串到目的地的末尾,会覆盖掉目的地末尾的’\0’
注意:1.源字符串必须以’\0’结束,为了知道我们要追加多少,目的地字符串也要有’\0’,为了知道追加到那。
2.目标空间必须足够大,能容纳下源字符串的内容。
3.目标空间必须可修改。
4.strcat不能自己给自己追加
模拟实现
char* my_strcat(char* dest, const char* src) {
char* ret = dest;
assert(dest);
assert(src);
while (*dest)
dest++;
while (*dest++ = *src++) {
;
}
return ret;
}
比较两字符串的大小
模拟实现
int my_strcmp(const char* str1, const char* str2) {
assert(str1&&str2);
while (*str1 == *str2) {
if (*str1 == '\0')
return 0;
str1++;
str2++;
}
if (*str1 > *str2)//return (*str1-*str2);
return 1;
else
return -1;
}
上面四个函数都属于长度不受限制的字符串函数,就是干到’\0’为止,下面三个是长度受限制的字符串函数,会规定操作的字符个数(多了一个参数)。
只拷贝count个字符过去,如果count大于source中的字符个数,会拿’\0’补齐。但是strncat追加时遇到这种情况source中有几个字符就追加几个字符,最后在补个’\0’就完事了,它跟strncpy还是不同的。
当count小于source中字符个数时,就追加count个字符,最后补’\0’,因为要保证追加完后仍然是个字符串。
就是在count个字符内比较大小,后面是不管的。
这个函数是在字符串中寻找子字符串,如果找到了就返回子字符串第一次出现的位置,没找到就返回NULL
基本的使用是这样的
int main() {
char arr1[] = "abcdefabcdef";
char arr2[] = "qwe";
char* ret = strstr(arr1, arr2);
if (ret != NULL)
printf("%s\n", ret);
else
printf("找不到\n");
return 0;
}
模拟实现
char* my_strstr(char* str1, char* str2) {
char* cp = str1;
char* s1 = cp;
char* s2 = str2;
if(*str2=='\0')
return str1;
while (*cp) {
s1 = cp;
s2 = str2;
while (*s1 && *s2 && *s1 == *s2) {
s1++;
s2++;
}
if (*s2 == '\0')
return cp;
cp++;
}
return NULL;
}
这个函数简单而言就是将第一个字符串中的内容划分成一段一段的,它会把第一个字符串中的第二个字符串中出现的字符转化为’\0’,从而实现分段。基本使用是这样的:
int main() {
char arr[] = "abcdef@ghijk#lmn$opqrst";
char copy[50] = { 0 };
strcpy(copy, arr);
char sep[] = "@#$";
char* ret = NULL;
for (ret = (strtok(arr, sep)); ret != NULL; ret = strtok(NULL, sep)) {
printf("%s\n", ret);
}
return 0;
}
我们需要将要处理的数组拷贝一份,因为这个函数的使用确实会改变数组本身,就是会把对应的位置改成’\0’。
库函数在执行的时候,发生了错误,会将一个错误码存放在errno这个变量中,errno是C语言提供的一个全局变量。而错误码的可读性是很差的,比如404比较常见,还有很多,我们把错误码转化为文字信息是比较重要的,所以就有这么一个函数。它的返回类型就是错误码的首元素位置,于是就可以打印出来。
比如说:打印0-9的错误信息
我们在遇到库函数执行错误的时候可以直接打印就行,变量就传errno就行。