强化训练:day8(求最小公倍数、数组中的最⻓连续⼦序列、字⺟收集)

2024-05-14 22:04

本文主要是介绍强化训练:day8(求最小公倍数、数组中的最⻓连续⼦序列、字⺟收集),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 1. 最小公倍数
    • 1.1 题目描述
    • 1.2 解题思路
    • 1.3 代码实现
  • 2. 数组中的最⻓连续⼦序列
    • 2.1 题目描述
    • 2.2 解题思路
    • 2.3 代码实现
  • 3. 字母收集
    • 3.1 题目描述
    • 3.2 解题思路
    • 3.3 代码实现
  • 总结

前言

  1. 最小公倍数
  2. 数组中的最⻓连续⼦序列
  3. 字⺟收集

1. 最小公倍数

1.1 题目描述

在这里插入图片描述

1.2 解题思路

  最小公倍数= A * B / 最大公约数(数学中的短除法),最大公约数:辗转相除法。

1.3 代码实现

#include <iostream>
using namespace std;
using ll = long long;
// 求最大公约数 -> 辗转相除法
ll solve(ll x, ll y) 
{ll ret = 0;while (x % y != 0) {ret = x % y;x = y;y = ret;}return y;
}
int main() 
{ll x, y;cin >> x >> y;ll num = solve(max(x, y), min(x, y));// 短除法ll ret = (x / num) * (y / num) * num;cout << ret;return 0;
}

2. 数组中的最⻓连续⼦序列

2.1 题目描述

在这里插入图片描述

2.2 解题思路

  排序 + 模拟,我们要找的是最长的连续子序列,题目明确说了,只要求值连续,不要求位置连续,因此可以进行排序来帮助我们进行解题。
  排序之后就是从小到大排列,那么问题就简单,我们一次枚举一遍,看看当前值和前一个值的大小是不是相差为1即可,符合条件就将我们的计算count++,不符合就从当前位置开始下一次的判断。不过需要注意的是重复数字的问题,因为对于重复数字还需要进行判断,只需要将我们的标记指针向后移动,计算count不变就可以了。
  我一开始就没有考虑到重复数字的问题,导致没写出来……

2.3 代码实现

class Solution {public:// 注意处理数据重复的问题!!!!!!!using ll = long long;int MLS(vector<int>& arr) {int n = arr.size();sort(arr.begin(), arr.end());int left = 0;int len = 0;while (left < n) {int right = left + 1;int count = 1;while(right < n){if(arr[right] - arr[right - 1] == 1) {right++;count++;}else if(arr[right] - arr[right - 1] == 0){right++;}else{break;}}len = max(len, count);left = right;}return len;}
};

3. 字母收集

3.1 题目描述

在这里插入图片描述

3.2 解题思路

  简单的路径dp问题,我们可以根据题目描述来设置状态,设置一个二维的dp数组,dp[i][j]表示到达 [ i ,j ] 位置最多能获得多少分。而每一个位置的分数与两个位置有关,那就是二维数组中当前位置的上方的数据和左边的数据。
  那么状态转移方程就是:dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]) + add(grid[i - 1][j - 1]); 继续来解释一下,先说add函数,它是用来表示 i,j 位置的字符所得的分数,至于为什么要减1,一会说。我们现在知道了当前位置所能获得的最大分数来自于两个方向,我们肯定需要取大的一个,所以用max,取到大的分数之后,还需要加上当前位置的字母的分数,才是到达当前位置所能获得的最大分数。
  至于为什么减1,则是因为,我们来第一个位置的,它需要它的上面的和左边的数据,但是这是会越界的,所以我们可以将dp表多开一行一列,来避免越界问题。比如,字母的位置是0 0(也就是第一个字符),那么它在dp表中的位置就是1 1,这样就可以避免越界问题了。但是在统计字符分数,就需要进行减一来找到原来的位置了。(不理解的可以画一个二维数组来看一下)

3.3 代码实现

#include <iostream>
using namespace std;
#include <vector>
#include <string>
#include <cstring>
int m, n;
int add(char s) 
{if (s == 'l') return 4;else if (s == 'o') return 3;else if (s == 'v') return 2;else if (s == 'e') return 1;else return 0;
}
int main() 
{cin >> m >> n;vector<vector<char>> grid(m, vector<char>(n));vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));   for (int i = 0; i < m; i++)for(int j = 0; j < n; j++)cin >> grid[i][j];for (int i = 1; i <= m; i++)for (int j = 1; j <= n; j++) {dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]) + add(grid[i - 1][j - 1]);}cout << dp[m][n];    return 0;
}

总结

  处理第一题,后面两个题还是有很多细节的地方要处理的,而且对于这种题,看题解可能一看就会了,一写就废了,原因就是有很多的边界问题以及条件判断并没有搞清楚或者说没有意识到,所以总是出错。而为了能避免这种情况,只能是多写,不能光看不写。
  那么第八天的内容就到此结束了,如果大家发现有什么错误的地方,可以私信或者评论区指出喔。我会继续坚持训练的,希望能与大家共同进步!!!那么本期就到此结束,让我们下期再见!!觉得不错可以点个赞以示鼓励!

这篇关于强化训练:day8(求最小公倍数、数组中的最⻓连续⼦序列、字⺟收集)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/989971

相关文章

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

poj2406(连续重复子串)

题意:判断串s是不是str^n,求str的最大长度。 解题思路:kmp可解,后缀数组的倍增算法超时。next[i]表示在第i位匹配失败后,自动跳转到next[i],所以1到next[n]这个串 等于 n-next[n]+1到n这个串。 代码如下; #include<iostream>#include<algorithm>#include<stdio.h>#include<math.

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

poj 1287 Networking(prim or kruscal最小生成树)

题意给你点与点间距离,求最小生成树。 注意点是,两点之间可能有不同的路,输入的时候选择最小的,和之前有道最短路WA的题目类似。 prim代码: #include<stdio.h>const int MaxN = 51;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int P;int prim(){bool vis[MaxN];

poj 2349 Arctic Network uva 10369(prim or kruscal最小生成树)

题目很麻烦,因为不熟悉最小生成树的算法调试了好久。 感觉网上的题目解释都没说得很清楚,不适合新手。自己写一个。 题意:给你点的坐标,然后两点间可以有两种方式来通信:第一种是卫星通信,第二种是无线电通信。 卫星通信:任何两个有卫星频道的点间都可以直接建立连接,与点间的距离无关; 无线电通信:两个点之间的距离不能超过D,无线电收发器的功率越大,D越大,越昂贵。 计算无线电收发器D

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

poj 1734 (floyd求最小环并打印路径)

题意: 求图中的一个最小环,并打印路径。 解析: ans 保存最小环长度。 一直wa,最后终于找到原因,inf开太大爆掉了。。。 虽然0x3f3f3f3f用memset好用,但是还是有局限性。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#incl

hdu 1102 uva 10397(最小生成树prim)

hdu 1102: 题意: 给一个邻接矩阵,给一些村庄间已经修的路,问最小生成树。 解析: 把已经修的路的权值改为0,套个prim()。 注意prim 最外层循坏为n-1。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstri

uva 10131 最长子序列

题意: 给大象的体重和智商,求体重按从大到小,智商从高到低的最长子序列,并输出路径。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vect