算法初阶双指针+C语言期末考试之编程题加强训练

双指针

常⻅的双指针有两种形式,⼀种是对撞指针,⼀种是左右指针。
对撞指针:⼀般⽤于顺序结构中,也称左右指针。
• 对撞指针从两端向中间移动。⼀个指针从最左端开始,另⼀个从最右端开始,然后逐渐往中间逼
近。
• 对撞指针的终⽌条件⼀般是两个指针相遇或者错开(也可能在循环内部找到结果直接跳出循
环),也就是:
◦ left == right (两个指针指向同⼀个位置)
◦ left > right (两个指针错开)
快慢指针:⼜称为⻳兔赛跑算法,其基本思想就是使⽤两个移动速度不同的指针在数组或链表等序列
结构上移动。
这种⽅法对于处理环形链表或数组⾮常有⽤。
其实不单单是环形链表或者是数组,如果我们要研究的问题出现循环往复的情况时,均可考虑使⽤快
慢指针的思想。
快慢指针的实现⽅式有很多种,最常⽤的⼀种就是:
• 在⼀次循环中,每次让慢的指针向后移动⼀位,⽽快的指针往后移动两位,实现⼀快⼀慢。

 

1. 移动零(easy)

「数组分两块」是⾮常常⻅的⼀种题型,主要就是根据⼀种划分⽅式,将数组的内容分成左右两部
分。这种类型的题,⼀般就是使⽤「双指针」来解决。
1. 题⽬链接:283. 移动零
2. 题⽬描述:
给定⼀个数组 nums ,编写⼀个函数将所有 0 移动到数组的末尾,同时保持⾮零元素的相对顺
序。
请注意 ,必须在不复制数组的情况下原地对数组进⾏操作。
⽰例 1:
输⼊: nums = [0,1,0,3,12]
输出: [1,3,12,0,0]
⽰例 2:
输⼊: nums = [0]
输出: [0]
思路:用cur指针扫描整个数组,另一个dest指针记录非零序列最后一个位置,根据cur扫描过程,分类划分,实现数组的划分,这样使得[0,dest]都是非0元素,[dest+1,cur-1]都是0.
具体实现:
class Solution {
public:
    void moveZeroes(vector<int>& nums) 
    {
        int cur = 0;
        int dest = -1;
        while(cur<nums.size())
        {
            if(nums[cur]!=0)
            {
                dest++;
                swap(nums[cur],nums[dest]);
            }
            cur++;
        }
    }
};

核心思路就是,选用cur指向第一个元素的左边第一个元素,让dset往右边扫描,遇到0就走,遇到非零停,然后让cur++,目的是让他指向0,然后交换,循环的条件是cur<nums.size(),直到最后一个元素为止。

第一题:阶乘求和

31a72cc652d540e68762a289ddfd5798.png思路:

遍历n,然后item来存放每一项,并且后一项是在前一项的基础上进行相乘,值得说的是,这里你定义的变量不能是int否则不能拿满分,原因是int类型包含的范围小,当n很大的时候,阶乘之和可能超过这个范围,导致发生溢出,因此应该用更大数据类型,如long long类型

#include <iostream>
using namespace std;

// 计算阶乘
long long factorial(int n) {
    long long result = 1;
    for (int i = 1; i <= n; i++) {
        result *= i;
    }
    return result;
}

int main() {
    int n;
    cin >> n;
    long long sum = 0;
    for (int i = 1; i <= n; i++) {
        sum += factorial(i);
    }
    cout << sum;
    return 0;
}

第二题:完数的判断

9202f442b931426598575497445117e4.png分析:

找出n内所有完数,那么就先遍历n,定义一个变量sum对它的因子进行求和,,因为因子是变化的这里这里定义一个数组将其全部存起来,然后判断sum与i是否相等输出即可

#include<stdio.h>
void PerfNumber(int n);
int main()
{
    int N;
    scanf("%d",&N);
    PerfNumber(N);
}
void PerfNumber(int n)
{
    for(int i=2;i<=n;i++)
     {
         int sum=0,k=0;
         int a[100]={0};
          for(int j=1;j<i;j++)
          {
              if(i%j==0)
              {
                  sum+=j;
                  a[k]=j;
                  k++;
              }
          }
          if(sum==i)
          {
              k=0;
              printf("%d its factors are ",i);
              while(a[k])
              {
                  printf("%d ",a[k]);
                  k++;
              } printf("\n");
          }
     }
     return;
}

