这是一道在线编程的题目,题目详情:java
回文字符串是指从左到右和从右到左相同的字符串,现给定一个仅由小写字母组成的字符串,你能够把它的字母从新排列,以造成不一样的回文字符串。c++
输入:非空仅由小写字母组成的字符串,长度不超过100;算法
输出:能组成的全部回文串的个数(由于结果可能很是大,输出对1000000007取余数的结果)。编程
例如:输入"aabb" 输出为2(由于“aabb”对应的全部回文字符串有2个:abba和baab)数组
函数头部 c: int palindrome(const char *s); c++ int palindrome(const string &s); java public static int palindrome(String s) ; xss
个人解题思路函数
根据回文字符串的特色,能够知道,一个字符串若是是回文字符串,那么字符串须要知足的条件:字符串中每一个字母的个数必须是偶数,或者整个字符串只有一个字母的个数是奇数!好比串"aabbccccdd"即每一个字母个数均为偶数,又如“aaabbcc”,只有一个字母a的个数是奇数,获得解题方法:测试
举个例子会更加清晰,好比字符串“aaabbccccdd”, a :3个, b:2个 , c:4个, d:2个 ,字符串长度为11,那么按公式来算就是ui
result = (11/2)! / 【(a/2)! * (b/2)! * (c/2)! * (d/2)! 】= 5! / (1! * 1! * 2! * 1!) = 120 / 2 = 60,这个方法是采用数学里面的排列组合的知识,应该是高中学的!spa
有学过排列组合的人应该不难想到这种解题方法,可是公式中使用了大量的阶乘,要知道阶乘比指数增加更快更恐怖,int,long等类型的变量是没法存储那么大的数字的,好比20的阶乘是243 2902 0081 7664 0000,程序可能会用到50的阶乘,各个类型的范围以下:
unsigned int :0~4294967295
int :2147483648~2147483647 (10位)
unsigned long :0~4294967295
long : 2147483648~2147483647
long long的最大值:9223372036854775807 (19位)
long long的最小值:-9223372036854775808
unsigned long long的最大值:18446744073709551615 (20位)
__int64的最大值:9223372036854775807
__int64的最小值:-9223372036854775808
unsigned __int64的最大值:18446744073709551615
正由于数据如此庞大,题目才会提出对1000000007取模的要求,我尝试了不少方法,试图避免溢出的状况,最后获得一个比较合适的方法,既然数值有可能溢出,那么咱们能够采用化整为零的方法,不直接计算分子中的(Lenth/2)!,将其认为两步走,
第一步,在作 (Lenth/2)!的同时( sum存储结果,sum = sum * k,k = 1,2,3 ……Lenth/2),对(arr[ n ]/2)! 作除法,根据数学的常识,存在k指的sum能被(arr[ n ]/2)! 整除,直到(arr[ n ]/2)! 整除结束,即n = 25,记下此时的 sum 和 k;
第二步,继续作(Lenth/2)!,开始位置为 iter= k+ 1,result = sum(result即回文字符串的个数),循环体为result = (result * iter) % 1000000007; iter = k+1,k+2,……Lenth/2; 循环结束时获得result的值,改值就是回文字符串的个数!
注:再啰嗦一点,来解释一下循环体的内容result = (result * iter) % 1000000007,若是在循环体中不对1000000007取余,极可能会出现溢出的状况,其数学原理为:对于整数a ,M ,r有 a % M = r, 那么 存在k,使得 a = kM + r,两边同时乘以n,获得na = knM + nr, 因此有(na) % M = (nr) % M
以上即是我对这个题目的理解,下面将贴出个人程序的关键算法:
1 for(iter = 1; iter <= len/2; iter++) 2 { 3 MUL = MUL * iter; 4 while((arr[iterM] ==0 || arr[iterM] == 1) && iterM <26 ) 5 { 6 iterM++; 7 } 8 if(iterM < 26 && MUL % arr[iterM] == 0) 9 {10 MUL = MUL / arr[iterM];11 iterM++;12 }13 if(iterM >= 26)14 {15 death = iter ;16 result = MUL;17 break;18 }19 }20 for(iter = death + 1; iter <= len/2 ; iter++)21 {22 result = (result * iter) % 1000000007;23 }
View Code
MUL初值为1,数组arr[26]中存储的数值不是每一个字母的个数,而是(字母个数/2)的阶乘,对,想必您也考虑到了,这里可能会出现溢出的状况,若是极端一点,字符串中存在一个字母的个数超过了50,算一下25的阶乘,那么这个算法就会溢出了,可是这种状况好像也能够特殊处理,我暂时还没想到比较好的办法,但愿在这获得大神的指教!
个人程序最终顺利经过了在线编程系统的检测,而且得到了相应的10个积分!下面罗列一些正确的测试用例,供你们使用!我没有贴出完整代码,须要的能够留下邮箱地址!
用例1: s = “xvwsxafqkcawfchxggvsxkq” result = 19958400
用例2: s = “hqaymehhrsfuqrpahrimsxftuxqrpsejouuehaqtsryxjhearxmogmi” result = 676517829
用例3: s = “zsmjajrycysuqjvyyraqvoyggmjgsuiyvclurvmygoivmsurgxsyyblvbgxsszlsly” result = 493584849
用例4: s = “btxgnhdjscrqkqvwzrhtqlekxrgettfvtzcfhtzlhqckkdkntwwrotzzfskddocobfdrkvqozkrxoqqjxcvdcqwo” result = 376760527