PTA 最短前缀(c语言版)

描述: 

一个字符串的前缀是从该字符串的第一个字符起始的一个子串。例如“carbon”的字串是: “c”,“ca”,“car”,“carb”,“carbo”和“carbon”。注意到这里我们不认为空串是字串,但是每个非空串是它自身的字串。我们现在希望能用前缀来缩略的表示单词。例如,"carbohydrate" 通常用“carb”来缩略表示。现在给你一组单词,要求你找到唯一标识每个单词的最短前缀。
在下面的例子中,"carbohydrate" 能被缩略成“carboh”,但是不能被缩略成“carbo”(或其余更短的前缀) ,因为已经有一个单词用“carbo”开始。
一个精确匹配会覆盖一个前缀匹配,例如,前缀“car”精确匹配单词“car”。因此,"car" 是“car”的缩略语是没有二义性的,“car”不会被当成“carriage”或者任何在列表中以“car”开始的单词。

输入格式:

输入包括至少2行,至多1000行。每行包括一个以小写字母组成的单词,单词长度至少是1,至多是20。

输出格式:

输出的行数与输入的行数相同。每行输出由相应行输入的单词开始,后面跟着一个空格接下来是相应单词的没有二义性的最短前缀标识符。

输入样例:

在这里给出一组输入。例如:

carbohydrate
cart
carburetor
caramel
caribou
carbonic
cartilage
carbon
carriage
carton
car
carbonate
 

输出样例:

在这里给出相应的输出。例如:

carbohydrate carboh
cart cart
carburetor carbu
caramel cara
caribou cari
carbonic carboni
cartilage carti
carbon carbon
carriage carr
carton carto
car car
carbonate carbona
 

代码如下:

#include <stdio.h>
#include <string.h>
#include <stdbool.h>

#define MAX_WORDS 1000
#define MAX_LENGTH 20

int main() {
    char words[MAX_WORDS][MAX_LENGTH + 1]; // 存储单词
    char prefixes[MAX_WORDS][MAX_LENGTH + 1]; // 存储前缀
    char temp[MAX_LENGTH + 1];
    int count = 0; // 单词计数

    // 读取单词直到EOF
    while (scanf("%s", words[count]) != EOF) {
        strcpy(prefixes[count], words[count]); // 默认前缀为整个单词
        count++;
    }

    // 检查每个单词的前缀
    for (int i = 0; i < count; i++) {
        int len = strlen(words[i]);
        for (int j = 1; j < len; j++) { // 从1开始,因为空串不是前缀
            bool unique = true;
            strncpy(temp, words[i], j);
            temp[j] = '\0'; // 确保字符串结束

            for (int k = 0; k < count; k++) {
                if (i != k && strncmp(words[k], temp, j) == 0) {
                    unique = false;
                    break;
                }
            }

            if (unique) {
                strcpy(prefixes[i], temp);
                break;
            }
        }
    }

    // 输出结果
    for (int i = 0; i < count; i++) {
        printf("%s %s\n", words[i], prefixes[i]);
    }

    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值