c语言实战开发(来查字符了)2019-12-06 Day11

本文介绍了C语言中指针与函数的关系,强调指针用于间接访问内存,以及何时在函数中使用指针参数。接着探讨了字符串的定义,包括字符数组和字符指针的区别,并提醒不能直接用字符指针接收字符串。最后,通过一个多文件的示例展示了如何查找字符串和单词在输入语句中出现的次数,强调了在解决复杂问题时需要注意的细节和思考方式。
摘要由CSDN通过智能技术生成

内容

1:指针和函数的关系
2:字符串定义
3:查字符demo(多文件的应用)

一.指针和函数的关系

1:指针最主要的作用:通过指针间接访问某个内存。
2:函数在什么情况下参数是指针?就需要考虑这个函数会不会改变传递过来的数据本身。需要改变外部的值的时候,就将这个变量定义为指针变量。

代码示例如下

#include<stdio.h>
#include<stdlib.h>
void test (int *pa,int *pb);//c语言中函数有了指针,基本就不需要返回值了 
int main (void){
    int a = 10;//如果不让a的值改变,则将接收a的改成const int *pa即可,即变成常量指针
    int b = 20;
    test(&a,&b);//直接取地址,不用定义指针变量装着然后再放进去 
    printf("%d,   %d", a ,b);
    system("pause");
    return 0;
} 
void test (int *pa,int *pb){
    (*pa)++;
    (*pb)++;
}
#include<stdio.h>
#include<stdlib.h>
void test2 (int *p,int count); //接收数组首地址和数组元素个数,因为函数里面是通过指针变量来接收数组的地址,而指针变量无法确定指向的内容的大小。 
//访问数组有两种方式,第一 num[3],第二 *(num+3)这两个是一回事 
int main (void){
    int num[] = {1,2,3,4,5};//将数组每个元素都加1,则需要用指针 
    test2(num,5);
        for(int i = 0;i < 5;i++){
        //printf("%d\n",num[i]);
    }
    system("pause");
    return 0;
} 

void test2 (int*p,int count) {//这里的int *p 可以改成int p[];使用起来都一样.
//数组做形参,本质上是地址,编译器就默认将int p[]转化为 int*p了. 
    for(int i = 0;i < count;i++){
    
        p[i]++;// p[i]是指访问地址在p[1] 的那个地方的值,也就是说这里它的类型就变了,在这里就变成整形了 。而p+i就是地址类型,和前面的就不一样
    }
}

由于这一条比较重要,所以再重复一遍:
p[i]是指访问地址在p[1] 的那个地方的值,也就是说这里它的类型就变了,在这里就变成整形了 。而p+i就是地址类型,和前面的就不一样

接收终端输入的字符串,然后输出这个字符串,用函数。

#include<stdio.h>
#include<stdlib.h>

void shuru (char*p);//注意接收的是字符串的首地址
int main (void){
    char name[100] = {};
    shuru(name);
    printf("%s",name);
    system("pause");
    return 0;
} 
void shuru (char*p){
        scanf("%s",p);//这里直接用%s就可以,而不用%c放在for循环里面。而且%s会默认加\0,而%c不会
}

还有一种写法,就是用%c

#include<stdio.h>
#include<stdlib.h>
void shuru (char*p);
int main (void){
    char name[100] = {};//这里用了初始化,所以除了赋值的全是\0,所以下面的p[i] = '\0'可以不写
    shuru(name);
    printf("%s",name);
    system("pause");
    return 0;
} 
void shuru (char*p){
        int i = 0;//用来记录写到哪一个了 
        while(1){
            char c = getchar();//从终端接收一个字符
            if(c == '\n'){
                //需要在数组里面添加一个\0
                p[i] = '\0' ;//将最后一个变成\0 ,这个自己想不到的
                break;
            } else{
                p[i] = c;
                i++;
            }
            
        }
}

二.字符串定义

字符串就==字符数组下面三种定义方式都是可以的

char a[10] = "hello";
char b[] = "hello";
char *c = "hello";

字符指针也可以理解成字符串,而字符数组有连续的内存空间,而字符指针比如这里的c只是存了一个地址,而没有像a或者b那样去存内容。所以注意不能直接字符指针来接收字符串,字符指针只为这个变量分配8个自己内存空间,这个空间只能用来存地址,不能存具体的数据

三.小demo

(以多文件形式来写)
从终端输入一个语句,查找某个字符串出现的次数。比如hello jack,hello,jac!其中jac出现了1次
然后查找某个单词出现的次数。
请输入字符串,
请输入查找的字符串
请输入查找的单词
提示:scanf遇到空格和\n退出,scanf不能输入带空格的字符串
只能自己定义一个输入语句的方法

QQ图片20191206204407.jpg

头文件

void input (char *p,char *des);
int find(char *sentence,char *word);//因为最后要计算个数,所以这个函数要有返回值
int length(char *p) ;
void myScanf (char *p);

实现文件

