icoding复习3

1. 不调用库函数,自己实现字符串的比较操作:该操作当比较的两个字符是都是字母,且两个字符互为大小写
(如a和A、e和E)时认为两个字符相同,否则不同,其比较结果按这两个字符的原值确定。函数的返回值规定如下:
返回值 < 0:第一个不匹配的字符在 ptr1 中的值低于 ptr2 中的值
返回值 == 0:两个字符串的内容相等
返回值 > 0:第一个不匹配的字符在 ptr1 中的值大于在 ptr2 中的值

int str_compare(const char* ptr1, const char* ptr2);
#include
#include
#include "dsstring.h" //请不要删除,否则检查不通过
//又是字符串操作!!!! 
int str_compare(const char* ptr1, const char* ptr2){
    int i;
    // a == 97 A == 65 
    for(i = 0; ptr1[i] && ptr2[i]; i++){
        if(ptr1[i] != ptr2[i]){//字符相等就继续,不等判断大小写关系 
            if(ptr1[i] + 'a' - 'A' == ptr2[i] || ptr2[i]+'a'-'A' == ptr1[i])
                continue;//这一步可以分开写,没有判断是否ptr1[i]和ptr2[i]为字母,但是icoding监测可以通过
                //其实当两个字符恰好相差32就不行了 
            else
                return ptr1[i]-ptr2[i];
        }
    }
    //后面的代码可以略掉 
    if(ptr1[i])
        return (int)ptr1[i];
    else
        return (int)ptr2[i];
     
}
//解法2 icoding检测数据不全 ! 100
int str_compare(const char* ptr1, const char* ptr2)
{
    // a == 97 A == 65 
    int i;
    for (i = 0; ptr1[i] != '\0' || ptr2[i] != '\0';) {
        if (ptr1[i] == ptr2[i] )  //判断字符是否相等 
            i++;
        else if(ptr1[i] <= 'z' && ptr1[i] >= 'a'&& ptr1[i] - 32 == ptr2[i])
        //判断是否为大小写不同的相同字母, (如果同为相同的大写或者相同的小写之前检测了), 
            i++;
        else if(ptr2[i] <= 'z' && ptr2[i] >= 'a'&& ptr2[i] - 32 == ptr1[i])
            i++;
        else
            return ptr1[i] - ptr2[i];//可以自动转换为int
        //注意点第二三个if是必要的, 不能直接判断ptr1[1] -ptr2[i] == +-32 , 也有可能有ASC码正好相差32的非字母字符 
    }
    return 0;
}

//90+问题代码
int str_compare(const char* ptr1, const char* ptr2)
{
    char a, b;
    int i;
    for (i = 0; ptr1[i] != '\0' && ptr2[i] != '\0'; i++) {
        a = ptr1[i];
        b = ptr2[i];
        if (a <= 'Z' && a >= 'A' && b <= 'z' && b >= 'a')
            a = a - 'A' + 'a';//有问题, 这部分的意思是a为大写, b同时也为字母并且为小写 , 那么
            //a转化为小写并且与b比较, 但是如果值不等的话会改变返回值大小 
        if (b <= 'Z' && b >= 'A' && a <= 'z' && a >= 'a')
            b = b - 'A' + 'a';
        if (a == b)
            continue;
        else
            return a - b;
    }
    return ptr1[i] - ptr2[i];
}

2.串替换

in,原始字符串,保持不变; out, 存放替换结果的字符串; outlen,out空间的大小
oldstr,要替换的旧字符串; newstr,替换成的新字符串
函数返回成功替换的次数,即有多少个子串被成功替换
在替换过程中,任何情况下所得字符串(及结束符)不应该超过 outlen,
如果某次替换所得字符串的长度超过 outlen,则不进行这次替换操作,整个替换操作结束。如:
原始串为 "aaabbbccc",outlen 为14, oldstr 为 "c",newstr 为 "333" 时,
两次替换后得 "aaabbb333333c",此时字符串占用空间为14字节。
如果再进行替换,则会超出 out 所占用的空间,所以停止替换操作。
此时函数应该返回 2, out指向的串为 "aaabbb333333c"
再如:原始串为 "aaabbbccc",outlen 为10, oldstr 为 "bb",
newstr 为 "123456",进行替换后所得的串应该为 "aaa123456" 
(长度为9)与结束符一共占 10 个字节,此时函数应该返回 1

#include "dsstring.h"
#include
#include

int get_len(const char *str){
    int i;
    for(i = 0; str[i]; i++)
        ;
    return i;
}
//另一种写法
int get_len(const char* s)
{
    int i = 0;
    while (*(s + i)) {
        i++;
    }
    return i;
    //返回值是实际长度, 不包括空字符
}

