【C++刷题集】-- day4

news2025/1/6 19:15:10

目录

选择题​​​​​​​ 

单选

编程题

计算糖果⭐

【题目解析】

【解题思路】

进制转换⭐​​​​​​​

【题目解析】

【解题思路】


选择题 

单选

1、 有以下程序
#include<iostream>
#include<cstdio>
using namespace std;
int main() {
	int m = 0123, n = 123;
	printf("%o %o\n", m, n);
	return 0;
}
程序运行后的输出结果是 ( )
0123 0173
0123 173
123 173
173 173

正确答案:


解析:

转换说明符
转换说明符描述
%d有符号十进制整数
%i有符号十进制整数
%u无符号十进制整数
%o八进制整数
%x十六进制整数(小写字母)
%X十六进制整数(大写字母)
%f浮点数
%e用科学计数法表示的浮点数(小写字母e)
%E用科学计数法表示的浮点数(大写字母E)
%g根据数值的大小自动选择%f%e格式
%G根据数值的大小自动选择%f%E格式
%c单个字符
%s字符串
%p指针地址
%n将已打印字符数保存在整型指针中
%%打印一个百分号

本题中:

  • m为前缀0 -> 八进制。
  • n为无前后缀 -> 十进制。 

所以根据八进制输出,就是123与173(8^2 * 1 + 8^1 * 7 + 8^0 * 3 = 123)

----------------------------------------------

2、 以下哪个选项一定可以将flag的第二个bit置0 ( )
flag&=~2
flag|=2
flag^=2
flag>>=2

正确答案:


解析:

位操作符名称功能
&按位与将两个操作数的每一位进行比较,如果两个操作数的对应位都为1,则该位结果为1,否则为0。
|按位或将两个操作数的每一位进行比较,如果两个操作数的对应位都为0,则该位结果为0,否则为1。
^按位异或将两个操作数的每一位进行比较,如果两个操作数的对应位相同,则该位结果为0,否则为1。
~按位取反将一个二进制数的每一位取反(0变成1,1变成0)。
左移<<将一个二进制数向左移动指定的位置。左侧空出来的位置用0填充。
右移>>将一个二进制数向右移动指定的位置。右侧空出来的位置用0填充(如果原始数字是正数),或者用1填充(如果原始数字是负数)。
#include <iostream>
using namespace std;

int main() {
   unsigned int a = 60; // 0011 1100
   unsigned int b = 13; // 0000 1101
   int result = 0;

   result = a & b; // 0000 1100
   cout << "a & b = " << result << endl;

   result = a | b; // 0011 1101
   cout << "a | b = " << result << endl;

   result = a ^ b; // 0011 0001
   cout << "a ^ b = " << result << endl;

   result = ~a; // 1100 0011
   cout << "~a = " << result << endl;

   result = a << 2; // 1111 0000
   cout << "a << 2 = " << result << endl;

   result = a >> 2; // 0000 1111
   cout << "a >> 2 = " << result << endl;

   return 0;
}
// 输出:
a & b = 12
a | b = 61
a ^ b = 49
~a = -61
a << 2 = 240
a >> 2 = 15

对于比特位设为0 / 1对应的位运算是:

#include <iostream>
using namespace std;

int main() {
    int flag = 5; // 5:二进制表示为101b
    cout << "flag: " << flag << endl;

    // 将第二个bit置0
    flag &= ~2; // 2:二进制表示为10b
    cout << "flag after clearing bit 2: " << flag << endl;

    // 将第二个bit置1
    flag |= 2; // 2:二进制表示为10b
    cout << "flag after setting bit 2: " << flag << endl;

    return 0;
}

----------------------------------------------

3、 请声明一个指针,其所指向的内存地址不能改变,但内存中的值可以被改变。
const int const *x = &y;
int * const x = &y;
const int *x = &y;
int const *x = &y;
const int * const x = &y;

正确答案:


解析:

const修饰指针的关键:

  • const在*前:const修饰的是解引用的结果不能修改,即指向的执行空间中的内容不能修改。
  • const在*后:const修饰指针变量本身,即指针的值不能修改,指针的指向不能修改。

