Leetcode 46. Permutations

題目

Given a collection of distinct numbers, return all possible permutations.
For example,[1,2,3]
have the following permutations:
[ [1,2,3], [1,3,2], [2,1,3], [2,3,1], [3,1,2], [3,2,1]]

分析

尋找一系列數(shù)字的全排列蒸其,以數(shù)組形式輸出
1取胎,是先找一個(gè)最小字典序的全排列哥纫,然后依次增大颖对。

void sort(int *a, int left, int right)
{
    if(left >= right)/*如果左邊索引大于或者等于右邊的索引就代表已經(jīng)整理完成一個(gè)組了*/
    {
        return ;
    }
    int i = left;
    int j = right;
    int key = a[left];
     
    while(i < j)                               /*控制在當(dāng)組內(nèi)尋找一遍*/
    {
        while(i < j && key <= a[j])
        /*而尋找結(jié)束的條件就是百揭,1,找到一個(gè)小于或者大于key的數(shù)(大于或小于取決于你想升
        序還是降序)2抡笼,沒(méi)有符合條件1的,并且i與j的大小沒(méi)有反轉(zhuǎn)*/ 
        {
            j--;/*向前尋找*/
        }
         
        a[i] = a[j];
        /*找到一個(gè)這樣的數(shù)后就把它賦給前面的被拿走的i的值(如果第一次循環(huán)且key是
        a[left]仔戈,那么就是給key)*/
         
        while(i < j && key >= a[i])
        /*這是i在當(dāng)組內(nèi)向前尋找,同上拧廊,不過(guò)注意與key的大小關(guān)系停止循環(huán)和上面相反杂穷,
        因?yàn)榕判蛩枷胧前褦?shù)往兩邊扔,所以左右兩邊的數(shù)大小與key的關(guān)系相反*/
        {
            i++;
        }
         
        a[j] = a[i];
    }
     
    a[i] = key;/*當(dāng)在當(dāng)組內(nèi)找完一遍以后就把中間數(shù)key回歸*/
    sort(a, left, i - 1);/*最后用同樣的方式對(duì)分出來(lái)的左邊的小組進(jìn)行同上的做法*/
    sort(a, i + 1, right);/*用同樣的方式對(duì)分出來(lái)的右邊的小組進(jìn)行同上的做法*/
                       /*當(dāng)然最后可能會(huì)出現(xiàn)很多分左右卦绣,直到每一組的i = j 為止*/
}


/**
 * Return an array of arrays of size *returnSize.
 * Note: The returned array must be malloced, assume caller calls free().
 */
int** permute(int* nums, int numsSize, int* returnSize) {
    *returnSize=1;
    for(int i=1;i<=numsSize;i++)
    {
        *returnSize=(*returnSize)*i;
    }
//先初始化所有的二維數(shù)組
    int **ans=(int **)malloc(sizeof(int *)*(*returnSize));
    ans[0]=(int *)malloc(sizeof(int)*numsSize);
    for(int j=0;j<numsSize;j++)
    {
        ans[0][j]=nums[j];
    }
//排序耐量,得到最小字典序的全排列
    sort(ans[0],0,numsSize-1);
//依次復(fù)制上個(gè)全排列,然后使用之前的算法滤港,對(duì)其遞增廊蜒,找到下一個(gè)全排列
    for(int i=1;i<(*returnSize);i++)
    {
        ans[i]=(int *)malloc(sizeof(int)*numsSize);
        for(int j=0;j<numsSize;j++)
        {
            ans[i][j]=ans[i-1][j];
        }
        
        
        int i1=numsSize-2,j1=numsSize-1,temp=0,p;
        while(i1>=0)
        {
            if(ans[i][i1]>=ans[i][j1])
            {
                i1--;
                j1--;
            }
            else
                break;
        }
        p=j1;
        j1++;
        while(j1<numsSize)
        {
            if(ans[i][j1]>ans[i][i1]&&ans[i][j1]<ans[i][p])
                p=j1;
            j1++;
        }
        if(i1>=0)
        {
            temp=ans[i][i1];
            ans[i][i1]=ans[i][p];
            ans[i][p]=temp;
            sort(ans[i],i1+1,numsSize-1);
        }
        else
            sort(ans[i],0,numsSize-1);
    }
    return ans;
}

2趴拧,當(dāng)然也可以使用遞歸進(jìn)行二維數(shù)組的賦值,就是對(duì)某一列依次賦值,個(gè)數(shù)為剩下的數(shù)字的全排列總數(shù)山叮,之后對(duì)下一列進(jìn)行剩下數(shù)字的全排列

