LeetCode //C - 1096. Brace Expansion II

1096. Brace Expansion II

Under the grammar given below, strings can represent a set of lowercase words. Let R(expr) denote the set of words the expression represents.

The grammar can best be understood through simple examples:

  • Single letters represent a singleton set containing that word.
    • R("a") = {"a"}
    • R("w") = {"w"}
  • When we take a comma-delimited list of two or more expressions, we take the union of possibilities.
    • R("{a,b,c}") = {"a","b","c"}
    • R("{{a,b},{b,c}}") = {"a","b","c"} (notice the final set only contains each word at most once)
  • When we concatenate two expressions, we take the set of possible concatenations between two words where the first word comes from the first expression and the second word comes from the second expression.
    • R("{a,b}{c,d}") = {"ac","ad","bc","bd"}
    • R("a{b,c}{d,e}f{g,h}") = {"abdfg", "abdfh", "abefg", "abefh", "acdfg", "acdfh", "acefg", "acefh"}

Formally, the three rules for our grammar:

  • For every lowercase letter x, we have R(x) = {x}.
  • For expressions e1, e2, ... , ek with k >= 2, we have R({e1, e2, ...}) = R(e1) ∪ R(e2) ∪ ...
  • For expressions e1 and e2, we have R(e1 + e2) = {a + b for (a, b) in R(e1) × R(e2)}, where + denotes concatenation, and × denotes the cartesian product.

Given an expression representing a set of words under the given grammar, return the sorted list of words that the expression represents.

Example 1:

Input: expression = "{a,b}{c,{d,e}}"

Output: "ac","ad","ae","bc","bd","be"

Explanation: arr becomes 15,15,15,9,10,10,10

Example 2:

Input: expression = "{{a,z},a{b,c},{ab,z}}"

Output: "a","ab","ac","z"

Explanation: Each distinct word is written only once in the final answer.

Constraints:
  • 1 <= expression.length <= 60
  • expressioni consists of '{', '}', ','or lowercase English letters.
  • The given expression represents a set of words based on the grammar given in the description.

From: LeetCode

Link: 1096. Brace Expansion II


Solution:

Ideas:

Recursively parse the expression, using union for commas and Cartesian product for concatenation, while accumulating results in sets to remove duplicates and sorting at the end.

Code:
c 复制代码
typedef struct {
    char** a;
    int size;
    int cap;
} Set;

static char* dupStr(const char* s) {
    char* p = (char*)malloc(strlen(s) + 1);
    strcpy(p, s);
    return p;
}

static void initSet(Set* s) {
    s->size = 0;
    s->cap = 16;
    s->a = (char**)malloc(sizeof(char*) * s->cap);
}

static void freeSet(Set* s) {
    for (int i = 0; i < s->size; i++) free(s->a[i]);
    free(s->a);
}

static void addStr(Set* s, const char* str) {
    for (int i = 0; i < s->size; i++) {
        if (strcmp(s->a[i], str) == 0) return;
    }

    if (s->size == s->cap) {
        s->cap *= 2;
        s->a = (char**)realloc(s->a, sizeof(char*) * s->cap);
    }

    s->a[s->size++] = dupStr(str);
}

static Set unionSet(Set x, Set y) {
    Set res;
    initSet(&res);

    for (int i = 0; i < x.size; i++) addStr(&res, x.a[i]);
    for (int i = 0; i < y.size; i++) addStr(&res, y.a[i]);

    freeSet(&x);
    freeSet(&y);
    return res;
}

static Set productSet(Set x, Set y) {
    Set res;
    initSet(&res);

    for (int i = 0; i < x.size; i++) {
        for (int j = 0; j < y.size; j++) {
            int len = strlen(x.a[i]) + strlen(y.a[j]);
            char* tmp = (char*)malloc(len + 1);

            strcpy(tmp, x.a[i]);
            strcat(tmp, y.a[j]);

            addStr(&res, tmp);
            free(tmp);
        }
    }

    freeSet(&x);
    freeSet(&y);
    return res;
}

static Set parse(char* expression, int* idx) {
    Set result;
    initSet(&result);   // empty union result

    Set current;
    initSet(&current);
    addStr(&current, "");   // identity for concatenation

    while (expression[*idx] && expression[*idx] != '}') {
        if (expression[*idx] == ',') {
            result = unionSet(result, current);

            initSet(&current);
            addStr(&current, "");

            (*idx)++;
        } else {
            Set next;
            initSet(&next);

            if (expression[*idx] == '{') {
                (*idx)++;
                next = parse(expression, idx);
                (*idx)++;   // skip '}'
            } else {
                char temp[2] = {expression[*idx], '\0'};
                addStr(&next, temp);
                (*idx)++;
            }

            current = productSet(current, next);
        }
    }

    result = unionSet(result, current);
    return result;
}

static int cmpStr(const void* a, const void* b) {
    return strcmp(*(char**)a, *(char**)b);
}

/**
 * Note: The returned array must be malloced, assume caller calls free().
 */
char** braceExpansionII(char* expression, int* returnSize) {
    int idx = 0;
    Set ans = parse(expression, &idx);

    qsort(ans.a, ans.size, sizeof(char*), cmpStr);

    *returnSize = ans.size;
    return ans.a;
}
相关推荐
地平线开发者5 小时前
【地平线 征程 6 工具链进阶教程】QAT 训练常见问题和排查
算法
地平线开发者5 小时前
征程 6 | 直方图量化配置与校准实例
算法
地平线开发者5 小时前
征程 6E/M Matrix 开发评板使用系列(一):开箱与点亮
算法·自动驾驶
Jerry5 小时前
LeetCode 59. 螺旋矩阵 II
算法
tachibana26 小时前
hot100 回文链表(234)
java·网络·数据结构·leetcode·链表
可编程芯片开发6 小时前
基于FOC控制器的BLDC无刷直流电机控制系统matlab编程与仿真
算法
aaaameliaaa6 小时前
进制练习题【找出只出现一次的数字、交换两个变量(不创建临时变量)、统计二进制中1的个数、打印整数二进制的奇数位和偶数位、求两个数二进制中不同位的个数】
c语言·数据结构·笔记·算法
灯厂码农6 小时前
C语言内存管理——内存对齐与共用体union
linux·服务器·c语言
QiLinkOS8 小时前
第三视觉理解徐玉生与他的商业活动(28)
大数据·c++·人工智能·算法·开源协议
wabs6668 小时前
关于动态规划【力扣1143.最长公共子序列的思考】
算法·leetcode·动态规划