题目选项:

  1. const int * const x = &y; 这个声明表示 x 是一个指向常量整数的指针,同时也是一个常量指针,即该指针所指向的内存地址和内存中的值都不能被修改。
  2. int * const x = &y; 这个声明表示 x 是一个指向整数的常量指针,即该指针所指向的内存地址不能被修改,但内存中的值可以被修改。
  3. const int * x = &y; 这个声明表示 x 是一个指向常量整数的指针,即该指针所指向的内存地址不能被修改,但内存中的值可以被修改。
  4. int const * x = &y; 这个声明和第三种声明等价,即x也是一个指向常量整数的指针
  5. const int * const x = &y; 这个声明表示 x 是一个既是常量又是常量指针的变量,即该变量所指向的内存地址和内存中的值都不能被修改。

----------------------------------------------

4、 以下C语言指令:
int a[5] = { 1,3,5,7,9 };
int* p = (int*)(&a + 1);
printf("% d, % d", *(a + 1),* (p - 1));

运行结果是什么?

2,1
3,1
3,9
运行时崩溃

正确答案:


解析:

考察核心:数组名只有在 和 sizeof 之后,才表明数组本身,其余都表示首元素的地址。

  • int a[5] = {1,3,5,7,9}; 定义了一个长度为5的整型数组a,并初始化。
  • int* p = (int*)(&a +1); 定义了一个指向整型数据类型的指针 p(int(*)[5]),将 p 的地址 + 1,即指向了 a[4] 后面的地址。

  • printf("%d,%d", *(a+1),*(p-1)); 输出 a[1] 和 p[-1] 所指向的值,即3和9。其中*(a+1)等价于a[1],*(p-1)等价于p[-1]。

----------------------------------------------

5、 二维数组X按行顺序存储,其中每个元素占1个存储单元。若X[4][4]的存储地址为0xf8b82140,X[9][9]的存储地址为0xf8b8221c,则X[7][7]的存储地址为 ( ) 
Oxf8b821c4
Oxf8b821a6
Oxf8b82198
Oxf8b821c0

正确答案:


解析:

        不知道有多少行,每行有多少列。假设:数组X的起始地址为start,总共有M行和N列。

  • &X[4][4] = start + 4*N + 4 = 0xf8b82140
  • &X[9][9] = start + 9*N + 9 = 0xf8b8221c
  • &X[7][7] = start + 7*N + 7

        在这个时候我们就可以发现,就是简单的二元一次方程了,但是还是很难进行计算,所起其实这里我们还需要进行题目关键点的提取,用以进行进一步的简化运算。 

  • &X[4][4] = 0xf8b82140 -> &X[4][9] = 0xf8b82145
  • 而 X[4][9] 和 X[9][9] 之间刚相隔了 5 行:0xf8b8221c - 0xf8b82145 = 5*N

        于是:0x21c - 0x145 = 0xD7 -> N = 43。

  • &X[7][7] = &X[7][9] - 2
  • &X[7][9] X[4][9]  + 3*43

        最后:&X[7][7] =  Oxf8b821c4

----------------------------------------------

6、 根据下面递归函数:调用函数Fun(2),返回值是多少 ( )
int Fun(int n)
{
	if (n == 5)
		return 2;
	else
		return 2 * Fun(n + 1);
}
2
4
8
16

正确答案:


解析:

----------------------------------------------

7、 以下程序的输出结果是:
#include <iostream>
using namespace std;
void func(char** m) {
	++m;
	cout << *m << endl;
}
int main() {
	static char* a[] = { "morning", "afternoon", "evening" };
	char** p;
	p = a;
	func(p);
	return 0;
}
afternoon
字符o的起始地址
字符o
字符a的起始地址

正确答案:


解析:

考察核心:数组名只有在 和 sizeof 之后,才表明数组本身,其余都表示首元素的地址。

        此处是 p = a,就是a的首元素的地址,而该数组的首元素 "moring" 类型为char*,所以 p 就需要给成二级指针。

        于是将p传给m,所以m指向首元素 "moring",随后前置++。也就是让这个指针向后偏移一个元素的大小,便是下一个元素 "afternoon" 。


        数组 a 的格式可以想象为如下:

----------------------------------------------

8、 求函数返回值,输入x=9999
int func(int x) {
	int count = 0;
	while (x)
	{
		count++;
		x = x & (x - 1);//与运算
	}
	return count;
}
8
9
10
12

