各位csdn的友友们好,上次阿博给大家讲了一些简单的字符串函数的功能和模拟实现,今天就和阿博一起再上一个台阶继续拿捏它们
它的功能就是在str1中找str2,而且是第一次出现的位置,如果没有找到,则返回的是空指针,如果找到了str2,并且找到多次,它只返回第一次出现的地址
代码测试
#include
#include
int main()
{
char arr1[] = "abcdebcdf";
char arr2[] = "bcd";
//char arr2[]="bcq";
char*p= strstr(arr1, arr2);
if (p == NULL)
{
printf("找不到\n");
}
else
printf("%s\n", p);
return 0;
}
这就是strstr相关的功能了,如果找到了,友友们一定要切记返回的是第一次出现的地址.
这个函数的功能是在一个字符串中找字符第一次出现的位置.
这个函数的功能是找一个字符在字符串中最后一次出现的位置.
2.strstr函数的模拟实现
#include
#include
#include
char* my_strstr(const char* str1, const char* str2)
{
assert(str1 && str2);
char* s1 = NULL;
char* s2 = NULL;
char* cp = (char*)str1;
while (*cp)
{
s1 = cp;
s2 = (char*)str2;
while (*s1 && *s2 && *s1 == *s2)
{
s1++;
s2++;
}
if (*s2 == '\0')
{
return cp;
}
cp++;
}
return NULL;
}
int main()
{
char arr1[] = "abcdebcdf";
//char arr2[] = "bcd";
char arr2[]="bcd";
char*p= my_strstr(arr1, arr2);
if (p == NULL)
{
printf("找不到\n");
}
else
printf("%s\n", p);
return 0;
}
1.delimiters参数是个字符串,定义了用作分隔符的字符集合
2.第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标记.
3.strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可修改
4.strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置.
5.strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标记。如果字符串中不存在更多的标记,则返回 NULL 指针.
代码测试1
//strtok的功能介绍
int main()
{
char arr[] = "[email protected]";
char buf[30] = { 0 };
strcpy(buf, arr); //这里是在buf文件里面操作,防止原文件被改变
const char* p = "@.";
char*str=strtok(buf, p);
printf("%s\n", str);
str = strtok(NULL, p);
printf("%s\n", str);
str = strtok(NULL, p);
printf("%s\n", str);
return 0;
}
这种写法没有错,但是如果分隔符比较多的话,我们需要写很多行代码,所以我们换一种方法.
int main()
{
char arr[] = "[email protected]";
char buf[30] = { 0 };
strcpy(buf, arr); //这里是在buf文件里面操作,防止原文件被改变
const char* p = "@.";
char* str = NULL;
for (str = strtok(buf, p); str != NULL; str = strtok(NULL, p)) //第一次我们传buf的地址,然后第二次,第三次,第四次......我们都传空指针的地址,到最后没有分割符了,str为空,跳出循环.
{
printf("%s\n", str);
}
return 0;
}
4.strerror和perror的功能介绍
strerror
代码测试
int main()
{
FILE* pf = fopen("text.txt", "r");
if (pf == NULL)
{
printf("打开文件失败,原因是:%s\n", strerror(errno));
return 1;
}
fclose(pf);
pf = NULL;
return 0;
}
这里阿博给友友们解答一下疑惑,其实c语言的库函数在调用失败的时候,会将一个错误码存放在一个叫:errno的变量中,当我们想知道调用库函数的时候产生了什么错误信息,就可以将errno中的错误码翻译成错误信息.
友友们注意,我们一定要把这个文件扩展名给打开,这样我们就可以完整的看到文件名了.
int main()
{
FILE* pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("打开文件失败"); //其实就可以把perror抽象的理解成printf+serror
return 1;
}
fclose(pf);
pf = NULL;
return 0;
}
void test1()
{
int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
int arr2[10] = { 0 };
memcpy(arr2, arr1, 20); //注意这里是字节个数,不是元素个数
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr2[i]);
}
}
void test2()
{
float arr1[] = { 1.0f,2.0f,3.0f,4.0f,5.0f };
float arr2[10] = { 0 };
memcpy(arr2, arr1, 12);
int i = 0;
for (i = 0; i < 10; i++) //注意这里是字节个数
{
printf("%f ", arr2[i]);
}
}
int main()
{
//test1();
test2();
return 0;
}
代码实现
#include
void* my_memcpy(void* dest, const void* src, size_t num)
{
assert(dest && src);
void* ret = dest;
while (num--)
{
//*dest = *src;
//dest++;
//src++; //err
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
return ret;
}
void test1()
{
int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
int arr2[10] = { 0 };
my_memcpy(arr2, arr1, 20); //注意这里是字节个数,不是元素个数
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr2[i]);
}
}
️️️友友们注意虽然void*的指针可以接收任意类型的数据,但是这种指针不能直接解引用和加减运算.️️️
7.内存重叠
代码测试
#include
void* my_memcpy(void* dest, const void* src, size_t num)
{
assert(dest && src);
void* ret = dest;
while (num--)
{
//*dest = *src;
//dest++;
//src++; //err
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
return ret;
}
void test1()
{
int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
// 1,2,1,2,3,4,5,8,9,10 我们期待的结果
my_memcpy(arr1+2, arr1, 20); //注意这里是字节个数,不是元素个数
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr1[i]);
}
}
void test1()
{
int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
// 1,2,1,2,3,4,5,8,9,10 我们期待的结果
memmove(arr1+2, arr1, 20); //注意这里是字节个数,不是元素个数
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr1[i]);
}
}
代码实现
#include
void* my_memmove(void* dest, const void* src, size_t num)
{
assert(dest && src);
void* ret = dest;
if (dest < src)
{
while (num--)
{
//*dest = *src;
//dest++;
//src++; //err
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
}
else
{
while (num--)
{
*((char*)dest + num) = *((char*)src + num); //这里的num已经是19了
}
}
return ret;
}
void test3()
{
int arr1[] = { 1,2,3,4,7 };
int arr2[] = { 1,2,3,4,6 };
int ret = memcmp(arr1, arr2, 17);
printf("%d\n", ret);
}
11.memcmp函数的模拟实现
代码实现
#include
int my_memcmp(const void* str1, const void* str2,size_t num)
{
assert(str1 && str2);
while (num--)
{
if(*(char*)str1!=*(char*)str2)
return *(char*)str1 - *(char*)str2;
str1 = (char*)str1 + 1;
str2 = (char*)str2 + 1;
}
return 0;
}
void test3()
{
int arr1[] = { 1,2,3,4,7 };
int arr2[] = { 1,2,3,4,6 };
int ret = my_memcmp(arr1, arr2, 16);
printf("%d\n", ret);
}
int main()
{
//test1();
//test2();
test3();
return 0;
}
void test4()
{
char arr[] = "hello world";
memset(arr, 'x', 5);
printf("%s\n", arr);
}
友友们注意memset是以字节为单位设置的
错误示范
void test5()
{
int arr[10] = { 0 };
memset(arr, 1, sizeof(arr)); // 01 01 01 01,这种写法无法将每个元素设置为1
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr[i]);
}
}
好了友友们,本期创作到这里就告一段落了,如果感觉有帮助的话,可以给阿博点个关注哦,后续会继续给友友们带来一些新的干货