week17 代码源oj每日一题div2

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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值