你应该知道的C语言干货(4)(strncpy,strncmp,strncat,strstr,strtok)

简介: 该库函数作用和strcpy很相似,不同点在于

#strncpy

该库函数作用和strcpy很相似,不同点在于

540441931b884cf6b3029db126118a25.png

ed01605a6a4d4b0fb4e1a7b1ab230da0.png

发现了吗,strncpy多了一个size_t num,这是要拷贝的字符数目,而strcpy是全拷贝。

废话不多说,来看代码

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
int main()
{
  char str1[30] = "XXXXXXXXXXXXXX";
  char str2[20] = "hahaha";
  strncpy(str1, str2, 7);
  return 0;
}

d33e9a4621c44daa864556856c96873b.png

如果说num小于str2的长度,正常复制

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
int main()
{
  char str1[30] = "XXXX";
  char str2[20] = "hahaha";
  strncpy(str1, str2, 5);
  return 0;
}


9bbd972ff5c3487cb8f4686f9482d002.png

因为初始化时,剩下的空间都初始化了,如果石str1未初始化,就会是这样:

cdfb1ea97f4e4c2888ad28e196aa3743.png

str2末尾没有\0,还需要自己再加,如果全部复制,不初始化str1也ok,但最好初始化

如果要复制的长度大于str2,也就是源字符串,那么多出来的部分拷贝\0


#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
int main()
{
  char str1[30] = "XXXXXXXXXXXXXXXXXXX";
  char str2[20] = "hahaha";
  strncpy(str1, str2, 10);
  return 0;
}


5aed804fdca744b19fdce3b60b9fe5a2.png

补充一点,二者copy时遇到\0都会停下来,strncpy遇到后超过部分还补\0

388ff71c4c8141508d54ba9c1a20dd28.png

#strncmp

7ca3ce9116be4397b5261379b12c5568.png

38b69c8c16b040649c04b096ffd63797.png

 这个和上面类似,比较num个字符,直到这n个字符都比完,比较方式同strcmp函数


#strncat

eee27cbc9ace47d798a3345f386a4e89.png

0c2d0ff98b4148178d78c5ddff31409a.png

同样的道理,向str1的末尾追加num个以str2为首地址开始的字符 ,但是有一些和strncpy不同的区别,看图:

b56699f7e1584bdaad187ca3e2190369.png

这个很正常,都可以理解,接下来看好了:

400b067837424b09aaa88f9755ca5a1e.png

哈?我明明只追加6个字符,没追加最后的\0啊?

是的 ,strncat不管追加多少字符,都会在追加的最后自己加上\0,那么这样呢?

415855565e1942d0ae5dee26f2c4e41b.png

即使超过源字符串长度,他不会像strncpy那样补\0,他只追加一个\0


#strstr

该库函数作用是在一个字符串中找另一个字符串,并返回那个找到的字符串的首位置,若没有找到,返回NULL。

接下来我们模拟实现他

(这里最好的算法是KMP算法,在数据结构里会讲到,这里没有用到)

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
const char* my_strstr(const char* str_orign, const char* str_find)
{
  const char* orign = str_orign;
  const char* p1 = orign;
  const char* p2 = str_find;
  while (*orign)
  {
    p1 = orign;
    p2 = str_find;
    while (*p1 && *p2 && (*p1 == *p2))
    {
      p1++;
      p2++;
    }
    if (*p2 == '\0')
      return orign;
    orign++;
  }
  return NULL;
}
//模拟实现strstr
int main()
{
  char str_orign[30] = "da wu wo x ni wo......";
  char str_find[20] = "woo";
  const char* p = my_strstr(str_orign, str_find);
  if (p != NULL)
    printf("%s", p);
  else
    printf("Can not find it\n");
  return 0;
}

#strtok

ee96e5e51b9e4261b9abff8a8392714f.png

这是个用来分割字符串的函数,sep是存放分隔符的,没有先后顺序之分,str则是我们要分割的字符串,看代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
int main()
{
  char str[30] = "haha.hehe#hiahia*";
  char sep[10] = ".#*";
  printf("%s", strtok(str, sep));
  return 0;
}

182bcb019fff4d609dc0821f50bd3168.png

咦,奶奶个腿滴,感觉咋不对劲呢,为什么没有全部分割呢?


接下来讲讲这个函数的原理:

strtok函数会找到str中的下一个标记,并将其末尾用\0结尾,也就是将分隔符换成\0,返回一个指向这个标记的指针,也就是说,我们第一次使用strtok时,返回的标记指向haha的h,然后我们打印时遇到\0停止:

43622f9ef4714c04b92cad472794e018.png

