icoding复习3

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;
}
 
 

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

风起风里

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值