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]]

分析

寻找一系列数字的全排列,以数组形式输出
1,是先找一个最小字典序的全排列,然后依次增大。

void sort(int *a, int left, int right)
{
    if(left >= right)/*如果左边索引大于或者等于右边的索引就代表已经整理完成一个组了*/
    {
        return ;
    }
    int i = left;
    int j = right;
    int key = a[left];
     
    while(i < j)                               /*控制在当组内寻找一遍*/
    {
        while(i < j && key <= a[j])
        /*而寻找结束的条件就是,1,找到一个小于或者大于key的数(大于或小于取决于你想升
        序还是降序)2,没有符合条件1的,并且i与j的大小没有反转*/ 
        {
            j--;/*向前寻找*/
        }
         
        a[i] = a[j];
        /*找到一个这样的数后就把它赋给前面的被拿走的i的值(如果第一次循环且key是
        a[left],那么就是给key)*/
         
        while(i < j && key >= a[i])
        /*这是i在当组内向前寻找,同上,不过注意与key的大小关系停止循环和上面相反,
        因为排序思想是把数往两边扔,所以左右两边的数大小与key的关系相反*/
        {
            i++;
        }
         
        a[j] = a[i];
    }
     
    a[i] = key;/*当在当组内找完一遍以后就把中间数key回归*/
    sort(a, left, i - 1);/*最后用同样的方式对分出来的左边的小组进行同上的做法*/
    sort(a, i + 1, right);/*用同样的方式对分出来的右边的小组进行同上的做法*/
                       /*当然最后可能会出现很多分左右,直到每一组的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;
    }
//先初始化所有的二维数组
    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);
//依次复制上个全排列,然后使用之前的算法,对其递增,找到下一个全排列
    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,当然也可以使用递归进行二维数组的赋值,就是对某一列依次赋值,个数为剩下的数字的全排列总数,之后对下一列进行剩下数字的全排列

void Permutation(int **ans,int k,int p,int *nums,int numsSize)//k第几列 p某个相同的数字第几行开始的
{
//剩下一个数字,就直接赋值返回
    if(numsSize==1)
    {
        ans[p][k]=nums[0];
        return;
    }
//找到一个数字应该重复多少次
    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];
        }
        //对剩下的数字再单独得到一个数组,以便递归调用
        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;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容