文章目录
1、整除光棍
思路:
这题就是模拟竖式除法,一步一步往被除数后加一个数字1。
#include<iostream>
#include<vector>
using namespace std;
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int x;
int s=1;
int len=1;
vector<int>num;
cin>>x;
while(s<x)
{
len++;
s=s*10+1;
}
while(s%x!=0)
{
num.push_back(s/x);
s=s%x*10+1;
len++;
}
num.push_back(s/x);
for(int i=0;i<num.size();i++)
{
cout<<num[i];
}
cout<<' '<<len;
return 0;
}
2、碰撞2
思路:
只要把能够发生碰撞的条件判断清楚,这题就非常简单了(其实本就很简单)。
#include<iostream>
#include<algorithm>
#include<vector>
#define ll long long
using namespace std;
struct h
{
ll x,y;
char d;
};
bool temp(h a,h b)
{
return a.y<b.y||a.y==b.y&&a.x<b.x;
}
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
ll n;
ll flag=0;
vector<h>v;
cin>>n;
for(int i=0;i<n;i++)
{
ll a,b;
cin>>a>>b;
v.push_back((h){a,b});
}
for(int i=0;i<n;i++)
{
cin>>v[i].d;
}
sort(v.begin(),v.end(),temp);
for(int i=0;i<n-1;i++)
{
if(v[i].y==v[i+1].y)
{
if(v[i].x<v[i+1].x&&v[i].d=='R'&&v[i+1].d=='L'||v[i+1].x<v[i].x&&v[i+1].d=='R'&&v[i].d=='L')
{
flag=1;
break;
}
}
}
if(flag==1)cout<<"Yes";
else cout<<"No";
return 0;
}
3、优美!最长上升子序列
思路:
1、方法一(这题数据范围较小,能过):时间复杂度O(n2)的dp,基操,应该都会。
2、方法二(数据范围较大时):贪心+二分
以下博主给出方法一的代码:
#include<iostream>
#include<vector>
using namespace std;
const int N=1e6+1;
int dp[N]; //dp[i]表示以v[i]结尾的最长上升子序列长度
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t;
cin>>t;
while(t--)
{
int n,ans=1;
vector<int>v(N);
cin>>n;
for(int i=1;i<=n;i++)
{
cin>>v[i];
dp[i]=1;
}
for(int i=1;i<=n;i++)
{
for(int j=2*i;j<=n;j+=i)
{
if(v[i]<v[j])
{
dp[j]=max(dp[j],dp[i]+1);
ans=max(ans,dp[j]);
}
}
}
cout<<ans<<'\n';
}
return 0;
}
4、巨大的牛棚
思路:
1、方法一:
DP,状态转移方程:
dp[i][j]=min(dp[i+1][j],min(dp[i][j+1],dp[i+1][j+1]))+1;
其中dp[i][j]表示以(i,j)为左上角的最大正方形的边长
2、方法二:
矩阵前缀和
方法一:
#include<iostream>
#include<cmath>
using namespace std;
bool a[1001][1001]={0};
int dp[1001][1001]={0}; //dp[i][j]表示以(i,j)为左上角的最大正方形的边长
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int n,t;
int ans=1;
cin>>n>>t;
for(int i=1;i<=t;i++)
{
int x,y;
cin>>x>>y;
a[x][y]=1;
}
for(int j=1;j<=n;j++)
if(a[n][j]==0)dp[n][j]=1;
for(int i=1;i<=n;i++)
if(a[i][n]==0)dp[i][n]=1;
for(int i=n-1;i>=1;i--) //第i行第j列
{
for(int j=n-1;j>=1;j--)
{
if(a[i][j]==0)
{
dp[i][j]=min(dp[i+1][j],min(dp[i][j+1],dp[i+1][j+1]))+1;
ans=max(ans,dp[i][j]);
}
}
}
cout<<ans;
return 0;
}
方法二:
#include<iostream>
using namespace std;
bool a[1001][1001]={0};
int sum[1001][1001]={0};
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int n,t;
cin>>n>>t;
for(int i=1;i<=t;i++)
{
int x,y;
cin>>x>>y;
a[x][y]=1;
}
sum[1][1]=a[1][1];
for(int i=1;i<=n;i++)
{
for(int j=1;j<=n;j++)
{
if(i==1&&j==1)continue;
else if(i==1&&j>1)
{
sum[i][j]=sum[i][j-1]+a[i][j];
}
else if(j==1&&i>1)
{
sum[i][j]=sum[i-1][j]+a[i][j];
}
else
{
sum[i][j]=sum[i-1][j]+sum[i][j-1]-sum[i-1][j-1];
}
}
}
return 0;
}
5、高利贷
思路:
这题就模拟,不断让答案往第6位小数精确。
#include<iostream>
#include<cmath>
using namespace std;
int main()
{
// std::ios::sync_with_stdio(false);
// cin.tie(0);
// cout.tie(0);
int n,m,k;
double p=0.1;
int len=1;
cin>>n>>m>>k;
while(len<=7)
{
double sum=0,s=1.0*n/m;
for(int i=1;i<=k;i++)
{
sum+=1.0/pow(1+p,i);
}
if(sum>s) //p太小了
{
p+=pow(10,-len);
}
else if(sum<s) //p太大了
{
p=p-pow(10,-len)+pow(10,-len-1);
len++;
}
else
{
break;
}
}
printf("%.6lf",p);
}
6、背包
思路:
这题用贪心算法(本人比较菜,看了题解才想明白,不过真的很巧妙!)
#include<iostream>
#include<algorithm>
#include<cmath>
#define ll long long
using namespace std;
const int N=2e5+1;
int a[N];
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t;
cin>>t;
while(t--)
{
int n;
int flag=0;
ll w;
ll sum=0;
cin>>n>>w;
ll k=ceil(w*1.0/2);
for(int i=1;i<=n;i++)cin>>a[i];
sort(a+1,a+1+n);
for(int i=1;i<=n;i++)
{
if(a[i]<=w-sum)sum+=a[i];
if(sum>=k||a[i]>=k&&a[i]<=w)
{
flag=1;
break;
}
}
if(flag==1)cout<<"YES";
else cout<<"NO";
cout<<'\n';
}
return 0;
}
7、三回文序列
思路:
当我们暂时想不到更好的方法时,就直接暴力枚举(这题正好也能过),但是需要进行优化,即记录某数字前缀和发生变化的下标。
#include<iostream>
#include<vector>
using namespace std;
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int t,n;
cin>>t;
while(t--)
{
vector<int>a(200001);
vector<vector<int> >sum(27),last(27);
//sum[i][j]表示序列的前j个数中数字i的个数(前缀和)
//last[i][j]表示序列的第j个 数字i 的下标
int res=1;
vector<int>ct(27);
bool flag=0;
cin>>n;
for(int i=1;i<=n;i++)
{
cin>>a[i];
ct[a[i]]++;
}
if(n==1)
{
cout<<1<<'\n';
continue;
}
for(int i=1;i<=26;i++) //求前缀和
{
if(ct[i]==0)continue;
sum[i].push_back(0);
last[i].push_back(0);
for(int j=1;j<=n;j++)
{
int k=sum[i][j-1];
if(a[j]==i)
{
k+=1;
last[i].push_back(j);
}
sum[i].push_back(k);
}
}
for(int i=1;i<=26;i++) //枚举三回文序列的开头或结尾数字
{
if(ct[i]==0)continue;
for(int len=0;len<=sum[i][n]/2;len++) //枚举开头数字或结尾数字的重复个数
{
if(len==0&&flag==1)continue;
if(len==0)flag=1;
int L=last[i][len]+1,R=n;
if(len!=0)R=last[i][ct[i]-len+1]-1;
if(L>R)continue;
int cnt=0;
for(int k=1;k<=26;k++) //求从下标为L到下标为R的出现个数最多的数的个数
{
if(ct[k]==0)continue;
cnt=max(sum[k][R]-sum[k][L-1],cnt);
}
res=max(res,cnt+2*len);
}
}
cout<<res<<'\n';
}
return 0;
}
8、简单的异或问题
思路:
透过现象看本质。
想在0~2m-1里通过xor运算获得一个数a,就只用(2m-1)和(2m-1-a)进行xor运算即可,但这里要求的是最多能选多少数,我们可以把所有的数(除了目标a)都选上,真正有用的只有(2m-1)和(2m-1-a),至于其他的数经过运算后都会变成0.所以k能取到的最大值就是2m-1。注意特殊情况。
#include<iostream>
#define ll long long
using namespace std;
ll qmi(ll x,ll y)
{
ll res=1;
while(y)
{
if(y&1)res*=x;
x*=x;
y>>=1;
}
return res;
}
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
ll ans;
ll n,m;
cin>>n>>m;
if(m==1)
{
if(n==0)ans=1;
else if(n==1)ans=2;
}
else if(n==0)ans=qmi(2,m);
else ans=qmi(2,m)-1;
cout<<ans;
return 0;
}
9、子串的循环挪动
思路:
这题比较简单,模拟即可。
#include<iostream>
#include<cstring>
using namespace std;
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
char s[10001];
int m;
cin>>s>>m;
while(m--)
{
int l,r,k;
int len;
bool flag=0;
char a[10001];
cin>>l>>r>>k;
l-=1,r-=1; //下标从0开始
len=r-l+1;
k%=len;
for(int i=1;i<=k;i++)
{
for(int j=l;j<=r;j++)
{
int u=j+k;
if(u>r)u-=len;
flag=1;
a[u]=s[j];
}
}
if(flag==1)
for(int j=l;j<=r;j++) //更新字符串
{
s[j]=a[j];
}
}
for(int i=0;i<strlen(s);i++)
{
cout<<s[i];
}
return 0;
}
10、弗拉德和糖果 II
思路:
这题想清楚了就很简单。不多说,看代码自然就懂了。
#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;
int main()
{
std::ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
int n;
long long m=0,sum=0;
vector<int>a;
cin>>n;
for(int i=0;i<n;i++)
{
int x;
cin>>x;
a.push_back(x);
}
sort(a.begin(),a.end());
for(int i=0;i<n-1;i++)
{
sum+=a[i];
}
if(sum<a[n-1]-1)
{
cout<<"NO";
}
else cout<<"YES";
return 0;
}