//理解:旧串换新串,每一个新串要求能够全部放下并且不忽略尾部空字符, 操作的是字符指针...! 
int str_replace(const char* in, char* out, int outlen, const char* oldstr, const char* newstr)
{
    int i, j = 0, ostr, nstr = 0;//i 指示in, j指示out. 
    int n = 0;
    
    if(outlen <= 0) return false; 
    
    //out需要分配空间??? 不需要..... 
    //下面思路整理:
    //in的字符不与oldstr第一个字符匹配,直接复制, 进入下一轮循环 
    //判断剩余空间是否足够, 一种溢出是新串字符加out内已有字符长度溢出,一种是in剩余字符加out已有字符溢出,直接剩余全部复制
    //空间足够,判断字符是否匹配,判断匹配成功,那么ostr指向oldstr最后一个'\0' 
    //执行换串操作 ,n++ 
    //最后如果串换完并且in中字符全部复制完但是j没有达到outlen-2,放空字符 
    for(i = 0; i < get_len(in) && j < outlen - 1;){
    //对于j,总长度为outlen, 最后一个下标为outlen-1,最后一个位置放'\0' ,所以j最大取outlen-2 
    
        if(in[i] != oldstr[0]){
            out[j++] = in[i++];
            continue; 
        }
            
        if(j + get_len(newstr) >= outlen - 1 || get_len(in) - i + j >= outlen - 1)
        {//先看剩余空间可以换串吗 ,如果不能就剩余全部复制 
            for(; j < outlen;)
                out[j++] = in[i++];l
            return n;
        }
        
        for(ostr = 0; otsr < get_len(oldstr); ostr++)
            if(oldstr[ostr] != in[i+ostr])
                break;
                
        if(ostr == get_len(oldstr) - 1){//if(!(oldstr[ostr]))等价 
            for(nstr = 0; j < get_len(newstr); j++)
                out[j] = newstr[nstr++];
            n++;
            i += get_len(oldstr);
        }
        else
            out[j++] = in[i++]; 
    }
    
    for(; j < outlen - 1; j++)
        out[j] = '\0';
    out[j] = '\0';

return n;
}

3. 块链串 
#include
#include
#define BLOCK_SIZE 4    // 可由用户定义的块大小
#define BLS_BLANK '#'   // 用于空白处的补齐字符

typedef struct _block {
    char ch[BLOCK_SIZE];    //块的数据域
    struct _block *next;    //块的指针域
} Block;

typedef struct {
    Block *head;        // 串的头指针
    Block *tail;        // 串的尾指针
    int len;            // 串的当前长度
} BLString;

//字符串初始化函数:
void blstr_init(BLString *T) {
    T->len = 0;
    T->head = NULL;
    T->tail = NULL;
}
这些定义已包含在头文件 dsstring.h 中,请实现块链串的子串查找操作:

bool blstr_substr(BLString src, int pos, int len, BLString *sub);
src为要查找的字符串
pos为子串开始的下标
len为子串的长度
sub在函数调用运行前指向一个已经初始化好的空串,在函数返回时,sub指向串src从第pos个字符起长度为len的子串
函数查找成功返回true,参数不正确返回 false
#include
#include
#include "dsstring.h" // 请不要删除,否则检查不通过

//满分代码

!!!!操作字符串长度 
int len(const char* s)
{
    int q = 0;
    while (*s != '\0') {//可以简化while(*s++) q++;
        q++;
        s++;//!!!
    }
    return q;
}
//易错点分析:
//1. 对于字符指针的操作, len函数书写
//2. 对于块的个数边界条件判定
//3. 对于块指针操作,分类讨论
//4. 尾指针置空,空白地方用'#'填充

int StrAssign(BLString* S, const char* cstr)
{//将cstr复制到块链串S中
 
    int i, j, k, len;
    Block *p, *q;
    len = strlen(cstr); //len为链串的长度
    if (len == 0)
        return 0;
    S->len = len;
    
    j = len / BLOCK_SIZE; //j为链串的结点数 ,也就是块个数` 
    if (len % BLOCK_SIZE)
        j++;
        
    for (i = 0; i < j; i++) {
        p = (Block*)malloc(sizeof(Block));
        if (!p)
            return 0;//可以简化
            
        //k指示每一个块内部字符ch[]下标
        for (k = 0; k < BLOCK_SIZE && *cstr; k++) //将字符串ctrs中的字符赋值给链串的数据域
            *(p->ch + k) = *cstr++;//!!!!!!
            
        if (i == 0) //如果是第一个结点
            S->head = q = p; //头指针指向第一个结点
        else {//q为跟踪指针 
            q->next = p;
            q = p;
        }
        if (!*cstr) //如果是最后一个链结点
        {
            S->tail = q; //将尾指针指向最后一个结点
            q->next = NULL; //将尾指针的指针域置为空
            for (; k < BLOCK_SIZE; k++) //最后一个结点用'#'填充
                *(q->ch + k) = BLS_BLANK;
        }
    }
    return 1;
}
bool blstr_substr(BLString src, int pos, int len, BLString* sub)
{
    char* t;
    if (pos < 0 || pos >= src.len || len < 1)
        return false;
    int n = pos / BLOCK_SIZE, h = pos % BLOCK_SIZE;
    Block* temp = src.head;
    
    for (int i = 0; i < n; i++)
        temp = temp->next;//temp直接指向pos位置那个块 
    
    char str[100];
    int i = 0;
    while (i < len) {
        if (h >= BLOCK_SIZE) {
            temp = temp->next;
            h = 0;
        } 
        else {
            if (!temp || temp->ch[h] == BLS_BLANK)
                break;
            str[i++] = temp->ch[h++];
        }
    }
    str[i] = '\0';
    StrAssign(sub, str);
    return true;
}

