首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >所有排列的有效计算

所有排列的有效计算
EN

Stack Overflow用户
提问于 2014-04-03 14:04:54
回答 2查看 228关注 0票数 1

受到一个含糊不清的问题的启发,我感到很难解决对它的更严格的解释,具体如下:

如何最有效地计算所有可能包含n个零(0位数)的整数值(32位值)?例如,给定n= 7,包含确切7个零的不同整数值的数目是:

代码语言:javascript
运行
复制
32*31*30*29*28*27*26 / (1*2*3*4*5*6*7) = 3.365.856

具有确切7个零的整数值示例如下:

代码语言:javascript
运行
复制
11111111000000011111111111111111

如果你想自己解决这个问题,不要读我的答案。否则,评估我的答案,改进它或张贴一个更好,更有效的答案。

EN

回答 2

Stack Overflow用户

发布于 2014-04-03 14:04:54

我的算法的思想如下:

  1. 创建一个整数值,其右侧(最小有效位)的零数与函数permut()的值相同。
  2. 开始将最左边的零点一点一点地移到左边。
  3. 将移动零点右侧的所有位作为目标值,并将相同的算法应用于缩短值。

该算法有两个重要的特点:

  • 它是递归的。
  • 它需要与值一样多的计算量来计算。

在这里,算法作为Java代码:

代码语言:javascript
运行
复制
public static void main(String[] args) {
    List<Integer> permutations = permut(7);
}

private static List<Integer> permut(int zeros) {
    List<Integer> permutations = new ArrayList<>();
    permut(zeros == 32 ? 0 : 0xFFFFFFFF << zeros, zeros, 31, permutations);
    return permutations;
}

/*
 * @param value
 *     for which to move the zero digit at bit position (zeros - 1)
 *     to the stopBit position
 * @param zeros
 *     number of 0 digits available at the right most bit positions
 *     of value
 * @param stopBit
 *     the left most bit position to move the zero digit at bit position
 *     (zeros - 1) to
 * @param values
 *     to add the newly calculated integer values to
 */
private static void power(int value, int zeros, int stopBit, List<Integer> values) {
    values.add(value);
    if (zeros == 0) return;
    int cleared = value | (1 << (zeros - 1));
    for (int bit = zeros - 1; bit < stopBit;) {
        power(cleared ^ (1 << ++bit), zeros - 1, bit - 1, values);
    }
}

如果您对算法的行为是否正确感到好奇,请使用修改后的main()方法尝试以下检查方法:

代码语言:javascript
运行
复制
public static void main(String[] args) {
    int zeros = 7;
    List<Integer> permutations = permut(zeros);
    System.out.println("Expected number of values:  " + combinations(zeros));
    System.out.println("Returned number of values:  " + permutations.size());
    System.out.println("Returned values are unique: " + (new HashSet<>(permutations).size() == permutations.size()));
    System.out.printf("All values contain %d zeros: %s\n", zeros, haveZeros(zeros, permutations));
}

private static long combinations(int zeros) {
    long p = 1;
    long q = 1;
    for (int count = 0; count < zeros;) {
        p *= (32 - count);
        q *= (++count);
    }
    return p / q;
}

private static boolean haveZeros(int zeros, List<Integer> values) {
    for (Integer value : values) {
        int count = 0;
        for (int bit = 1; bit != 0; bit = bit << 1) {
            if ((value & bit) == 0) count++;
        }
        if (count != zeros) {
            System.out.println(Integer.toBinaryString(value));
            return false;
        }
    }
    return true;
}
票数 1
EN

Stack Overflow用户

发布于 2014-04-03 14:39:35

假设您想要枚举实际数字,您可以简单地创建一个大小为32的字符数组,其大小为n0,然后对该数组进行置换。

代码语言:javascript
运行
复制
import java.io.*;
import java.util.*;

public class Solution
{

    static char [] arr;
    public static void main(String[]args)
    {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        arr = new char[5];
        for(int i = 0; i < arr.length; i++)
        {
            if(n > 0)
            {
                arr[i] = '0';
                n--;
            }
            else
                arr[i] = '1';
        }
        permute(0);
    }

    public static void permute(int i)
    {
        if(i >= arr.length)
        {
            System.out.println(arr);
            return;
        }

        int [] dups = new int[2];

        for(int j = i; j < arr.length; j++)
        {
            if(dups[arr[j]-'0'] == 1)
                continue;
            dups[arr[j]-'0'] = 1;
            char temp = arr[i];
            arr[i] = arr[j];
            arr[j] = temp;
            permute(i+1);
            temp = arr[i];
            arr[i] = arr[j];
            arr[j] = temp;
        }
    }
}

太慢了,因为有m!排列,其中m是数组的大小。为了加快速度,我把尺寸设为5。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/22840189

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档