看板 C_and_CPP 關於我們 聯絡資訊
※ 引述《liu2007 (薯)》之銘言: : 我打算用遞迴寫字元的排列 : 可是這個字元陣列裡有時候可能會有重複的情況 : 譬如說 : a b c d e f # # : 當這八個字元下去做排列的時候 : 理應只有 8!/2! 的答案 : 但是對於電腦來說還是有8!個答案 : 請問要怎麼樣避免這種狀況發生?? : 可否題點一下解決的方法@.@? : 使用遞迴實在是好難...... 遞迴概念超簡單,但用C寫實在是很難抓住細節,會寫到很想殺幾個人... 取字串排列會用到一個函數,是給一個字元a和一個字串str,求將a插入str任二個 字元之間,或是放在str頭或尾,的全部可能的結果. 在此稱這個函數是poss_insert. 一開始先寫個全部插入情況的基本型,這是當輸入字元和字串有重覆字母時, 會有重覆結果的情況. #include <stdio.h> #include <stdlib.h> #include <string.h> struct list { char* str; struct list* next; }; struct list* poss_insert(char a, char* str, struct list** ls); int main(int argc, char *argv[]) { struct list* ls; poss_insert('#', "a#", &ls); int i; struct list* p = ls; while (p != NULL) { printf("%s\n", p->str); p = p->next; } system("PAUSE"); return 0; } struct list* poss_insert(char a, char* str, struct list** ls) { int slen = strlen(str); if (slen == 0) { (*ls) = (struct list*)malloc(sizeof(struct list)*1); (*ls)->str = (char*)malloc(sizeof(char)*(slen+1)); (*ls)->str[0] = a; (*ls)->str[1] = '\0'; (*ls)->next = NULL; } else { /* Let a be after str */ (*ls) = (struct list*)malloc(sizeof(struct list)*1); (*ls)->str = (char*)malloc(sizeof(char)*(slen+1)); strcpy((*ls)->str, str); (*ls)->str[slen] = a; (*ls)->str[slen+1] = '\0'; /* Let a insert into str */ char* tstr = (char*)malloc(sizeof(char)*(slen-1)); strncpy(tstr, str, slen-1); tstr[slen-1] = '\0'; poss_insert(a, tstr, &(*ls)->next); free(tstr); struct list *p = (*ls)->next, *r; char* q; while(p != NULL) { q = &(str[slen-1]); p->str = strcat(p->str, q); p = p->next; } } return (*ls); } 以上程式中,在先直接把a放到str後面的程式段落,可以發現如果a與str的最後一字 相同,這個動作可以跳過. 因為此時做這個動作,是製造出重覆結果的關鍵步驟. 所以,無重覆版本改成以下情況,改法是在發現a是str最後一字時,把struct list 結構的當前節點當做是空節點. 是藉著將 (*ls)->str 指定為 NULL 方式. 而這樣修改,其他用以讀取串列並產生輸出的程式都要記得檢查 ->str == NULL 情況. int main(int argc, char *argv[]) { struct list* ls; poss_insert('+', "c&c+", &ls); int i; struct list* p = ls; while (p != NULL) { if (p->str != NULL) { printf("%s\n", p->str); } p = p->next; } system("PAUSE"); return 0; } struct list* poss_insert(char a, char* str, struct list** ls) { int slen = strlen(str); if (slen == 0) { (*ls) = (struct list*)malloc(sizeof(struct list)*1); (*ls)->str = (char*)malloc(sizeof(char)*(slen+1)); (*ls)->str[0] = a; (*ls)->str[1] = '\0'; (*ls)->next = NULL; } else { /* Let a be after str */ (*ls) = (struct list*)malloc(sizeof(struct list)*1); if (str[slen-1] == a) { (*ls)->str = NULL; } else { (*ls)->str = (char*)malloc(sizeof(char)*(slen+1)); strcpy((*ls)->str, str); (*ls)->str[slen] = a; (*ls)->str[slen+1] = '\0'; } /* Let a insert into str */ char* tstr = (char*)malloc(sizeof(char)*(slen-1)); strncpy(tstr, str, slen-1); tstr[slen-1] = '\0'; poss_insert(a, tstr, &(*ls)->next); free(tstr); struct list *p = (*ls)->next, *r; char* q; while(p != NULL) { q = &(str[slen-1]); if (p->str != NULL) { p->str = strcat(p->str, q); } p = p->next; } } return (*ls); } 以上主程式情況,輸出: c&c++ c&+c+ c+&c+ +c&c+ 請按任意鍵繼續 . . . -- 媽,我寫出來了! (揮手) -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 218.160.215.90 ※ 編輯: yauhh 來自: 218.160.215.90 (03/18 07:57)