#include<stdio.h>
#include<stdlib.h>
#include"myString.h"
void myScanf (char *p){
    int i = 0;
    while(1){
        char c = getchar();
        if(c == '\n'){
            p[i] = '\0';
            break;
        }
        p[i] = c;
        i++;
    }
}
void input (char *p,char *des){
    //提示用户操作 
    printf("%s:",des);
    //scanf("%s",p);//直接输入到地址里面就可以 %s遇到空格或者回车符表示结尾,所以如果你输入hello world,scanf会把他当成两个语句来处理 
    //只能自己写一个输入语句的方法 
    //输入语句 
    myScanf(p);
    
}
//计算字符串的长度 
int length(char *p){
    int i = 0;
    for(i = 0;p[i] != '\0';i++){
        //里面啥事都不用做 
    }
    return i;
}
int find(char *sentence,char *word){
    //1.获取两个字符串的长度(找到\0),用一个方法 
    int sLength = length(sentence);
    int wLength = length(word);
    //2.判断查询的字符串长度是否比句子短 
    if(sLength < wLength){
        return 0;
    }
    //3.开始用两个循环查找
    int start = 0;
    int count = 0;
    for(int i =0;i <sLength;i++){
        //记录当前开始比较的初始位置
        start = i;
        //从当前位置开始去和查找的单词进行比较
        int j =0; 
        for(;j<wLength;j++){
            //判断j对应的值和start+j进行比较。尤其注意这里是start+j。也就是偏移量 
            if(sentence[start+j] != word[j]){
                //出现了不相同的,当前比较结束,直接退出
                break;
                //然后i++,再start = i,那就指向了下一个i了 
            }
        }
        //判断怎么出来的
        //如果都相同 
        if(j == wLength){
            //start也发生了变化,将start位置到i的下一个 
            i =  start + j ;
            count++;
        }
    } 
    
return count; 
}

主函数

#include<stdio.h>
#include<stdlib.h>
#include"myString.h"
//两个语句有两种情况
//jksndkjasbfjkab
//sdsbdkjabd 
//与klsnajkcsnjlncsa
//nsadkjsackjsacjlsacjsaadsjla
//第二种情况就没必要做操作了 
int main(void){
    char sentence[100] = {};
    char word [20] = {};
    input (sentence,"请输入语句");
    input(word,"请输入查找的单词");
    int count = find(sentence,word);
    printf("出现%d次",count);
    system("pause");
    return 0;
}

下面是自己写的

#include<stdio.h>
#include<stdlib.h>
//定义句子和查找句子的数组 
char  sentence[100] = {};//这里是sentence[]而不是*sentence[] 
char  word[100] = {};
void myScanf(char*p);
int cLength(char*p);
void isCompared (int a,int  b);
int Find (char *p1,char *p2);
int main (void){
    /*//定义句子和查找句子的数组 
    char  sentence[100] = {};//这里是sentence[]而不是*sentence[] 
    char  word[100] = {};*/
    //定义函数,输入两个数组
    myScanf(sentence);
    myScanf(word);
    //计算长度,如果两者长度关系合适,则进行比较,如果不合适,则退出
    int sLength = cLength(sentence);
    int wLength = cLength(word);
    // 如果两者长度关系合适,则进行查找,如果不合适,则退出
    isCompared (sLength,wLength);
    //进行查找
    //打印个数
    int shumu = Find (sentence,word);
    printf("%d\n",shumu); 
    
    
    
    return 0;
}
void myScanf(char*p){
    char c;
    int i = 0; 
    printf("请输入:\n");
    while(1){
        c = getchar();//接收字符
        if(c == '\n'){
            //如果遇到回车,则退出,并且将最后一个元素赋为'\0'
            p[i] = '\0';
            break; 
        } else{
            //printf("赋值一次\n"); 
            p[i] = c;
            i++;
        }
    }
}
int cLength(char*p){
    //用i来记录个数 
    int i = 0;
    for(i = 0;p[i] != '\0';i++){//这里是p[i] != '\0'而不是i != '\0',一开始这里错了就没看出来 
    }//这里什么都不用做
    return i; 
} 
void isCompared (int a,int b){
    while(1){
        if(a < b){
        printf("输入字符串长度不合适\n");
        myScanf(word); 
    }else{
        break;//如果数目大小不合适则重新输入,合适就退出。 
    }   
    }
    
}
int Find (char *p1,char *p2){
    printf("进入查找函数\n");
    //计算长度,如果两者长度关系合适,则进行比较,如果不合适,则退出
    int sLength = cLength(sentence) - 1;//把\0去掉,这里一开始忘了 
    int wLength = cLength(word) - 1;
    int start = 0;//定义起始位置
    int i = 0;//定义大字符串查找起始位置
    int j = 0;//定义小字符串查找起始位置
    int count = 0;//存储相同个数 
    for(i = 0;i < sLength ;i++){
        start = i;
        for(j = 0;j < wLength;j++){ 
            //printf("进入循环");
            if(p1[start + j] != p2[j]){
                break; 
            }
        }
        //看怎么退出的循环
    if(j == wLength){//找到一个正常退出的循环 
        //start = start + wLength;//更新start位置
        //printf("进入循环");
        i = start + j; 
        count++;//数目加1 
    }
    /*if(start == sLength - 1){
        printf("进入循环");
        break;
    }*/
        
    } 
    return count;
}

总结

今天学到的知识中有两个亮点
1:p[i]就相当于转类型了,详见上
2:烧脑的demo书写。这个demo上课听得有点懵逼,下课写还是很多bug,纠正之后还是很多bug,最后休息了一阵,当天下午终于把bug找全了。在这个过程中收获了很多细节的知识(详见注释),最主要还是不要一根筋,光想着现在就必须写完写对,说不定出去走一圈回来就会了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值