第三题:字符串的输入输出处理

8d7de0936174441e93aadbd85d418971.png思路:

解决这道题还是用istringstream来处理,并且定义了一个函数来实现字符串的切割。更方便处理字符串流的文本,并且注意当读取n的时候,要把末尾\n给去掉,可以用getchar也可以用cin.ignore(),然后是多组输入,这里用一个while循环,即一直读取str,当小于n时原样输出,大于时,输出字符串流。

#include <iostream>

#include <string>

#include <sstream>

#include <vector>

using namespace std;



void stringSplit(string str); //这里加不加const都行,加上是因为编程习惯,这个符号在函数中不能修改
int main() {

    int n, i = 0;

    string str;

    cin >> n;

    getchar();

    while (getline(cin, str)) {

        if (i++ < n) {

            cout << str << endl << endl;

        }
        else {
            stringSplit(str);
        }
    }
return 0;
}

void stringSplit(string str) {

    istringstream iss(str);

    string receive;

    while (iss >> receive)
    {
        cout << receive << endl << endl;
    }

}

第四题:线性筛求素数(了解即可)

505388fb8beb4083838db1815adb127d.png

思路:这里用线性筛给大家实现,线性筛比普通遍历的方法效率更高,当处理一些比较麻烦的数据时,线性筛的作用就发挥出来了,核心在于去重,有兴趣博友可以去b站搜一下具体的讲解,这里不做阐述。

#include<bits/stdc++.h>
using namespace std;
//int isprime(int n);
const int maxn = 1e6;
int prime[maxn + 1];//放素数
bool isprime[maxn + 1];
int cnt = 0;//素数个数
void linesieve(int n)
{
	memset(isprime, true, sizeof(isprime));//全部弄成true
	for (int i = 2; i <= n; i++)
	{
		//遍历所有
		if (isprime[i])
		{
			prime[++cnt] = i;
		}
		for (int j = 1; j <= cnt && prime[j] * i <= n; ++j)
		{
			isprime[prime[j] * i] = false;
			if (i % prime[j] == 0)break;
		}
	}
	for (int i = 2; i <= n; i++)
	{
		if (isprime[i])
			cout << i << " " << endl;
	}
}
int main()
{
	int m;
	cin >> m;
	linesieve(m);
	return 0;
}

第五题:自定义函数处理最大公约数与最小公倍数

faabd99eb2db485a9b986a7a735b6023.png思路:

题目很简单,但也是基础说不定就考了呢,这里介绍辗转相除法求最大公约数和简单自增来求最小公倍数。求最大公倍数,首先要知道最大公倍数一定比这两个数都大,因此我们从a,b种大的数开始自增,直到这个数都能整除a,b,返回值即可。辗转相除法,有兴趣去b站学一下,这里不做阐述,给出代码,即先定义mod,每次循环,把b赋给a,把mod赋给b,直到b等于0跳出循环。

#include<bits/stdc++.h>
using namespace std;
int gcd(int a, int b);
int ret(int a, int b);
int main()
{
	int a, b;
	cin >> a >> b;
	int gcd1 = gcd(a, b);
	int ret1 = ret(a, b);
	cout << gcd1 << " " << ret1;
	return 0;
}
int gcd(int a, int b)
{
	int mod;
	while (b)
	{
		mod = a % b;
		a = b;
		b = mod;
	}
	return a;
}
int ret(int a, int b)
{
	if (a > b)
		swap(a, b);//确保a最大
	int i;
	for (i = a;; i++)
	{
		if (i % a == 0 && i % b == 0)
			break;
	}
	return i;
}

第六题:蓝桥杯2022-刷题统计

96c87fd1df914af1a4caacbca2fe41b0.png思路:

这道题虽然我只有80分,但还是想给大家介绍80 分的做法,就是学习这种分析,一步步整。首先我们要输出做题天数,并且周一到周五的做题量和周六周日不同,那么分开考虑,n为总题量,我们定义一个计数器,然后循环,条件为n>0.如果n<=0跳出循环,先遍历一周,i<5就每次n-a,并且在减之前判断n是否<0即可。

#include<bits/stdc++.h>
#pragma warning(disable:4996)
using namespace std;
const int N = 2022;
int main()
{
	long long a, b, n;
	scanf("%lld %lld %lld", &a, &b, &n);
	int cnt = 0;
	while (n>0)
	{
		for (int i = 0; i < 7; i++)
		{
			if (i < 5)
			{
				if (n <= 0)
					break;
				else
				{
					n = n - a;
					cnt++;
				}
			}
			else
			{
				if (n <= 0)
					break;
				else
				{
					n = n - b;
					cnt++;
				}
			}
		}
		if (n <= 0)
			break;
	}
	cout << cnt;
	return 0;
}

第七题:输出矩阵

a3a149b0320f408690e9f06558e8d180.jpeg

 

分析:这个题比较抽象,方法也很多这里给大家介绍一种很好理解的就是,先遍历二维数组先全部放一个字母然后通过控制层数行数列数再来另一个字母来覆盖我们已经设置的字母。比方说

假设输入的N为3,那么我们需要生成一个大小为5x5的矩阵。下面是每一层中填充字符的过程:

第0层:填充字符'A'
layer = 0,范围是从第0行到第4行,从第0列到第4列。
A A A A A
A A A A A
A A A A A
A A A A A
A A A A A
第1层:填充字符'B'
layer = 1,范围是从第1行到第3行,从第1列到第3列。
A A A A A
A B B B A
A B B B A
A B B B A
A A A A A
第2层:填充字符'C'
layer = 2,范围是第2行,第2列。
A A A A A
A B B B A
A B C B A
A B B B A
A A A A A
这样,通过不断递增layer和字符ch的方式,我们可以在每一层中正确地填充相应的字符。每次内层循环都会覆盖上一层填充的字符,所以最后得到了符合要求的矩阵。

即layer是来控制层级,从最外层到最内层结束,每一层使用两个嵌套循环i和j来遍历,i控制行,从layer开始到n*2-1-layer-1结束,在每个位置(i, j)处,将字符ch赋值给matrix[i][j],完成对该层的字符填充。接着,逐渐增加字符ch的值,以便在下一层填充不同的字符。通过这种方式,代码会按照层级从外到内的顺序,依次填充不同的字符。

#include <iostream>
using namespace std;

int main() {
    int N;
    cin >> N; // 输入N

    char matrix[101][101];

    // 初始化矩阵
    for (int i = 0; i < N * 2 - 1; i++) {
        for (int j = 0; j < N * 2 - 1; j++) {
            matrix[i][j] = 'Z';
        }
    }

    // 填充字符
    char ch = 'A';
    for (int layer = 0; layer < N; layer++) {
        for (int i = layer; i < N * 2 - 1 - layer; i++) {
            for (int j = layer; j < N * 2 - 1 - layer; j++) {
                matrix[i][j] = ch;
            }
        }
        ch++;
    }

    // 输出矩阵
    for (int i = 0; i < N * 2 - 1; i++) {
        for (int j = 0; j < N * 2 - 1; j++) {
            cout << matrix[i][j];
        }
        cout << endl;
    }

    return 0;
}

第八题:平方矩阵

70115b46bc2c4830b8aea1be86491cdc.png

 

思路:乘热打铁,与上题类似,这道题应该如何处理,

#include <iostream>
using namespace std;

int main()
{
    int N;
    while (cin >> N, N)
    {
        int matrix[101][101];

        // 初始化矩阵
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < N; j++) {
                matrix[i][j] = 1;
            }
        }

        // 填充字符
        int ch = 1;
        for (int layer = 0; layer < N; layer++)
        {
            for (int i = layer; i < N - layer; i++)
            {
                for (int j = layer; j < N - layer; j++)
                {
                    matrix[i][j] = ch;
                }
            }
            ch++;
        }

        // 输出矩阵
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < N; j++) {
                cout << matrix[i][j] << " ";
            }
            cout << endl;
        }
    }
    return 0;
}

 

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