正确答案:


解析:

假设 x = 7 -> 111

  • 第一次循环:count++ -> count = 1,x = 7&6 = 111&110 = 110,x = 6。
  • 第二次循环:count++ -> count = 2,x = 6&5 = 110&101 = 100,x = 4。
  • 第三次循环:count++ -> count = 3,x = 4&3 = 100&011 = 0,x = 0。

最终放回的就是3。

#注:这个算法的功能就是,返回x中总共有多少个1。

        于是接下就是将9999转换为二进制就可以了,10011100001111b,有8个零。

计算方式此处以5举例:

----------------------------------------------

9、 下列程序执行后,输出的结果为 ( )
#include <stdio.h>
int cnt = 0;
int fib(int n) {
	cnt++;
	if (n == 0)
		return 1;
	else if (n == 1)
		return 2;
	else
		return fib(n - 1) + fib(n - 2);
}
void main() {
	fib(8);
	printf("%d", cnt);
}
41
67
109
177

正确答案:


解析:

        cnt实际在统计,斐波那契递归总的次数。

----------------------------------------------

10、 在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是 ( )
struct A
{
	int a;
	short b;
	int c;
	char d;
};

struct B
{
	int a;
	short b;
	char c;
	int d;
};
16,16
13,12
16,12
11,16

正确答案:


解析:

结构体的大小计算遵循结构体的对齐规则:

  • 结构体的第一个成员放在结构体变量在内存中存储位置的0偏移处开始

  • 从第2个成员往后的所有成员,都要放在一个对齐数(成员的大小和默认对齐数的较小值)的整数的整数倍的地址处,VS中默认对齐数为8

  • 结构体的总大小是结构体的所有成员的对齐数中最大对齐数的整数倍。

  • 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍

        #注:VS中的默认对齐数为8,不是所有编译器都有默认对齐数,当编译器没有默认对齐数的时候,成员变量的大小就是该成员的对齐数。(Linux中就没有默认对齐数概念)


补充:

#结构体为什么要内存对齐?

  • 平台原因(移植原因):
    • 不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
  • 性能原因:
    • 数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。

总体来说:结构体的内存对齐是拿空间来换取时间的做法。

----------------------------------------------

编程题

计算糖果

计算糖果_牛客题霸_牛客网 (nowcoder.com)


【题目解析】

        A,B,C是三个人手里的糖果数量,我们不知道A,B,C是多少?但是我们知道A - B, B - C, A + B, B + C的结果,这个结果题目是通过输入测试用例给我们的。所以本题本质是一个表达式求解问题。

【解题思路】

  1. A - B、B - C、A + B、B + C 这道题目的实质是:判断三元一次方程组是否有解及求解。
  2. 最后需要判断一下逆推是否成立。
#include <iostream>
#include <vector>
using namespace std;

// 判断糖果数是否合理
inline bool judge_no(float num)
{
    if(num < 0 || (int)num < num) 
        return true;
    return false;
}

int main() {
    vector<int> val(5, 0);
    for(int i = 1; i < 5; i++)
    {
        int number = 0;
        cin >> number;
        val[i] = number;
    }

    float A = (val[1] + val[3]) / 2.0; 
    if(judge_no(A))
    {
        cout << "No" << endl;
        return 0;
    }
    float B = val[3] - A;
    if(judge_no(B))
    {
        cout << "No" << endl;
        return 0;
    }
    float C = val[4] - B;
    if(judge_no(C))
    {
        cout << "No" << endl;
        return 0;
    }

    // 反推合理性
    if(A - B != val[1] ||
       B - C != val[2] ||
       A + B != val[3] ||
       B + C != val[4])
    {
        cout << "No" << endl;
        return 0;
    }

    printf("%d %d %d", (int)A, (int)B ,(int)C);
    return 0;
}

进制转换

进制转换_牛客题霸_牛客网 (nowcoder.com)


【题目解析】

        简单的进制准换,使用我们平时所用的方式即可。

【解题思路】

        就是进行取模余数就是当前低进制的位的值是多少,通过除掉进制数,进入下一个进制位的计算。

计算方式此处以5举例:

#include <iostream>
#include <string>
#include <algorithm>
using namespace std;
int main() {
    const char* table = "0123456789ABCDEF";
    int m, n;
    cin >> m >> n;
    bool flag = false;

    // 如果是负数,则转成正数,并标记一下
    if (m < 0)
    {
        m = 0 - m;
        flag = true;
    }

    // 按进制换算成对应的字符添加到ret
    string ret;
    while (m)
    {
        ret += table[m % n];
        m /= n;
    }

    if (flag)
        ret += '-';
    if(ret.empty())
        ret += table[0];

    auto rit = ret.rbegin();
    while(rit != ret.rend())
        cout << *rit++;; 

    cout << endl;
    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/758035.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

(文章复现)售电市场环境下电力用户选择售电公司行为研究(附matlab代码)

参考文献&#xff1a; [1]孙云涛,宋依群,姚良忠等.售电市场环境下电力用户选择售电公司行为研究[J].电网技术,2018,42(04):1124-1131. 1.基本原理 1 .1演化博弈 与古典博弈理论相比较&#xff0c;演化博弈假设参与人是有限理性的&#xff0c;参与人会根据自己和他人的经验选…

【前端】网页开发精讲与实战 CSS Day 4

&#x1f680;Write In Front&#x1f680; &#x1f4dd;个人主页&#xff1a;令夏二十三 &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd; &#x1f4e3;系列专栏&#xff1a;前端 &#x1f4ac;总结&#xff1a;希望你看完之后&#xff0c;能对你有…

Vue+Ts+Echart使用以及后台接口对接逻辑【实战】

官网传送门 一.echarts介绍 是一个js插件 性能好可流畅远行PC和移动设备 兼容主流浏览器 提供很多图标,用户且可自行修改。 2.使用npm安装 npm install echarts二.echarts基本使用、自定义图例、选择7天日期查询图表数据内容 获取本地时间以及当前时间前几天后几天 /*** 获…

安装Visual Studio Installer Projects 2022插件

VS主界面--扩展--管理扩展--搜索VS插件“Visual Studio Installer Projects 2022”并安装。

JavaWeb(5)——HTML、CSS、JS 快速入门

一、JavaScript 对象 二、JavaScript BOM对象 和 DOM对象 关于BOM主要对 Window 和 location 进行说明&#xff1a; 三、JavaScript 事件监听 事件绑定 常见事件 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8">…

ORA-01122 ORA-01208 故障处理---惜分飞

数据库突然故障ORA-01122 ORA-01208,导致实例crash Tue Jul 11 09:06:43 2023 Thread 1 cannot allocate new log, sequence 254989 Private strand flush not complete Current log# 3 seq# 254988 mem# 0: E:\APP\ADMINISTRATOR\ORADATA\xff\REDO03.LOG Thread 1 advanced …

23款奔驰E300加装原厂香氛负离子系统,清香宜人,久闻不腻

奔驰原厂香氛合理性可通过车内空气调节组件营造芳香四溢的怡人氛围。通过更换手套箱内香氛喷雾发生器所用的香水瓶&#xff0c;可轻松选择其他香氛。香氛的浓度和持续时间可调。淡雅的香氛缓缓喷出&#xff0c;并且在关闭后能够立刻散去。车内气味不会永久改变&#xff0c;香氛…

javascript(js)组成及使用

组成&#xff1a; js是由核心ECMAScript、文档对象模型DOM、浏览器对象模型BOM三部分组成。 ECMAScript&#xff1a;定义了语言的语法、类型、语句、关键字、保留字、操作符、对象。有普通模式和严格模式的区分。文档对象模型(DOM)&#xff1a;由XML经过扩展用于HTML的应用程序…

代码随想录算法学习心得 45 | 300.最长递增子序列、674.最长连续递增序列、718.最长重复子数组...

一、最长递增子序列 链接&#xff1a;力扣 描述&#xff1a;给你一个整数数组 nums &#xff0c;找到其中最长严格递增子序列的长度。 子序列 是由数组派生而来的序列&#xff0c;删除&#xff08;或不删除&#xff09;数组中的元素而不改变其余元素的顺序。例如&#xff0c…

Android 开发规范(基础版)

背景 项目的代码时间时间很长,经过太多人手,代码的规范性堪忧,目前存在较多的比较自由的「代码规范」,这非常不利于项目的维护,代码可读性也不够高。 分析现有项目的代码的情况,输出的『定制化规范』文档,用于提高代码的可读性和可维护性。 收益 对于个人:帮助团队写「…

最长递增子序列(力扣)动态规划 JAVA

给你一个整数数组 nums &#xff0c;找到其中最长严格递增子序列的长度。 子序列 是由数组派生而来的序列&#xff0c;删除&#xff08;或不删除&#xff09;数组中的元素而不改变其余元素的顺序。例如&#xff0c;[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。 示例 1&#xf…

路径规划算法:基于袋獾优化的路径规划算法- 附代码

路径规划算法&#xff1a;基于袋獾优化的路径规划算法- 附代码 文章目录 路径规划算法&#xff1a;基于袋獾优化的路径规划算法- 附代码1.算法原理1.1 环境设定1.2 约束条件1.3 适应度函数 2.算法结果3.MATLAB代码4.参考文献 摘要&#xff1a;本文主要介绍利用智能优化算法袋獾…

【vueJs源码】阅读之vm.$watch函数

我们经常使用watch肯定知道它&#xff0c;他和computer一样都是数据发生变化都会触发它。今天我们就来了解一下它的原理。 他的用法 Vue.prototype.$watch function (expOrFn: string | (() > any),cb: any,options?: Record<string, any> ): Function这是vuejs源…

Redis数据类型 — String

目录 String类型的内部编码 存储字符串采用两种编码方式的优缺点 选择SDS的原因 存储整数型采用OBJ_ENCODING_INT的原因 String 类型的底层的数据结构实现主要是 long 和 SDS&#xff08;简单动态字符串&#xff09; 数据结构&#xff1a;SDShttps://blog.csdn.net/weixin…

六种最新智能优化算法(LSO、SWO、ZOA、EVO、KOA、GRO)求解23个基准测试函数(含参考文献及MATLAB代码)

一、六种算法简介 &#xff08;1&#xff09;光谱优化算法LSO 光谱优化算法&#xff08;Light Spectrum Optimizer&#xff0c;LSO&#xff09;由Mohamed Abdel-Basset等人于2022年提出。 参考文献&#xff1a;Abdel-Basset M, Mohamed R, Sallam KM, Chakrabortty RK. Light…

单链表基本操作(java)

单链表 一.自定义链表类二.自定义节点类三.链表中的基本方法1.头插法2.尾插法3.在任意位置插入4.删除第一次的关键字为key的节点5.删除所有关键字为key的元素6.是否包含关键字key7.获取链表长度8.遍历节点并输出9.清空链表 一.自定义链表类 public class MySingleList {Node h…

vue+element Cascader 级联选择器 > 实现省市区三级联动

vueelement Cascader 级联选择器 > 实现省市区三级联动 先看下实现效果吧&#xff08;嘻嘻&#xff09; 看完我们就开始啦 安装element-china-area-data1 npm install element-china-area-data5.0.2 -S上代码 <el-cascadersize"large":options"options…

我的创作纪念日-静下来走一段路

静下来走一段路 机缘 前段时间收到了公司发的入职周年庆典的邮件&#xff0c;发觉已经在程序员的工作上走了这么多年了。 原本想写些东西只是为了生成一些可供自己反复查看的笔记&#xff0c;后来为了督促自己学习&#xff0c;博客越学越多。 回忆起我写的第一篇博客《芯片、模…

移动端开发框架mui之上拉刷新、下拉加载数据(选项卡左右移动自动和刷新指定选项卡内容)

移动端开发框架mui之上拉刷新、下拉加载数据&#xff08;加载多选项卡数据&#xff09; 之前的代码&#xff1a; <!doctype html> <html><head><meta charset"utf-8"><title></title><meta name"viewport" cont…

JVM内存结构—— 程序计数器,虚拟机栈 解析

JVM的内存结构 1. 程序计数器(PC Register )寄存器 1.1 全称:Program Counter Register 1.2 作用 首先,java源代码 被 编译成 二进制的 字节码 (jvm指令) jvm跨平台就是这一套指令,linux 下,windows下指令都是一致的 指令 经过 解释器 把每一条指令 解释成 机器码…