c语言每日一练(3)
前言:
每日一练系列,每一期都包含5道选择题,2道编程题,博主会尽可能详细地进行讲解,令初学者也能听的清晰。每日一练系列会持续更新,暑假时三天之内必有一更,到了开学之后,将看学业情况更新。
五道选择题:
1、执行下面的程序段,语句3的执行次数为( )
for(i = 0; i <= n-1; i++) // (1)
for(j = n; j > i; j--) // (2)
state// (3)
A.n(n+2)/2 B.(n-1)(n+2)/2 C.n(n+1)/2 D.(n-1)(n+2)
解析:先看代码构成,两个循环嵌套在了一起,每一次循环都会执行一次语句3,那么我们的目标就是计算出总循环的次数。
第一层循环的i被初始化为0,循环继续条件为i<=n-1,且每次循环i+1。继续观察,发现没有别的可以影响i的值的因素了,0~n-1次数为n,因此第一层循环的次数为n。
接下来看第二层循环,第二层循环的j被初始化为了n,循环继续条件为j>i,且每次循环j减1。第二层循环受到i的值的影响,我们在做这种题时可以先代几个值进去尝试。
当i=0时,走到第二层循环,那么此时循环继续进行的条件就变成了j>0,j被初始化为n,每次循环j减1。那么第二层循环就会走n次。
我们接着走,当i=1时,走到第二层循环,那么此时循环继续进行的条件就变成了j>1,j又会被初始化为n,那么循环的次数就会是n-1
当i=2时为n-2.......一直到i=n-1时二层循环走1次可以看出
总共的循环次数是n+n-1+n-2+.....+1最后的结果为(1+n)*n/2,所以答案是C
2、对于代码段,下面描述正确的是( )
t=0;
while(printf("*"))
{
t++;
if (t<3)
break;
}
A. 其中循环控制表达式与0等价 B. 其中循环控制表达式与'0'等价
C. 其中循环控制表达式是不合法的 D. 以上说法都不对
解析: 观察四个选项,发现它们都避不开循环控制表达式,所以我们从循环控制表达式入手。循环控制表达式为printf("*"),而printf是有返回值的,它在屏幕上输出了几个字符,返回值就为几。printf("*")只会在屏幕上打印出一个*字符,所以这个控制表达式可以等价为1,也就是可以将循环看作while(1),循环条件恒为真,也就是死循环。
因此,如果有表达式可以实现循环控制表达式恒为真,那就等价于我们的循环控制表达式。
先看A选项,A:循环控制表达式与0等价,显然是错的,0恒为假,一次循环都不会进去更别说死循环了。
再看B选项,B:循环控制表达式与'0'等价,'0'的ASCII码值是48,48非0,循环控制表达式恒为真,故答案为B。
C选项是循环控制表达式不合法,一般来说,要出现循环控制表达式不合法,很可能是你的语法出错了,而题目中的死循环,虽然它死循环了,但循环是会进行的,所以是合法的,C错。D不用看,一般都不会选这个。所以最后答案为B
3、如下函数是求两个int数字最大公约数的,指出其中存在的问题【多选】( )
int gcd(char x,char y)
{
int min = x < y ? x : y;
for (min = 0; min > 0; min--)
if (x % min = 0 && y % min = 0)
return min;
}
A. 参数类型不对 B.循环变量min初值不对 C.判断等于的符号不对 D.返回类型不对
解析:题目函数的用途是求两个数字的最大公约数,让我们指出代码存在的问题。这种问题,即使你不知道怎么用代码求最大公约数也是能做的。
先看A选项,A说参数类型不对,那么我们观察可以发现,x,y是char型的,而题目要求两个int数字的最大公约数,参数类型不匹配,故要选A。
再看B选项,B说循环变量min初值不对,min初值被设成了0,而循环条件为min>0,因此循环一次都不会进行,更不会走到if的判断中。会直接走到return min上,最后只会返回x,y中的较小的那个,这样求最大公约数显然是错的,故要选B。
再看C选项,C说判断等于的符号不对,观察发现问题代码,if (x % min = 0 && y % min = 0)我们判断相等时要用两个=号,一个=号是赋值的意思,故要选C。
最后再看D选项,D说返回类型不对,min是int型的,而函数的返回类型也是故D错,不选。综上所述,我们的答案是ABC
4、下列条件语句中,功能与其他语句不同的是( )
A. if(a) printf("%dn",x); else printf("%dn",y);
B. if(a==0) printf("%dn",y); else printf("%dn",x);
C. if (a!=0) printf("%dn",x); else printf("%dn",y);
D. if(a==0) printf("%dn",x); else printf("%dn",y);
解析:找不同,这个很简单,顺着走看功能就行
先看A选项,if(a) printf("%dn",x); else printf("%dn",y);顺着读就是,当a为非0值时,打印x,当a为0时打印y
再看B选项,if(a==0) printf("%dn",y); else printf("%dn",x);当a为0时打印y,当a为非0打印x,和A一样
再看C选项,if (a!=0) printf("%dn",x); else printf("%dn",y);当a不等于0时打印x,为0时打印y,和A一样
最后看D选项,if(a==0) printf("%dn",x); else printf("%dn",y);当a等于0时打印x,不等于0时打印y,和之前的都不一样,故答案为D
5、我们知道C语言的 break 语句只能跳出离它最近的一层循环,可是有时候我们需要跳出多层循环,下列跳出多层循环的做法正确的是【多选】( )
A. 将程序写成函数用return结束函数,便可跳出循环
A解析:使用return可以直接结束函数,别说跳出循环了,它直接撂担子不干了,所以A正确
B. 修改外层循环条件例如
for (int i = 0; i < MAX1; i++)
{
for (int j = 0; j < MAX2; j++)
{
if (condition)
{
i = MAX1;//1
break;//2
}
}
//3
}
B解析:当condition满足的时候,i=MAX1,然后break,break跳出一层循环,跳到3的位置,继续走发现没有代码执行了,跳到开始的条件判断,发现不满足i<MAX1这个条件,故循环结束。可以看出,当condition满足时,直接跳出这个多层循环,故B正确
C. 在外层循环设置判断条件例如
for (; symbol != 1 && condition2; )
{
for (; symbol != 1 && condition3; )
{
if (condition1)
symbol = 1;
}
}
C解析:当condition满足时,symbol被赋值为1。可以看出,两层循环进行的条件均是symbol!=1,也就是说, 当symbol不等于1时两层循环才会有进行的可能,所以当symbol为1时,两层循环都会被跳出,C正确
D. 在外层循环后面加入break例如
for ( ;symbol; )
{
for ( ;symbol!=1; )
{
if (condition)
symbol = 1;//1
}
if (symbol == 1)//2
break;
}
D解析:当condition满足时,symbol等于1,会跳到2的位置。symbol等于1,满足if(symbol==1),故跳出循环,D正确。综上所述,答案为ABCD
编程题1:
int GetNumberOfK(int* nums, int numsLen, int k) {
int i = 0; int count = 0;
for (i = 0; i < numsLen; i++)
//遍历一遍数组
{
if (nums[i] == k)
//k是目标,当等于时计数器+1
{
count++;
}
}
return count;//返回统计结果
}
补充:二分法
提醒:二分法对于这道题而言可读性太差,没特殊要求的情况下不建议使用,这里博主也不会过多解释,只能说只可意会不可言传,得靠自己领悟,初学者掌握不了也是很正常的,因为这道题二分法要考虑的东西很多
int two_search(int* data, int len, int k, int flag) {//flag:0-找左边, 1-找右边
int left = 0, right = len - 1, mid;
while (left <= right) {
mid = left + (right - left) / 2;
if (data[mid] > k)
right = mid - 1;
else if (data[mid] < k)
left = mid + 1;
else {
if (flag == 0) {//flag==0时,找最左边的数字
if (mid == left || data[mid - 1] != k) return mid;
else right = mid - 1;//把中心向左推
}
else {//flag==1时,找最右边的数字
if (mid == right || data[mid + 1] != k) return mid;
else left = mid + 1;//把中心向右推
}
}
} return -1;
} int GetNumberOfK(int* data, int dataLen, int k) {
if (dataLen == 0) return 0;
int left = two_search(data, dataLen, k, 0);
int right = two_search(data, dataLen, k, 1);
if (left == -1 && right == -1) return 0; //表示没有找到k这个数据
return right - left + 1;
}
编程题2:
面试题 05.06. 整数转换 - 力扣(LeetCode)
int convertInteger(int A, int B) {
unsigned int x = A ^ B; int count = 0;
//异或(^)可以将A和B的两个值相同的位变为0,不同的位变为1
//因此此时的x的位上的1的数量为A和B不同的位的个数
while(x)
{
if(x&1==1)
//与1进行且运算,1只有最后一位为1,其他位为0
//可以将x最后一位的值剥离出来
//当剥离的结果为1意味着,找到了1个A和B不同的位
{
count++;
}
x>>=1;
//将x右移1位,如此可以遍历x上的所有位
}
return count;
//返回计数结果
}
好了,今天的练习到这里就结束了,感谢各位友友的来访,祝各位友友前程似锦O(∩_∩)O