蓝桥杯-代码填空之一

最后更新于:2022-04-01 14:53:18

N进制小数—车票找零方案计数—串的反转—串的轮换—大数分块乘法—二进制串转整数 ①N进制小数 将任意十进制正小数分别转换成2,3,4,5,6,7,8,9进制正小数,小数点后保留8位,并输出。例如:若十进制小数为0.795,则输出: 十进制正小数 0.795000 转换成 2 进制数为: 0.11001011 十进制正小数 0.795000 转换成 3 进制数为: 0.21011011 十进制正小数 0.795000 转换成 4 进制数为: 0.30232011 十进制正小数 0.795000 转换成 5 进制数为: 0.34414141 十进制正小数 0.795000 转换成 6 进制数为: 0.44341530 十进制正小数 0.795000 转换成 7 进制数为: 0.53645364 十进制正小数 0.795000 转换成 8 进制数为: 0.62702436 十进制正小数 0.795000 转换成 9 进制数为: 0.71348853 以下代码提供了这个功能。其中,dTestNo表示待转的十进制小数。iBase表示进制数。请填写缺失的部分。 ~~~ void fun(double dTestNo, int iBase) { int iT[8]; int iNo; printf("十进制正小数 %f 转换成 %d 进制数为: ",dTestNo, iBase); for(iNo=0;iNo<8;iNo++) { dTestNo *= iBase; iT[iNo] = ________________; if(___________________) dTestNo -= iT[iNo]; } printf("0."); for(iNo=0; iNo<8; iNo++) printf("%d", iT[iNo]); printf("\n"); } void main ( ) { double dTestNo= 0.795; int iBase; for(iBase=2;iBase<=9;iBase++) fun(dTestNo,iBase); printf("\n"); } ~~~ 这道题看到函数传进来的是一个小数和进制数, 题中建立一个大小为8的数组,再看看输出,就知道,是把每一个相应进制数存在数组中, 在本上算一算就发现,小数乘以相应进制,整数部分即为该进制第一个小数(这个是小数转换进制的基础知识), 显然题目,也是这么做的,填空就不难了,第一个就是取整数部分,直接强制类型转换就可以, 第二个就是判断什么时候需要减去整数部分,前面是*=,原来的小数变化了,求下一个数,肯定要把整数部分清零, 所以就是,当数组内容不为0的时候,也可以说是大于0的时候。 答案:(答案不唯一,只要能满足要求即可) 空1:  (int)dTestNo    空2:  dTestNo>=1.0   ②车票找零方案计数 公交车票价为5角。假设每位乘客只持有两种币值的货币:5角、1元。 再假设持有5角的乘客有m人,持有1元的乘客有n人。由于特殊情况,开始的时候,售票员没有零钱可找。 我们想知道这m+n名乘客以什么样的顺序购票则可以顺利完成购票过程。 显然,m < n的时候,无论如何都不能完成,m >=n的时候,有些情况也不行。 比如,第一个购票的乘客就持有1元。 下面的程序计算出这m+n名乘客所有可能顺利完成购票的不同情况的组合数目。 注意:只关心5角和1元交替出现的次序的不同排列,持有同样币值的两名乘客交换位置并不算做一种新的情况来计数。 //m: 持有5角币的人数 //n: 持有1元币的人数 //返回:所有顺利完成购票过程的购票次序的种类数 ~~~ int f(int m, int n) { if(m < n) return 0; if(n==0) return 1; return _______________________; } ~~~ 这道题,一看函数类型int 还有终止条件,显然是递归, 递归的话就简单了,你是 return f(m-1,n-1)+1还是 return f(m-1,n)+f(m,n-1) 呢? m-1,n-1就是 当前人数,先来一个五毛钱的,再来一个1元的, 那排序基本上就是 0.5  1   0.5   1显然方案不全, 所以就是m-1,n + m,n-1了, 答案: f(m-1, n) + f(m, n-1) ③反转串 我们把“cba”称为“abc”的反转串。 下面的代码可以把buf中的字符反转。其中n表示buf中待反转的串的长度。请补充缺少的代码。 ~~~ void reverse_str(char* buf, int n) { if(n<2) return; char tmp = buf[0]; buf[0] = buf[n-1]; buf[n-1] = tmp; _______________________________; } ~~~ 题目要求很简单,就是把串反转,看题干,也很明白它的做法, 就是取头尾,互换,下一步肯定是往里缩进一格,首位都缩进,肯定长度减少2了, 这样,接着把串和长度传下去就可以了,也是一个递归,因为传递的是指针,所以不需要返回值。 答案: reverse_str(buf+1,n-2) (答案不唯一) ④串的轮换 串“abcd”每个字符都向右移位,最右的移动到第一个字符的位置,就变为“dabc”。这称为对串进行位移=1的轮换。同理,“abcd”变为:“cdab”则称为位移=2的轮换。 下面的代码实现了对串s进行位移为n的轮换。请补全缺失的代码。 ~~~ void shift(char* s, int n) { char* p; char* q; int len = strlen(s); if(len==0) return; if(n<=0 || n>=len) return; char* s2 = (char*)malloc(_________); p = s; q = s2 + n % len; while(*p) { *q++ = *p++; if(q-s2>=len) { *q = ___________; q = s2; } } strcpy(s,s2); free(s2); } ~~~ 这道题,我是从下往上看的,看到最后有个 strcpy(s,s2) 和 free(s2)  就明白, 它的方法就是建立一个中间串s2,s2存储正确顺序,最后直接赋值给s, 过程,malloc肯定就是开辟s2的空间,开辟空间大小显然与s长度len有关, 后面p指针指向s头部,q指向中间串s2的相应位置,n%len 就是计算具体指向那个地方, 然后通过循环,当p!=NULL时,将p指向内容赋值给q指向内容,然后两者再往后移动, 这里要注意是先赋值再移动, 假如题目是  abcd 2,那么通过上述循环,s2串内容将是   空空ab  (空代表什么都没有) 那个if就是判断q是否指到了末尾,当指到末尾,就给q赋值为NULL, 将q指向s2头部,接着赋值。这样就达到了构建s2的目的。 简单的说,就是: p是从s头指向尾不变, 而q从s2中间位置向后移动,如果长度大于等于串长度,再指向s2头部, 以 abcd 2为例就是: p从指向a移动到d, q先指向c的位置,将a,b,赋值到s2串第3,4的位置,if成立,所以将后面第5个设置NULL, 指回s2头部,这是p指向的是c,到指到d下一个是空,所以循环跳出,s2串构建完成。 答案: len+1 0(指的是空,也可以写成NULL或者‘\0'或者(char)0 ⑤大数分块乘法   对于32位字长的机器,大约超过20亿,用int类型就无法表示了,我们可以选择int64类型, 但无论怎样扩展,固定的整数类型总是有表达的极限!如果对超级大整数进行精确运算呢? 一个简单的办法是:仅仅使用现有类型,但是把大整数的运算化解为若干小整数的运算,即所谓:“分块法”。   如图 ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-22_576a44af781f3.jpg) 表示了分块乘法的原理。可以把大数分成多段(此处为2段)小数,然后用小数的多次运算组合表示一个大数。 可以根据int的承载能力规定小块的大小,比如要把int分成2段,则小块可取10000为上限值。 注意,小块在进行纵向累加后,需要进行进位校正。 以下代码示意了分块乘法的原理(乘数、被乘数都分为2段)。 ~~~ void bigmul(int x, int y, int r[]) { int base = 10000; int x2 = x / base; int x1 = x % base;  int y2 = y / base; int y1 = y % base;  int n1 = x1 * y1;  int n2 = x1 * y2; int n3 = x2 * y1; int n4 = x2 * y2; r[3] = n1 % base; r[2] = n1 / base + n2 % base + n3 % base; r[1] = ____________________________________________; // 填空 r[0] = n4 / base; r[1] += _______________________;  // 填空 r[2] = r[2] % base; r[0] += r[1] / base; r[1] = r[1] % base; } int main(int argc, char* argv[]) { int x[] = {0,0,0,0}; bigmul(87654321, 12345678, x); printf("%d%d%d%d\n", x[0],x[1],x[2],x[3]); return 0; } ~~~ 这题应该比较简单了,根据图很容易填上第一个空, 要注意它的r[3]、r[2]、r[1]、r[0]对应图中r4,r3,r2,r1, 第二个空,根据下面r0也可以照葫芦画瓢的填出来。 答案:(答案不唯一) n2 / base + n3 / base + n4 % base    r[2] / base                      ⑥二进制串转整数 下列代码把一个二进制的串转换为整数。请填写缺少的语句; ~~~ char* p = "1010110001100"; int n = 0; for(int i=0;i<strlen(p); i++) { n = __________________; } printf("%d\n", n); ~~~ 就是将2进制的串转换成10进制, 如果有cmath库函数,直接用个Pow, 这题好像没提供, 但是,不要忘了,将10进制转换2进制可以用while循环做: ~~~ i=0 while(num>2) { arr[i++]=num%2; num/=2; } ~~~ 这题完全可以逆着来。 还要注意一下,题目中的是字符,而不是数字。 答案: n * 2 + (p[i] - '0') 上述题目的答案都不是唯一的,当答案和标准答案不一样时,会将答案带入程序运行,通过运行结果来判断正误。
';