icoding复习3相关推荐

  1. icoding复习7, 8

    icoding复习7 哈希,AVL 查找 必考点!!!  1. 哈希表创建 typedef enum{     HASH_OK,     HASH_ERROR,     HASH_ADDED,     ...

  2. icoding复习5 树 感觉难度巨大....

    icoding 复习5 1. 先序遍历  已知二叉树按照二叉链表方式存储,利用栈的基本操作写出先序遍历非递归形式的算法: void pre_order(BiTree root); 二叉树的相关定义如下 ...

  3. icoding复习4 数组 十字链表

    icoding 复习4 1. 矩阵加法 实现三元组表示的两个稀疏矩阵的加法. #define MAXSIZE 100          //假设非零元个数的最大值为100 typedef struct ...

  4. icoding复习6 图

    icoding复习6 1. 邻接表1 试在邻接表存储结构上实现图的基本操作 insert_vertex 和 insert_arc,相关定义如下: typedef int VertexType; typ ...

  5. icoding复习1,2

    icoding复习 1 链表 倒数查找 1. 已知一个带有表头结点的单链表, 假设链表只给出了头指针L.在不改变链表的前提下,请设计一个尽可能高效的算法, 查找链表中倒数第k个位置上的结点(k为正整数 ...

  6. 已知小红今年12岁c语言编程,C语言程序设计第轮复习习题.doc

    C语言程序设计第轮复习习题 第1章 C语言概述.以下叙述正确的是 . A.在C程序中,main函数必须位于子程序的最前面 B.C程序的每一行中只能写一条语句 C.在对一个C程序进行编译的过程中,可发现 ...

  7. 可微偏导数一定存在_数学分析复习——偏导数(1)

    前言:微积分开始就是死刷题,背定义.然后我发现自己遗忘的速度简直怀疑人生.特别是在学物理以后,发现微积分根本就没有理解.一上来基础就没打好.所以希望能够慢慢地把数学分析,线性代数,偏微分,实变补起来 ...

  8. 考研计算机专业课怎么复习,2016考研计算机专业课如何复习?

    2016考研计算机专业课如何复习? ?基础复习阶段 以指定参考书为主,兼顾笔记,进行专业课的第一轮复习.看书要以理解为主,不必纠缠于细节,并在不懂的知识点处做上标记. 第一步,选择一本难度适宜.内容全 ...

  9. 计算机应用 含升学方向,对口升学《计算机应用基础》复习资料总汇(含答案))讲述.doc...

    对口升学<计算机应用基础>复习资料总汇 第一部分?? 一.单项选择题 1.世界上第一台电子数字计算机取名为(????). A.UNIVAC????B.EDSAC????C.ENIAC??? ...

最新文章

  1. 重磅!腾讯发布AI生态计划将赋能1000合作伙伴!
  2. 云企业网的应用于功能详解
  3. 为什么redis是单线程的以及为什么这么快?
  4. NFS mount.nfs: access denied by server while mounting 一个解决办法
  5. VTK:图表之RandomGraphSource
  6. sql重新注册服务器,sql server无法新建注册服务器怎么办?
  7. 【Python】绘制二维散点图
  8. 流氓百度乱发短信的经历
  9. mysql 5.7.11 x64_mysql 5.7.11 winx64安装配置教程
  10. 2021-06-28获取,更新,删除DOM节点
  11. 【UVA1592】Database (字符串读入技巧和map的使用)
  12. 计算机win7卡顿如何解决方法,win7系统运行卡顿的解决方法
  13. Zabbix 安装教程
  14. 创新创业孵化知识体系-创业实践导论之-技术专栏
  15. PHP中将Word文件转换为PDF
  16. Kaldi 使用,egs下通用样例及功能小结
  17. ShuZu数组中插入一个数
  18. 服务器站点地址是什么意思,云服务器地址是什么意思
  19. 重要:关于PPT转图片需要注意的问题
  20. mini,minis2,御2,御pro可以飞航线,飞跟随

热门文章

  1. TCP握手/挥手的过程分析
  2. Python turtle库实现基本剖析
  3. mysql数据库密码为空_注意MySQL数据库用户root密码为空
  4. 动态ram依靠什么存储信息_ROM、RAM、DRAM、SRAM和FLASH傻傻分不清
  5. 托马斯反驳牛顿被骂,普朗克颜值过高遭上帝捉弄,狄拉克却因爱情成话痨
  6. 道理我都懂,但你到底为什么偏偏喜欢咬我??
  7. 复工后,看看他们都是怎么上班的!
  8. 万万想不到:吸烟的辐射量比原子弹爆炸点还厉害!
  9. 如何把文件压缩变成一张图片?
  10. 用 Python 实现一个大数据搜索引擎