哥尼斯堡的“七桥问题”——欧拉回路

哥尼斯堡是位于普累格河上的一座城市&#xff0c;它包含两个岛屿及连接它们的七座桥&#xff0c;如下图所示。 可否走过这样的七座桥&#xff0c;而且每桥只走过一次&#xff1f;瑞士数学家欧拉(Leonhard Euler&#xff0c;1707—1783)最终解决了这个问题&#xff0c;并由此创立…

Pacifist:一款专为技术开发者打造的软件提取工具

对于技术开发者而言&#xff0c;有效且便捷的工具可以显著提高工作效率。Pacifist&#xff0c;作为一款专业的软件提取工具&#xff0c;专为技术开发者而设计&#xff0c;旨在提供简单、安全的软件提取和管理工作。 一、Pacifist的技术特点 Pacifist主要采用AppleScript作为其…

ROS小练习——话题订阅

目录 一、话题与消息获取 二、代码编写 1、C 2、python 三、编译运行 一、话题与消息获取 rostopic list rostopic type /turtle1/pose rosmsg info turtlesim/Pose 二、代码编写 1、C //包含头文件 #include "ros/ros.h" #include "turtlesim/Pose…

js vue 输入正确手机号/邮箱后,激活“发送验证码”按钮

按钮禁止点击状态&#xff1a; 按钮能够点击状态&#xff1a; 我采用的方式是监听手机号/邮箱输入框的输入事件&#xff0c;即实判断用户输入的数据是否满足规则&#xff0c;如果满足手机号/邮箱规则&#xff0c;则激活“获取验证码”按钮。 话不多说&#xff0c;上代码 样式…

Java期末复习题之封装

点击返回标题->23年Java期末复习-CSDN博客 第1题. 定义一个类Person,定义name和age私有属性&#xff0c;定义有参的构造方法对name和age进行初始化。在测试类中创建该类的2个对象&#xff0c;姓名、年龄分别为lili、19和lucy、20&#xff0c;在屏幕打印出2个对象的姓名和年龄…

【Lidar】基于Python的三维点云数据转二维平面+散点图绘制

最近一直在搞点云相关的操作&#xff0c;有时候在处理点云数据时需要查看处理后的数据是否满足需求&#xff0c;所以就想着写一套展示点云的代码。之前已经分享过如何可视化点云了&#xff0c;感兴趣的可以自己去看下&#xff1a;【Lidar】基于Python的Open3D库可视化点云数据。…

微信商城小程序怎么弄

随着移动互联网的快速发展&#xff0c;微信小程序已经成为了许多商家和企业拓展业务的新渠道。其中&#xff0c;微信商城小程序更是受到了广大用户的喜爱。那么微信商城小程序怎么弄呢&#xff1f;下面给大家做个详细讲解。 首先&#xff0c;你需要在微信公众平台注册一个小程…

孩子都能学会的FPGA:第二十三课——用FPGA实现格雷码的编码和解码

&#xff08;原创声明&#xff1a;该文是作者的原创&#xff0c;面向对象是FPGA入门者&#xff0c;后续会有进阶的高级教程。宗旨是让每个想做FPGA的人轻松入门&#xff0c;作者不光让大家知其然&#xff0c;还要让大家知其所以然&#xff01;每个工程作者都搭建了全自动化的仿…

文心一言大模型应用开发入门

本文重点介绍百度智能云平台、文心一言、千帆大模型平台的基本使用与接入流程及其详细步骤。 注册文心一言 请登录文心一言官方网站 https://yiyan.baidu.com/welcome 点击登录&#xff1b;图示如下&#xff1a; 请注册文心一言账号并点击登录&#xff0c;图示如下&#xff1…

深入理解数据在内存中是如何存储的,位移操作符如何使用(能看懂文字就能明白系列)文章超长,慢慢品尝

系列文章目录 C语言笔记专栏 能看懂文字就能明白系列 &#x1f31f; 个人主页&#xff1a;古德猫宁- &#x1f308; 信念如阳光&#xff0c;照亮前行的每一步 文章目录 系列文章目录&#x1f308; *信念如阳光&#xff0c;照亮前行的每一步* 前言引子一、2进制和进制转化为什么…

