char[] 操作
字符串的初始化方法
1、char str[10] = { 'H', 'e', 'l', 'l', 'o', '\0' };
2、char str[10] = "Hello";
char str[] = "linguoyuan\n";
字符串处理在程序中很常见,C库也提供了很多字符串的处理函数,它们有一个共同特点就是都要包含头文件string.h。
1、计算字符串长度:strlen、sizeof
头文件:string.h
格式:strlen (字符数组名)
功能:计算字符串s的(unsigned int型)长度,不包括'\0'在内
说明:返回s的长度,不包括结束符'\0'。
与此有点类似有sizeof,但它和strlen函数不一样,它只是一个操作符,而且sizeof()返回的是所声明的变量的内存大小,eg:char str[10] = "lin"; strlen(str)返回3,而sizeof(str)返回的是10。
上面是对静态数组处理的结果,如果是对指针,结果就不一样了
char* ss = "0123456789";
sizeof(ss) 结果 4 ===》ss是指向字符串常量的字符指针,sizeof 获得的是一个指针的之所占的空间,应该是
长整型的,所以是4
sizeof(*ss) 结果 1 ===》*ss是第一个字符其实就是获得了字符串的第一位'0' 所占的内存空间,是char类型的,占了 1 位
strlen(ss)= 10 >>>> 如果要获得这个字符串的长度,则一定要使用 strlen
sizeof返回对象所占用的字节大小.
strlen返回字符个数.
所以注意下面的情况:
(1)
char Array[3] = {'0'};
sizeof(Array) == 3;
char *p = Array;
sizeof(p) == 1;//sizeof(p)结果为4
在传递一个数组名到一个函数中时,它会完全退化为一个指针
(2)
char q[]="abc";
char p[]="a\n";
sizeof(q),sizeof(p),strlen(q),strlen(p);
结果是 4 3 3 2
2、复制字符串函数:strcpy、strncpy、memcpy、memmove
(1)原型:char *strcpy(char *dest,char *src);
头文件:string.h
功能:把从src地址开始且含有NULL结束符的字符串赋值到以dest开始的地址空间
说明:src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串。
返回指向dest的指针。
(2)strncpy与strcpy的不同之处就是带一个参数指定最多拷贝多少个字节。
(3)
void *memcpy(void *dest, const void *src, size_t n);
memcpy函数从src所指的内存地址拷贝n个字节到dest所指的内存地址,和strncpy不同,memcpy并不是遇到'\0'就结束,而是一定会拷贝完n个字节。注意:src和dest所指内存区域不能重叠,函数返回指向destin的指针
void *memmove(void *dest, const void *src, size_t n);
memmove也是从src所指的内存地址拷贝n个字节到dest所指的内存地址,虽然叫move但其实也是拷贝而非移动。但是和memcpy有一点不同,memcpy的两个参数src和dest所指的内存区间如果重叠则无法保证正确拷贝,而memmove却可以正确拷贝。
3、比较字符串:strcmp、memcmp、strncmp
(1)strcmp
功能:比较字符串s1和s2。
原型:strcmp(const char *s1,const char * s2)
当s1<s2时,返回值<0
当s1=s2时,返回值=0
当s1>s2时,返回值>0
两个字符串自左向右逐个字符相比(按ASCII值大小相比较),直到出现不同的字符或遇'\0'为止。
以下是一个实现strcmp的例子:
#include<stdio.h>
int mystrcpy(char s1[],char s2[])
{
int i;
for(i=0;s1[i]==s2[i];i++)
if(s1[i]=='\0')
return 0;
if(s1[i]-s2[i]>0)
return 1;
else
return -1;
}
int main(int argc,char*argv[])
{
char s1[]="linguoyuan";
char s2[]="xingwenpeng";
int i=mystrcpy(s1,s2);
printf("%d\n",i);
return 0;
}
(2)memcmp从前到后逐个比较缓冲区s1和s2的前n个字节(不管里面有没有'\0'),如果s1和s2的前n个字节全都一样就返回0,如果遇到不一样的字节,s1的字节比s2小就返回负值,s1的字节比s2大就返回正值。
(3)strncmp与strcmp的唯一区别是即使两个字符串还没遇到'\0',比较到了n个字符,也已样返回。
4、连接字符串:strcat、strncat
原型:
char *strcat(char *dest, const char *src);
char *strncat(char *dest, const char *src, size_t n);
返回值:dest指向哪,返回的指针就指向哪
strcat把src所指的字符串连接到dest所指的字符串后面
char d[10] = "foo";
char s[10] = "bar";
strcat(d, s);
printf("%s %s\n", d, s);
调用strcat函数后,缓冲区s的内容没变,缓冲区d中保存着字符串"foobar",注意原来"foo"后面的'\0'被连接上来的字符串"bar"覆盖掉了,"bar"后面的'\0'仍保留。
strcat和strcpy有同样的问题,调用者必须确保dest缓冲区足够大,否则会导致缓冲区溢出错误。strncat函数通过参数n指定一个长度,就可以避免缓冲区溢出错误。注意这个参数n的含义和strncpy的参数n不同,它并不是缓冲区dest的长度,而是表示最多从src缓冲区中取n个字符(不包括结尾的'\0')连接到dest后面。如果src中前n个字符没有出现'\0',则取前n个字符再加一个'\0'连接到dest后面,所以strncat总是保证dest缓冲区以'\0'结尾,这一点又和strncpy不同,strncpy并不保证dest缓冲区以'\0'结尾。所以,提供给strncat函数的dest缓冲区的大小至少应该是strlen(dest)+n+1个字节,才能保证不溢出。(这是宋老师书的例子)。
5、搜索字符或字符串:strchr、strrchr
原型:
char *strchr(const char *s, int c);
char *strrchr(const char *s, int c);
返回值:如果找到字符c,返回字符串s中指向字符c的指针,如果找不到就返回NULL
strchr在字符串s中从前到后查找字符c,找到字符c第一次出现的位置时就返回,返回值指向这个位置,如果找不到字符c就返回NULL。strrchr和strchr类似,但是从右向左找字符c,找到字符c第一次出现的位置就返回
char *strstr(const char *haystack, const char *needle);
返回值:如果找到子串,返回值指向子串的开头,如果找不到就返回NULL
strstr在一个长字符串中从前到后找一个子串(Substring),找到子串第一次出现的位置就返回,返回值指向子串的开头,如果找不到就返回NULL。
6、分割字符串:strtok
char *strtok(char *str, const char *delim);
参数str是待分割的字符串,delim是分隔符
以下是一个使用strtok来一个字符串分割成整数:
#include <stdio.h>
#include <string.h>
void main(void)
{
char a[1024];
int b;
char* c;
fgets(a,sizeof(a),stdin);
b = atoi(strtok(a," "));
printf("%d\n",b);
while( (c = strtok(NULL," ")) != NULL )
{
b = atoi(c);
printf("%d\n",b);
}
}
注意:
strtok函数中有一个静态指针变量记住上次处理到字符串中的什么位置,所以不需要每次调用时都把字符串中的当前处理位置传给strtok。
下面是一些平时练习字符串的例子。
////////删除字符串里的某一个字符///////////
1 #include<stdio.h>
2 void delet_char(char*str,char ch)
3 {
4 int i,j;
5 char s[3];
6 for(i=j=0;str[i];i++)
7 if(str[i]!=ch)
8 s[j++]=str[i];
9 printf("%s\n",s);
10 }
11 int main(int argc,char*argv[])
12 {
13 char ch='b';
14 char str[]="abc";
15
16 printf("%s\n",str);
17 delet_char(str,ch);
18
19 return 0;
20 }
//////////////////////////////////////////
1
2 #include<stdio.h>
3 int main(int argc,char*argv[])
4 {
5 char a,b;
6 int i=0;
7 int j;
8
9 a=getchar();
10 char buf[10];
11
12 while(a!='\n')
13 {
14 buf[i]=a;
15 a=getchar();
16 i++;
17 }
18 buf[i]='\0';
19 printf("shuru:%s\n",buf);
20
21 for(j=i-1,i=0;i<j;j--,i++)
22 {
23 b=buf[j];
24 buf[j]=buf[i];
25 buf[i]=b;
26 }
27 printf("shuchu:%s\n",buf);
28 }
//////////三种方法实现字符串复制////////////
1 #include<stdio.h>
2 #include<string.h>
3
4 int main(int argc,char*argv[])
5 {
6 char s1[]="Hello World!";
7 char s2[20];
8 int i=0;
9
10 while(s1[i])
11 s2[i++]=s1[i];
12 s2[i]='\0';
13 printf("while:%s\n",s2);
14 for(i=0;i<strlen(s1);i++)
15 s2[i]=s1[i];
16 printf("for:%s\n",s2);
17 strcpy(s2,s1);
18 printf("strcpy:%s\n",s2);
19 return 0;
20 }
/////////////////mystrncpy/////////////
1 #include<stdio.h>
2 #include<string.h>
3 #include<stdlib.h>
4 #define N 20
5 char*my_strncpy(char dest[],char src[],int count)
6 {
7 char*tmp = dest; //注意传进来的是地址
8 if(count<=0||N<count)
9 {
10 printf("Usage count or Space not enough!\n");
11 return NULL;
12 }
13 while(count&&(*tmp = *src))
14 {
15 tmp++;
16 src++;
17 count--;
18 }
19 printf("count=%d\n",count);
20 if(count)
21 while(count)
22 {
23 printf("in while,count=%d\n",count);
24 *tmp++ = '\0';
25 count--;
26 }
27 return dest;
28 }
29
30 int main(int argc,char*argv[])
31 {
32 char dest[N] = {'\0'};
33 char src[15] = "Hello!";
34 if(argc!=2)
35 {
36 printf("You need input a number!\n");
37 return 0;
38 }
39 printf("After my_strncpy,dest=%s\n",my_strncpy(dest,src,atoi(argv[1])));
40 return 0;
41 }
////////////////将字符串中的某个字母转化为大写字母//////
1 #include<stdio.h>
2 #define N 20
3 int main(int argc,char*argv[])
4 {
5 char a[N]="I like this game";
6 int num=0;
7 int i;
8 for(i=0;i<N;i++)
9 if(a[i]>='a'&&a[i]<='z'&&a[i]=='g')
10 {
11 a[i]='g'-32;
12 num++;
13 }
14 if(num)
15 printf("After converse,the new string:%s\n",a);
16 else
17 printf("Has not character'e'need to converse!\n");
18 return 0;
19 }
char[] 操作的更多相关文章
- C语言:char[]操作
C语言中char[]的操作函数1.赋值操作在C语言中,char型数组是不可以直接赋值的.例如在如下的代码中,会得到错误: char c1[20] = "women";char c2 ...
- Java中的字符串操作(比较String,StringBuiler和StringBuffer)
一.前言 刚开始学习Java时,作为只会C语言的小白,就为其中的字符串操作而感到震撼.相比之下,C语言在字节数组中保存一个结尾的\0去表示字符串,想实现字符串拼接,还需要调用strcpy库函数或者自己 ...
- Java的字符串操作
目录 Java的字符串操作 一.不同字符串操作的对比 1.1 C++中const修饰指针 const在星号的左边,是被指向的常量不可变 const在星号的右边,是指针的指向不可变 二. Java字符串 ...
- [转载]Java数组扩容算法及Java对它的应用
原文链接:http://www.cnblogs.com/gw811/archive/2012/10/07/2714252.html Java数组扩容的原理 1)Java数组对象的大小是固定不变的,数组 ...
- Java 基础知识总结 (四、String)
四.String public final class String extends Object implements Serializable, Comparable<String>, ...
- CC2540 USB Dongle 使用说明
CC2540做的USB Dongle可以烧写不同的固件从而做很多PC端的应用,下面我们来介绍下下载固件的方法和一些典型应用: 下载接口: 3V3引脚连接到CC Debugger的Target Volt ...
- 慕课网:C++ & 数据结构
课程链接:james_yuan的课程 这部分太枯燥了,如果教材难度稍大,根本就学不下去,所以就先看看通俗的视频吧. 课程目录 1.C++远征之起航篇 - C++亮点尽在其中 2.C++远征之离港篇 - ...
- String类、正则表达式
一.String类 String使用非常频繁,用来描述一个字符串.String中实现了很多 对字符串方便的操作方法. String内部使用char[]实现字符串的数据保存 字符串的&quo ...
- java学习笔记(3)——面向对象
this关键字 this是当前对象的引用,是运行期间当前对象本身. 可以使用this明确的访问当前对象的属性或者方法,类似于“我” this()可以调用本类的其他构造器,可以使用构造器的重用简化代码的 ...
随机推荐
- typedef和自定义结构体类型
在自定义结构体类型时会用到typedef关键字.大家都知道typedef是取别名的意思,在C语言中跟它容易混淆的有const,#define等,其区别不在本篇文章讨论之列. /*定义单链表结点类型*/ ...
- eclipse格式化代码末班修改
在窗口->首选项->输入format(格式)搜索,或者找Java->代码样式->格式化程序: 几个内置的不能调格式化代码风格,但是可以根据内置的新建一个,出来很多选项,开始调吧 ...
- Android -- Drawable与Bitmap测试
Drawable 以下这个是测试加载10 ...
- saltstack知识点
1.salt '*' sys.doc 显示minion设备上支持的命令 2.salt-call 可以显示比salt更多的信息,可以用来调试,检查,如果需要更详细的信息使用 salt-call -l ...
- 二、freemarker.controller半自动静态化+Tomcat虚拟资源映射
描述:本内容主要是讲2个tomcat之间同时共享一个静态话页面,统一入口是springMVC的一个controller,静态化的更新只需要传false.true.把完成的web项目放入a.b服务器To ...
- Readhat Linux5.5 安装SVNService(经验总结)
Subversion独立服务和与apache整合服务. 一 .Svn独立服务安装 操作系统: Redhat Linux5.5 安装包获取: 下载 http://subversion.tigris.or ...
- Netsharp快速入门(之1) 介绍及需求说明
作者:秋时 杨昶 时间:2014-02-15 转载须说明出处 第一章 快速入门介绍 Netsharp是一个企业基础业务管理平台,介绍Netsharp分三个系列,分别是: 1. N ...
- C#预编译指令
近日工作涉及到于外部系统交互,对方提供接口:但是在双方系统未联调时,引用外部DLL,相关类实例化,提示异常错误(错误消息正常):后面操作无法进行,那如何写调试代码,即在调试时不运行某段代码,而在正式发 ...
- Upgrading to Java 8——第四章 The Stream API
在这章中我们将学习Stream API,在JDK 8 中的一项新的特性.为了理解这一章的主题,你需要知道如何使用Lambda表达式和java.util.function里的预定义的函数式接口. 一个S ...
- hdu 1285 确定比赛名次 拓扑排序
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1285 有N个比赛队(1<=N<=500),编号依次为1,2,3,....,N进行比赛,比赛 ...