庞果网 合法字符串

题目详情:
用n个不同的字符(编号1 - n),组成一个字符串,有如下2点要求:
    1、对于编号为i 的字符,如果2 * i > n,则该字符可以作为最后一个字符,但如果该字符不是作为最后一个字符的话,则该字符后面可以接任意字符;
    2、对于编号为i的字符,如果2 * i <= n,则该字符不可以作为最后一个字符,且该字符后面所紧接着的下一个字符的编号一定要 >= 2 * i。
问有多少长度为M且符合条件的字符串。
例如:N = 2,M = 3。则abb, bab, bbb是符合条件的字符串,剩下的均为不符合条件的字符串。


输入:n,m  (2<=n,m<=1000000000);
输出:满足条件的字符串的个数,由于数据很大,输出该数Mod 10^9 + 7的结果。


函数头部
int validstring(int n,int m) {

}


代码如下:

 

import java.math.BigDecimal;
import java.util.LinkedList;
import java.util.Queue;

public class Main {
    public static int validstring(int n,int m) {
        if (n > 2 || m > 1000000000){
            System.err.println("‘n’或者‘m’超出范围了!");
            return 0;
        }
        long count = countValid(n, m);
        return BigDecimal.valueOf(count % (Math.pow(10, 9) + 7)).intValue();
    }
    
    //n个字符m位的全排列组合
    private static long countValid(int n, int m) {
        String v = "";
        long s = 0;
        char[] charArray = buildCharArray(n);
        // 队列用来保存被访问节点的分支节点(邻接点)
        Queue<String> que = new LinkedList<String>();
        que.offer(v);// 起点入队列
        while (!que.isEmpty()) {
            v = que.poll();// 弹出当前顶点
            // 将当前节点的分支节(邻接)点加入到队列中
            for (int i = 0; i < n; i++) {
                String u = v + charArray[i];
                if (u.length() == m) {//m位,则校验是不是符合题目的要求
                    if (checkStr(u, charArray, 0)){
                        System.out.println(u);
                        ++s;
                    }
                } else{
                    que.add(u); // 邻接点入队
                }
            }
        }
        return s;
    }
    
    //校验字符串是否符合规则
    private static boolean checkStr(String str, char[] charArray, int j){
        boolean returnFlag = false;
        int n = charArray.length;
        int m = str.length();
        char c = str.charAt(j);
        Integer i = getCharArrayIndex(charArray, c);
        //如果2 * i > n,则该字符可以作为最后一个字符,但如果该字符不是作为最后一个字符的话,则该字符后面可以接任意字符;
        if (2 * (i + 1) > n){
            if (j == m - 1){
                returnFlag = true;
            }else{
                returnFlag = checkStr(str, charArray, j + 1);
            }
        }else{//如果2 * i <= n,则该字符不可以作为最后一个字符,且该字符后面所紧接着的下一个字符的编号一定要 >= 2 * i。
            if (j == m - 1){
                returnFlag = false;
            }else{
                char d = str.charAt(j + 1);
                int ii = getCharArrayIndex(charArray, d);
                if (ii + 1 < 2 * (i + 1)){
                    returnFlag = false;
                }else{
                    returnFlag = checkStr(str, charArray, j + 1);
                }
            }
        }
        return returnFlag;
    }
    
    private static Integer getCharArrayIndex(char[] charArray, char c){
        Integer returnIndex = null;
        for (int i = 0; i < charArray.length; i++){
            char d = charArray[i];
            if (d == c){
                returnIndex = i;
            }
        }
        return returnIndex;
    }
    
    //产生1-n的随机字符
    private static char[] buildCharArray(int n){
        StringBuffer returnValue = new StringBuffer();
        for (int i = 0; i < n; i++){
            char c = 'a';
            c = (char)(c + i);
            returnValue.append(c);
        }
        return returnValue.toString().toCharArray();
    }

    //start 提示:自动阅卷起始唯一标识,请勿删除或增加。 
    public static void main(String args[]) 
    { 
           System.out.println(validstring(2, 10) + " 次");
    } 
    //end //提示:自动阅卷结束唯一标识,请勿删除或增加。
}


 

 

posted @ 2013-08-01 18:49  jlins  阅读(428)  评论(0编辑  收藏  举报