组合(从长度为n的字符串中取m个字符)---java两种实现方法

对于这类组合问题,虽然感觉很简单,但是用java代码实现起来却不是那么容易的。
这其中最容易用到的应该是递归的思想了,这种方法也比较容易理解:
方法一:
递归实现代码:

/**
     * 可能种类在4000万的时候时间效率为7.6s左右
     * @param headIndex:当前所要添加字符在原始字符串中的index
     * @param length:当前字符串长度
     * @param inStr:原始字符串
     * @param res:当前字符串
     * @param n:所要组合的字符串长度
     */
    public static void DiguiZuhe(int headIndex,int length,String inStr,String res,int n){
        String s = res;
        for (int i = headIndex; i < inStr.length() + length - n; i++) {
            if (length <= n) {
                res = res + inStr.substring(i, i + 1); 
                DiguiZuhe(i + 1, length + 1, inStr, res, n);
                if (length == n) {
                    System.out.println(res);
                }
            }else {
                return;
            }
            res = s;
        }
    }
    //-------------------------------分割线-----------------------------
    public static void main(String[] args) {
        String inStr = "abcdefg";
        int n = 3;
        String res = "";
        DiguiZuhe(0, 1, inStr, res, n);
    }

方法二:

/**
     * * 具体思路:将我们的原始的字符串用01数组来代替,0表示没有,1表示有
     * 例如:字符串:abcdefg;01数组:[1110000]->abc;[1010100]->ace
     * 长度为n的字符串,取m个字符
     * 首先会初始化一个01数组,长度为n,前m数字为1,后面n-m个数字为0
     * 然后扫描01数组,每一次发现10的时候,都给它变为01,同时将左边的1都移动到最左边,已经是最左边了就不移动,
     * 继续扫描直到所有的1都移动到了最右边
     * 例如:
     * 1 1 1 0 0
     * 1 1 0 1 0
     * 1 0 1 1 0
     * 0 1 1 1 0
     * 1 1 0 0 1
     * 1 0 1 0 1
     * 0 1 1 0 1
     * 1 0 0 1 1
     * 0 1 0 1 1
     * 0 0 1 1 1
     * 纯算法时间效率:可能种类在4千万左右时间为1.2s左右
     * @param input
     * @param strNum
     * @return
     */
    public static void Combination(String input,int strNum){

        int length = input.length();
        int[] array = new int[length];
        //如果长度相等,就只有一种可能
        if (length == strNum) {
            System.out.println(input);
            return;
        }else {
            //初始化数组
            for (int i = 0; i < length; i++) {
                if (i <= strNum -1) {
                    array[i] = 1;
                }else {
                    array[i] = 0;
                }
            }
        }
        while(isContainOneZero(array)){
            transform(array);
            //将01组成的数组转化为字符串,本来这种算法比较高效的,但是在这里转为String字符串的时候(当数据量特别大的时候)
            //主要时间开销就在这里了
            StringBuffer res = new StringBuffer();
            for (int i = 0; i < array.length; i++) {
                if (array[i] == 1) {
                    //拼接起来
                    res.append(input.charAt(i));
                }
            }
            System.out.println(res.toString());
        }
    }
    /**
     * 验证是否包含10
     * @param array
     * @return
     */
    public static boolean isContainOneZero(int[] array){

        for (int i = 0; i < array.length - 1; i++) {
            if (array[i] == 1 && array[i + 1] == 0) {
                return true;
            }
        }
        return false;
    }
    /**
     * 验证是否所有的1都移到了最左边
     * @param array
     * @param i
     * @return
     */
    public static boolean validate(int[] array,int i){

        for (int j = 0; j < i; j++) {
            if (array[j] == 0 && array[j + 1] == 1) {
                return true;
            }
        }
        return false;
    }
    /**
     * 转换数组
     * @param array
     */
    public static void transform(int[] array){

        for (int i = 0; i < array.length - 1; i++) {
            if (array[i] == 1 && array[i + 1] == 0) {
                swap(array, i, i + 1);
                int n = getOneNum(array, i);
                for (int j = 0; j < i; j++) {
                    if (j <= n - 1) {
                        array[j] = 1;
                    }else {
                        array[j] = 0;
                    }
                }
                return;
            }
        }
    }
    /**
     * 获取1的数量
     * @param array
     * @param i
     * @return
     */
    public static int getOneNum(int[] array,int i){
        int num = 0;
        for (int j = 0; j < i; j++) {
            if (array[j] == 1) {
                num++;
            }
        }
        return num;
    }
    /**
     * 交换方法
     * @param array
     * @param a
     * @param b
     */
    public static void swap(int[] array,int a,int b){

        array[a] = array[a] + array[b];
        array[b] = array[a] - array[b];
        array[a] = array[a] - array[b];     
    }
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 字符串拼接是指将两个或多个字符串连接起来形成一个新的字符串。在Java,可以使用"+"运算符来进行字符串拼接,也可以使用String类的concat()方法实现。 例如: String str1 = "Hello"; String str2 = "World"; String str3 = str1 + " " + str2; // 使用"+"运算符进行字符串拼接 String str4 = str1.concat(" ").concat(str2); // 使用concat()方法进行字符串拼接 在以上代码,str3和str4的值都是"Hello World",它们是由str1和str2拼接而成的。需要注意的是,使用"+"运算符进行字符串拼接时,如果其一个操作数是字符串,另一个操作数会自动转换为字符串类型。如果要将其他类型的数据转换为字符串类型,可以使用String类的valueOf()方法。 例如: int num = 123; String str5 = "The number is " + String.valueOf(num); // 将int类型的数据转换为字符串类型 在以上代码,str5的值是"The number is 123",它是由字符串"The number is "和整数123拼接而成的。 ### 回答2: Java字符串是指由零个或多个Unicode字符组成的有限序列。在Java编程字符串是非常常见的一种数据类型。Java有许多种创建字符串对象的方法,其最常见的一种是使用String类。 在Java,String类是一个包含各种有用方法的类,它允许我们在程序创建并操作字符串。其一个可以使用的方法字符串拼接或字符串拼接运算。字符串拼接运算是将两个或多个字符串连接成一个新的字符串。在Java,它通过"+"号来执行。 例如,在Java,我们可以这样使用字符串拼接运算: String str1 = "hello "; String str2 = "world"; String str3 = str1 + str2; System.out.println(str3); 这将输出“hello world”。 需要注意的是,在执行字符串拼接运算时,Java会自动调用toString()方法将数据类型转换为字符串类型。因此,例如,如果我们想用一个int类型的变量来连接字符串,我们需要将其转换为字符串类型: int num = 10; String str = " the result is: " + Integer.toString(num); System.out.println(str); 这将输出“the result is: 10”。 在实际的编码字符串拼接操作经常用于构建消息字符串,例如在向用户显示错误消息或日志消息时,我们通常会将不同的字符串拼接在一起以形成完整的消息。另外,在实际的编码,我们应该尽可能的使用StringBuilder或StringBuffer来执行字符串拼接操作,以提高字符串拼接操作的性能。 ### 回答3: Java的String是一种不可变的对象,意味着一旦字符串对象创建后,其的内容就不可更改。而字符串拼接的过程涉及操作创建新的字符串对象,因此需要特别留意与优化。 Java两种字符串拼接的方法: 1.使用‘+’运算符连接字符串 这是一种最常见的字符串拼接方法。例如:String str = "Hello" + " World";拼接出的结果为 Hello World。也可以将字符串变量与常量拼接:String name = "Tom"; String greeting = "Hello " + name; 使用‘+’运算符拼接字符串,底层代码实现是先创建一个新的StringBuilder对象,然后将操作数添加到这个StringBuilder对象里,最后拿到StringBuilder对象里的字符串。 需要注意,大量的字符串连接所产生的新的临时对象也会占用内存,导致较慢的时间和空间性能。如下代码所演示: String message = ""; for (int i = 1; i < 1000; i++) { message += "" + i; } 在上面的例子,message每次操作都会新建一个StringBuilder对象,每次字符串长度增长都必须创建一个新的String对象,最后的字符串还需要拷贝到一个新的String对象返回。这样的连续字符串拼接最终导致频繁创建和销毁大量的字符串对象,降低了系统的性能。 2.使用StringBuilder拼接字符串 StringBuilder类可以用来创建可改变的字符串,这就意味着它可以构建一个可变的字符串,并可以动态的增加或修改其内容。StringBuilder类提供了多种方法,可以将不同类型的数据拼接到一个字符。 对于大量的字符串拼接,使用StringBuilder所达到的效果比使用‘+’运算符更优。如下所演示: StringBuilder message = new StringBuilder(); for (int i = 1; i < 1000; i++) { message.append(i); } String finalMessage = message.toString(); 这里使用了append()方法向StringBuilder添加了每一个数字。最后通过调用toString()方法来获取拼接后的字符串。 总结:在实际开发应该尽量避免直接使用‘+’运算符来进行字符串拼接。如果只是简单的拼接几个字符串或者单个变量,可以使用‘+’运算符。但是对于大量的拼接,我们应该使用StringBuilder,这样能够显著的提高系统的效率和性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值