所以我们打印出来haha,那么如何分割剩下的字符串呢?

接下来看代码,这么用:

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

接下来传的就是NULL空指针,函数将从我们第一次使用后保存的位置开始查找下一个标记,找到后将其末尾分隔符换成\0,返回标记,看这次我们的打印结果:

793dd2bdb8b64ba2b629edee5a56af4d.png

分隔成功~  


#下期预告

内存函数:

memset

memcpy

memmove

memcmp

目录
相关文章
|
3月前
|
安全 编译器 C语言
C语言学习记录——字符串相关函数及部分模拟(strcmp、strncmp、strncat、strncpy、strstr、strtok、strerror)
C语言学习记录——字符串相关函数及部分模拟(strcmp、strncmp、strncat、strncpy、strstr、strtok、strerror)
28 1
|
4月前
|
C语言
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)(下)
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)
34 0
|
4月前
|
C语言
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)(上)
C语言进阶21收尾(编程练习)(atoi,strncpy,strncat,offsetof模拟实现+找单身狗+宏交换二进制奇偶位)
103 0
|
4月前
|
C语言
C语言进阶⑬(字符串函数)+(指针编程题)strlen+strcpy+strcat+strstr+strtok+strerror(下)
C语言进阶⑬(字符串函数)+(指针编程题)strlen+strcpy+strcat+strstr+strtok+strerror
18 0
|
4月前
|
安全 C语言
C语言进阶⑬(字符串函数)+(指针编程题)strlen+strcpy+strcat+strstr+strtok+strerror(中)
C语言进阶⑬(字符串函数)+(指针编程题)strlen+strcpy+strcat+strstr+strtok+strerror
36 0
|
2天前
|
存储 C语言
C语言程序设计核心详解 第十章:位运算和c语言文件操作详解_文件操作函数
本文详细介绍了C语言中的位运算和文件操作。位运算包括按位与、或、异或、取反、左移和右移等六种运算符及其复合赋值运算符,每种运算符的功能和应用场景都有具体说明。文件操作部分则涵盖了文件的概念、分类、文件类型指针、文件的打开与关闭、读写操作及当前读写位置的调整等内容,提供了丰富的示例帮助理解。通过对本文的学习,读者可以全面掌握C语言中的位运算和文件处理技术。
|
2天前
|
存储 C语言
C语言程序设计核心详解 第七章 函数和预编译命令
本章介绍C语言中的函数定义与使用,以及预编译命令。主要内容包括函数的定义格式、调用方式和示例分析。C程序结构分为`main()`单框架或多子函数框架。函数不能嵌套定义但可互相调用。变量具有类型、作用范围和存储类别三种属性,其中作用范围分为局部和全局。预编译命令包括文件包含和宏定义,宏定义分为无参和带参两种形式。此外,还介绍了变量的存储类别及其特点。通过实例详细解析了函数调用过程及宏定义的应用。
|
8天前
|
Linux C语言
C语言 多进程编程(三)信号处理方式和自定义处理函数
本文详细介绍了Linux系统中进程间通信的关键机制——信号。首先解释了信号作为一种异步通知机制的特点及其主要来源,接着列举了常见的信号类型及其定义。文章进一步探讨了信号的处理流程和Linux中处理信号的方式,包括忽略信号、捕捉信号以及执行默认操作。此外,通过具体示例演示了如何创建子进程并通过信号进行控制。最后,讲解了如何通过`signal`函数自定义信号处理函数,并提供了完整的示例代码,展示了父子进程之间通过信号进行通信的过程。
|
8天前
|
C语言
C语言 字符串操作函数
本文档详细介绍了多个常用的字符串操作函数,包括 `strlen`、`strcpy`、`strncpy`、`strcat`、`strncat`、`strcmp`、`strncpy`、`sprintf`、`itoa`、`strchr`、`strspn`、`strcspn`、`strstr` 和 `strtok`。每个函数均提供了语法说明、参数解释、返回值描述及示例代码。此外,还给出了部分函数的自实现版本,帮助读者深入理解其工作原理。通过这些函数,可以轻松地进行字符串长度计算、复制、连接、比较等操作。
|
9天前
|
SQL 关系型数据库 C语言
PostgreSQL SQL扩展 ---- C语言函数(三)
可以用C(或者与C兼容,比如C++)语言编写用户自定义函数(User-defined functions)。这些函数被编译到动态可加载目标文件(也称为共享库)中并被守护进程加载到服务中。“C语言函数”与“内部函数”的区别就在于动态加载这个特性,二者的实际编码约定本质上是相同的(因此,标准的内部函数库为用户自定义C语言函数提供了丰富的示例代码)