void Permutation(int **ans,int k,int p,int *nums,int numsSize)//k第幾列 p某個(gè)相同的數(shù)字第幾行開(kāi)始的
{
//剩下一個(gè)數(shù)字著榴,就直接賦值返回
    if(numsSize==1)
    {
        ans[p][k]=nums[0];
        return;
    }
//找到一個(gè)數(shù)字應(yīng)該重復(fù)多少次
    int temp=1;
    for(int i=1;i<numsSize;i++)
    {
        temp=temp*i;
    }
//
    for(int i=0;i<numsSize;i++)
    {
        for(int j=p;j<p+temp;j++)
        {
            ans[j][k]=nums[i];
        }
        //對(duì)剩下的數(shù)字再單獨(dú)得到一個(gè)數(shù)組,以便遞歸調(diào)用
        int *numsTemp=(int *)malloc(sizeof(int)*(numsSize));
        for(int j=0;j<i;j++)
        {
            numsTemp[j]=nums[j];
        }
        for(int j=i;j<numsSize-1;j++)
        {
            numsTemp[j]=nums[j+1];
        }
        
        //for(int j=0;j<numsSize-1;j++)
        //{
        //    printf("%d ",numsTemp[j]);
        //}
        //printf("\n");
        
        Permutation(ans,k+1,p,numsTemp,numsSize-1);
        free(numsTemp);
        p=p+temp;
    }
}

/**
 * Return an array of arrays of size *returnSize.
 * Note: The returned array must be malloced, assume caller calls free().
 */
int** permute(int* nums, int numsSize, int* returnSize) {
    *returnSize=1;
    for(int i=1;i<=numsSize;i++)
    {
        *returnSize=(*returnSize)*i;
    }
    int **ans=(int **)malloc(sizeof(int *)*(*returnSize));
    for(int i=0;i<(*returnSize);i++)
    {
        ans[i]=(int *)malloc(sizeof(int)*numsSize);
    }
    Permutation(ans,0,0,nums,numsSize);
    
    
    return ans;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末屁倔,一起剝皮案震驚了整個(gè)濱河市脑又,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌锐借,老刑警劉巖问麸,帶你破解...
    沈念sama閱讀 206,482評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異钞翔,居然都是意外死亡严卖,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,377評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門布轿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)哮笆,“玉大人,你說(shuō)我怎么就攤上這事汰扭〕碇猓” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 152,762評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵萝毛,是天一觀的道長(zhǎng)项阴。 經(jīng)常有香客問(wèn)我,道長(zhǎng)珊泳,這世上最難降的妖魔是什么鲁冯? 我笑而不...
    開(kāi)封第一講書人閱讀 55,273評(píng)論 1 279
  • 正文 為了忘掉前任拷沸,我火速辦了婚禮色查,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘撞芍。我一直安慰自己秧了,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,289評(píng)論 5 373
  • 文/花漫 我一把揭開(kāi)白布序无。 她就那樣靜靜地躺著验毡,像睡著了一般。 火紅的嫁衣襯著肌膚如雪帝嗡。 梳的紋絲不亂的頭發(fā)上晶通,一...
    開(kāi)封第一講書人閱讀 49,046評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音哟玷,去河邊找鬼狮辽。 笑死一也,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的喉脖。 我是一名探鬼主播椰苟,決...
    沈念sama閱讀 38,351評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼树叽!你這毒婦竟也來(lái)了舆蝴?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 36,988評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤题诵,失蹤者是張志新(化名)和其女友劉穎洁仗,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體仇轻,經(jīng)...
    沈念sama閱讀 43,476評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡京痢,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,948評(píng)論 2 324
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了篷店。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片祭椰。...
    茶點(diǎn)故事閱讀 38,064評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖疲陕,靈堂內(nèi)的尸體忽然破棺而出方淤,到底是詐尸還是另有隱情,我是刑警寧澤蹄殃,帶...
    沈念sama閱讀 33,712評(píng)論 4 323
  • 正文 年R本政府宣布携茂,位于F島的核電站,受9級(jí)特大地震影響诅岩,放射性物質(zhì)發(fā)生泄漏讳苦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,261評(píng)論 3 307
  • 文/蒙蒙 一吩谦、第九天 我趴在偏房一處隱蔽的房頂上張望鸳谜。 院中可真熱鬧,春花似錦式廷、人聲如沸咐扭。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,264評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)蝗肪。三九已至,卻和暖如春蠕趁,著一層夾襖步出監(jiān)牢的瞬間薛闪,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,486評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工俺陋, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留豁延,地道東北人怀各。 一個(gè)月前我還...
    沈念sama閱讀 45,511評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像术浪,于是被迫代替她去往敵國(guó)和親瓢对。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,802評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容