ORACLE数据库实验总集 实验四 Oracle数据库物理存储结构管理

一、实验目的 &#xff08;1&#xff09;掌握 Oracle数据库数据文件的管理 &#xff08;2&#xff09;掌握 Oracle数据库控制文件的管理 &#xff08;3&#xff09;掌握 Oracle数据库重做日志文件的管理 &#xff08;4&#xff09;掌握 Oracle数据库归档管理&#xff0c; 二、…

Golang 原生Rpc Server实现

Golang 原生Rpc Server实现 引言源码解析服务端数据结构服务注册请求处理 客户端数据结构建立连接请求调用 延伸异步调用定制服务名采用TPC协议建立连接自定义编码格式自定义服务器 参考 引言 本文我们来看看golang原生rpc库的实现 , 首先来看一下golang rpc库的demo案例: 服…

忘记PDF密码了,怎么办?

PDF文件有两种密码&#xff0c;一个打开密码、一个限制编辑密码&#xff0c;因为PDF文件设置了密码&#xff0c;那么打开、编辑PDF文件就会受到限制。忘记了PDF密码该如何解密&#xff1f; PDF和office一样&#xff0c;可以对文件进行加密&#xff0c;但是没有提供恢复密码的功…

动态代理IP和静态代理IP有什么区别,适用场景是什么?

互联网行业的从业者经常会用到一种工具&#xff0c;那就是代理IP工具。动态代理IP和静态代理IP是两种常见的代理IP技术&#xff0c;它们在网络通信中起到了重要的作用&#xff0c;比如大数据行业的从业者会经常需要用到动态代理IP&#xff0c;跨境行业的从业者会经常用到静态代…

MySQL 忘记root密码后重置密码操作

在忘记 MySQL 密码的情况下&#xff0c;可以通过 --skip-grant-tables 关闭服务器的认证&#xff0c;然后重置 root 的密码&#xff0c;具体操作步骤如下。 步骤 1)&#xff1a;关闭正在运行的 MySQL 服务。打开 cmd 进入 MySQL 的 bin 目录。 步骤 2)&#xff1a;输入mysqld -…

Constraining Async Clock Domain Crossing

Constraining Async Clock Domain Crossing 我们在normal STA中只会去check 同步clock之间的timing,但是design中往往会存在很多CDC paths,这些paths需要被正确约束才能保证design function正确,那么怎么去约束这些CDC paths呢? 以下面的design为例,如下图所示 这里clk…

基于Linux的网络防火墙设计方法

摘要 随着Internet的迅速发展&#xff0c;网络越来越成为了人们日常生活不可或缺的一部分&#xff0c;而随之引出的网络安全问题也越来越突出&#xff0c;成为人们不得不关注的问题。 为了在一个不安全的网际环境中构造出一个相对安全的环境&#xff0c;保证子网环境下的计算机…

lorenz相图

观察Lorenz在各个不同维度上的相图。 lorenz_demo(50) function xdot g(t,x) xdot zeros(3,1); sig 10.0; rho 28.0; bet 8.0/3.0; xdot(1) sig*(x(2)-x(1)); xdot(2) rho*x(1)-x(2)-x(1)*x(3); xdot(3) x(1)*x(2)-bet*x(3); endfunction lorenz_demo(time) [t,x] ode…

系统思考与啤酒游戏经营沙盘

结束一家汽车零配件公司《系统思考与啤酒游戏经营沙盘》的内训课&#xff0c;4个小组基本上都有共同的心智模式&#xff0c;这也代表团队有一些集体的盲点。不仅仅对啤酒游戏经营沙盘做了复盘&#xff0c;同时也借用学员画出的系统环路图完成真实案例的研讨以及团队共识&#x…

JVM 运行时内存篇

面试题&#xff1a; 讲一下为什么JVM要分为堆、方法区等&#xff1f;原理是什么&#xff1f;&#xff08;UC、智联&#xff09; JVM的分区了解吗&#xff0c;内存溢出发生在哪个位置 &#xff08;亚信、BOSS&#xff09; 简述各个版本内存区域的变化&#xff1…