ACM日常训练日记——7.24
- Atcoder训练
- Template Matching
暴力枚举,问题给出的nxn字符串数组有没有包含下面的mxm数组
- Template Matching
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
int n,m;
int main(){
int n,m;
cin>>n>>m;
vector<string>v(n+1);
vector<string>s(n+1);
for(int i=1;i<=n;i++){
cin>>v[i];
}
for(int i=1;i<=m;i++){
cin>>s[i];
}
int ans=0;
for(int i=1;i<=n;i++){
if(v[i].find(s[1])!=string::npos){
int k=i;
int j=1;
while(k<=m){
if(v[k].find(s[j])!=string::npos){
ans++;
k++;
j++;
}else{
ans=0;
k=0;
j=0;
break;
}
}
if(ans==m){
cout<<"Yes";
return 0;
}
}
}
cout<<"No";
return 0;
}
- Double Factorial
思维,看出来结果只与乘积里面为十的倍数有关,当n为奇数,里面没有偶数,所以答案恒为零,在偶数中我们只需要找到10的因子,只有2和5
2的因子数量肯定远远多于5的,所以我们只需要计算五的因子数有多少个即可,需要注意的是每次计算要除2,才满足是10的因子
不断将n除以 5 并累加结果,我们可以计算出因子 5 的个数
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll n,m;
unordered_map<ll,ll>mp;
int ans=0;
int main(){
ll n;
cin>>n;
if(n%2!=0){
cout<<0;
}else{
n/=2;
ll ans=0;
while(n>0){
n/=5;
ans+=n;
}
cout<<ans;
}
return 0;
}
- ID
我TLE了,我用set去存放大小的时候,记录下标位置特别麻烦,只能两个for循环去找,特别慢
优化思路想着把下标存进后面的值就成功,后面看了别人的代码又开一个map去存放下标就可以了
我的代码
优化
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
ll v[100010], d[100010];
map<pair<ll, ll>, ll> ans;
int main() {
ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
ll n, m;
cin >> n >> m;
map<ll, set<ll>> mp;
map<ll, map<ll, ll>> position; // 存储每个元素在set中的位置
for (int i = 1; i <= m; i++) {
ll x, y;
cin >> x >> y;
v[i] = x, d[i] = y;
mp[x].insert(y);
}
// 预先计算每个元素在set中的位置
for (auto &p : mp) {
ll x = p.first;
int pos = 1;
for (auto it = p.second.begin(); it != p.second.end(); ++it) {
position[x][*it] = pos++;
}
}
// 直接获取每个元素在set中的位置
for (int i = 1; i <= m; i++) {
d[i] = position[v[i]][d[i]];
}
for (int i = 1; i <= m; i++) {
printf("%06lld%06lld\n", v[i], d[i]);
}
}
-
牛客萌新联赛
- 国际旅行Ⅰ
送分我还wa了三,原因是第二次输入是m个我写成了n,死活没看到,题目已经全部联通,不需要考虑,排序就行
- 国际旅行Ⅰ
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll n,m,q;
unordered_map<ll,ll>mp;
ll v[100010];
int ans=0;
ll d,e;
int main(){
cin>>n>>m>>q;
for(int i=0;i<n;i++)cin>>v[i];
sort(v,v+n);
for(int i=1;i<=m;i++)cin>>d>>e;
while(q--){
ll f;
cin>>f;
cout<<v[f-1]<<'\n';
}
return 0;
}
2.水灵灵的小学弟
名字都是一样的DHY,注意有t组输出,我wa了一次
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll n,m;
unordered_map<ll,ll>mp;
int ans=0;
int main(){
cin>>n;
while(n--)
cout<<"DHY"<<'\n';
return 0;
}
3.重生之zbk要拿回属于他的一切
之前写过差不多一模一样,函数可以留着用,就记录连续子串个数,我记得最简单可以四行,我忘了...
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll n,m;
unordered_map<ll,ll>mp;
int ans=0;
ll countSubstringOccurrences(const string& str, const std::string& substring) {
ll count = 0;
size_t pos = str.find(substring);
while (pos != string::npos) {
count++;
pos = str.find(substring, pos + 1);
}
return count;
}
int main(){
ll n;
cin>>n;
string s;
cin>>s;
cout<<countSubstringOccurrences(s,"chuan");
return 0;
}
4.这是签到
计算行列式的值,线性代数,记得公式用递推就行,n,m的值不大于五,其实枚举也行,我写递推会快一点
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll n,m,q;
unordered_map<ll,ll>mp;
ll v[10][10];
ll d,e;
// 计算行列式的值
ll determinant(vector<vector<ll>>& matrix, ll n) {
if (n == 1) {
return matrix[0][0];
} else if (n == 2) {
return matrix[0][0] * matrix[1][1] - matrix[0][1] * matrix[1][0];
}
ll det = 0;
for (int p = 0; p < n; ++p) {
vector<vector<ll>> subMatrix(n - 1, vector<ll>(n - 1));
for (int i = 1; i < n; ++i) {
ll colIndex = 0;
for (int j = 0; j < n; ++j) {
if (j == p) continue;
subMatrix[i - 1][colIndex] = matrix[i][j];
colIndex++;
}
}
det += (p % 2 == 0 ? 1 : -1) * matrix[0][p] * determinant(subMatrix, n - 1);
}
return det;
}
int main() {
int n, m;
cin >> n >> m;
vector<vector<ll>> matrix(n, vector<ll>(m));
for (int i = 0; i < n; ++i) {
for (int j = 0; j < m; ++j) {
cin >> matrix[i][j];
}
}
ll maxSize = max(n, m);
ll minDet = 2e18;
for (int size = 1; size <= maxSize; ++size) {
vector<vector<ll>> subMatrix(size, vector<ll>(size, 0));
for (int i = 0; i < size; ++i) {
for (int j = 0; j < size; ++j) {
if (i < n && j < m) {
subMatrix[i][j] = matrix[i][j];
} else {
subMatrix[i][j] = 0;
}
}
}
if (size <= n && size <= m) {
minDet = min(minDet, determinant(subMatrix, size));
}
}
cout << minDet << endl;
return 0;
}
5.狼狼的备忘录
不难,只是题目给的比较坑,把第一个要求弄明白就行,然后可以用map结合string和set来保持字典序即可
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll n,m,q;
//unordered_map<ll,ll>mp;
ll v[10][10];
ll d,e;
//第一个条件....
bool pd(const string a, const string b) {
if (a.size() > b.size()) return false;
return b.substr(b.size() - a.size()) == a;
}
int main() {
int n;
cin >> n;
map<string, set<string>> mp;
for (int i = 0; i < n; ++i) {
string name;
int op;
cin >> name >> op;
for (int j = 0; j < op; ++j) {
string info;
cin >> info;
mp[name].insert(info);
}
}
for (auto& pair : mp) {
set<string>& infos = pair.second;
set<string> toRemove;
for (const string& info : infos) {
for (const string& other : infos) {
if (info != other && pd(info, other)) {
toRemove.insert(info);
}
}
}
for (const string& info : toRemove) {
infos.erase(info);
}
}
// 输出结果
cout << mp.size() << endl;
for (const auto& pair : mp) {
cout << pair.first << " " << pair.second.size();
for (const string& info : pair.second) {
cout << " " << info;
}
cout << '\n';
}
return 0;
}
6.A*BBBB
这道题一般的高精度乘法超时,直接用更快的FFT优化算法
数据开大一点,就是板子了
查看代码
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <map>
#include <queue>
#include <set>
#include <vector>
using namespace std;
#define L(x) (1 << (x))
const double PI = acos(-1.0);
const int Maxn = 13300015;
double ax[Maxn], ay[Maxn], bx[Maxn], by[Maxn];
char sa[Maxn/2],sb[Maxn/2];
int sum[Maxn];
int x1[Maxn],x2[Maxn];
int revv(int x, int bits)
{
int ret = 0;
for (int i = 0; i < bits; i++)
{
ret <<= 1;
ret |= x & 1;
x >>= 1;
}
return ret;
}
void fft(double * a, double * b, int n, bool rev)
{
int bits = 0;
while (1 << bits < n) ++bits;
for (int i = 0; i < n; i++)
{
int j = revv(i, bits);
if (i < j)
swap(a[i], a[j]), swap(b[i], b[j]);
}
for (int len = 2; len <= n; len <<= 1)
{
int half = len >> 1;
double wmx = cos(2 * PI / len), wmy = sin(2 * PI / len);
if (rev) wmy = -wmy;
for (int i = 0; i < n; i += len)
{
double wx = 1, wy = 0;
for (int j = 0; j < half; j++)
{
double cx = a[i + j], cy = b[i + j];
double dx = a[i + j + half], dy = b[i + j + half];
double ex = dx * wx - dy * wy, ey = dx * wy + dy * wx;
a[i + j] = cx + ex, b[i + j] = cy + ey;
a[i + j + half] = cx - ex, b[i + j + half] = cy - ey;
double wnx = wx * wmx - wy * wmy, wny = wx * wmy + wy * wmx;
wx = wnx, wy = wny;
}
}
}
if (rev)
{
for (int i = 0; i < n; i++)
a[i] /= n, b[i] /= n;
}
}
int solve(int a[],int na,int b[],int nb,int ans[])
{
int len = max(na, nb), ln;
for(ln=0; L(ln)<len; ++ln);
len=L(++ln);
for (int i = 0; i < len ; ++i)
{
if (i >= na) ax[i] = 0, ay[i] =0;
else ax[i] = a[i], ay[i] = 0;
}
fft(ax, ay, len, 0);
for (int i = 0; i < len; ++i)
{
if (i >= nb) bx[i] = 0, by[i] = 0;
else bx[i] = b[i], by[i] = 0;
}
fft(bx, by, len, 0);
for (int i = 0; i < len; ++i)
{
double cx = ax[i] * bx[i] - ay[i] * by[i];
double cy = ax[i] * by[i] + ay[i] * bx[i];
ax[i] = cx, ay[i] = cy;
}
fft(ax, ay, len, 1);
for (int i = 0; i < len; ++i)
ans[i] = (int)(ax[i] + 0.5);
return len;
}
string mul(string sa,string sb)
{
int l1,l2,l;
int i;
string ans;
memset(sum, 0, sizeof(sum));
l1 = sa.size();
l2 = sb.size();
for(i = 0; i < l1; i++)
x1[i] = sa[l1 - i - 1]-'0';
for(i = 0; i < l2; i++)
x2[i] = sb[l2-i-1]-'0';
l = solve(x1, l1, x2, l2, sum);
for(i = 0; i<l || sum[i] >= 10; i++) // 进位
{
sum[i + 1] += sum[i] / 10;
sum[i] %= 10;
}
l = i;
while(sum[l] <= 0 && l>0) l--; // 检索最高位
for(i = l; i >= 0; i--) ans+=sum[i] + '0'; // 倒序输出
return ans;
}
int main()
{
int t;
cin>>t;
cin.sync_with_stdio(false);
while(t--){
string a,b;
cin>>a>>b; cout<<mul(a,b)<<endl;
}
return 0;
}
- 动态规划专项训练
第二种类型没有明确的起点和终点,是连续的必须选择一个丢掉,每一次的状态要么从当前新开始a[i],要么必须选择f[i-1]+a[i].
1.最大子串和
这是典型的第二种类型,不知道起点终点,每一次都有可能是当前的值,否则就是前面的选上加当前,取max
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll f[1000005];
ll F[1000005];
ll v[1000005];
map<pair<ll,ll>,ll>ans;
int main(){
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
ll n,m;
cin>>n;
ll ma=0;
for(int i=0;i<n;i++)cin>>v[i];
for(int i=0;i<n;i++){
f[i]=max(f[i-1]+v[i],v[i]);
ma=max(f[i],ma);
}
cout<<ma;
}
2.最长不下降子序列
一样的解法,但是这里是长度,每个自己的长度为1,每一次更新都是跟前面比它小的判断这道题提交不了了,题没了
#include <bits/stdc++.h>
using namespace std;
using ll =long long;
ll v[100010],d[100010];
ll f[100010];
map<pair<ll,ll>,ll>ans;
int main(){
ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int n;
cin>>n;
ll ans=0;
for(int i=0;i<n;i++)cin>>v[i];
for(int i=0;i<n;i++)f[i]=1;
for(int i=0;i<n;i++){
for(int j=0;j<i;j++){
if(v[i]>v[j]){
f[i]=max(f[j]+1,f[i]);
ans=max(f[i],ans);
}
}
}
cout<<ans;
}