2021-11-7——2021-11-14ACM笔记

删数问题
键盘输入一个高精度的正整数 NN(不超过 250250 位),去掉其中任意 kk 个数字后剩下的数字按原左右次序将组成一个新的非负整数。编程对给定的 NN 和 kk,寻找一种方案使得剩下的数字组成的新数最小。

 #include <bits/stdc++.h>
using namespace std;
int main()
{
char a[255];
int k;
int len;
cin>>a;
cin>>k;
int j;
int i;
len=strlen(a);
while(k)
{
for(i=0;i<len-1;i++)
{if(a[i]>a[i+1])
{for(j=i;j<len-1;j++)
a[j]=a[j+1];
break;}



}
len--;
k--;
}
i=0;
while(i<=len-1&&a[i]=='0')i++;
if(i==len)
cout<<"0";
else
for(j=i;j<len;j++)
	cout<<a[j];

return 0;
}

之前做过的题,又作了一遍,容易找思路但还是有很多细节:要用i和len去比较来判断前置有多少个0,然后从之后开始输出。len和k在if判断外进行自减,来防止12345 2这种前一项比后一项小的情况。这样len减了相当于最后一个数去掉了。

P7911 [CSP-J 2021] 网络连接

#include <bits/stdc++.h>
const int N=(int)1e3+5;
std::map<std::string,int>map;
inline int in();
inline bool check(std::string);
int main(int argc,char**argv){
#ifndef ONLINE_JUDGE
	freopen("7911.in","r",stdin);
	freopen("7911.out","w",stdout);
#endif
	std::ios::sync_with_stdio(false);
	register int n=in();
	for(register int i=1;i<=n;++i){
		std::string op,ad;
		std::cin>>op>>ad;
		if(!check(ad))std::cout<<"ERR\n";
		else if(op[0]=='S'){
			if(map.count(ad))
				std::cout<<"FAIL\n";
			else{
				map[ad]=i;
				std::cout<<"OK\n";
			}
		}
		else{
			if(map.count(ad)){
				std::cout<<map[ad]<<std::endl;
			}
			else{
				std::cout<<"FAIL\n";
			}
		}
	}
}
inline bool check(std::string str){
	register int cnt1=0,cnt2=0;
	register int len=str.size();
	for(register int i=0;i<len;++i){
		if(str[i]=='.')
			++cnt1;
		if(str[i]==':')
			++cnt2;
		if(cnt2==1&&cnt1<3)return 0;
		
		if((str[i]<'0'||str[i]>'9')&&(str[i]!='.')&&(str[i]!=':'))
			return 0;
		
	}
	if(cnt1!=3||cnt2!=1)return 0;
	int num[5]={-1,-1,-1,-1,-1};
	register int now=0;
	str[len]='.';
	for(register int i=0;i<len;++i){
		if(num[now]==-1&&(str[i]<'0'||str[i]>'9'))
			
			return 0;
		else if(num[now]==-1&&(str[i]=='0'&&str[i+1]!='.'&&str[i+1]!=':'))
			return 0; // 前导 0
		else if(str[i]<'0'||str[i]>'9')
			
			++now;
		else if(num[now]==-1)
			num[now]=(str[i]&15);
		else
			num[now]=(num[now]<<1)+(num[now]<<3)+(str[i]&15);
			
		if(num[now]>65535)return 0;
	}
	register bool tp1=(0<=num[0]&&num[0]<=255);
	register bool tp2=(0<=num[1]&&num[1]<=255);
	register bool tp3=(0<=num[2]&&num[2]<=255);
	register bool tp4=(0<=num[3]&&num[3]<=255);
	register bool tp5=(0<=num[4]&&num[4]<=65535);
	return tp1&&tp2&&tp3&&tp4&&tp5;
}
inline int in(){
	register int x;
	std::cin>>x;
	return x;
}

首先观察可知,原题可以分为两个部分:
检查一个地址是否合法;
加入一个地址。
后者可以用 map 维护,是基础操作,所以关键点就落在了前者上。
首先是一波最基础的判断:
. 出现 3 次,: 出现 1 次。
它们出现的顺序必须是 .、.、.、:。
没有别的什么奇怪字符(除 0,1,2,3,4,5,6,7,8,9,:,. 之外)。
读入的时候注意判断一下前导零就可以了。
注意到本题数据范围,有可能某些写法需要开 long long,但是统计答案的时候动态检验,不合理即退出,这样就不是很困难,也不需要开 long long。

P7910 [CSP-J 2021] 插入排序

#include<bits/stdc++.h>
using namespace std;
int n,q,a[8005],b[8005],num,x,v,ans;
int main()
{
	scanf("%d%d",&n,&q);
	for(int i=1;i<=n;i++)
	{
		scanf("%d",&a[i]);
	}
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=i;j++)
		{
			if(i==j||a[j]<a[i]||a[j]==a[i])b[i]++;
			else b[j]++;
		}//这里的b[i]就是第i个数在数组a[i]中的排位 
	}
	for(int i=0;i<q;i++)
	{
		scanf("%d",&num);
		if(num==1)
		{
			scanf("%d%d",&x,&v);
			for(int i=1;i<=n;i++)
			{
				if(i==x)continue;
				if((a[i]<a[x]||a[i]==a[x]&&i<x)&&(a[i]>v||a[i]==v&&i>x))
				{
					b[i]++;b[x]--;
				}
				else if((a[i]>a[x]||a[i]==a[x]&&i>x)&&(a[i]<v||a[i]==v&&i<x))
				{
					b[i]--;b[x]++;
				}
			}
			a[x]=v;
		}
		if(num==2)
		{
			scanf("%d",&x);
			printf("%d\n",b[x]);
		}
	}
	return 0;
}

这题做了一周,最后还是按一个比较通俗易懂的题解想出来的。
注意到一个数组 aa 进行“示范代码”之后,数 ai 在 aj 之前的充要条件是以下两者中成立一个:
1.ai<aj
2.ai=aj且i<j
考虑维护一个数组 b 储存各个数在数组 a中的相对排名,那么开始输入 a时进行统计(用以上两条判断)之后:
每进行一次操作 2 ,输出bx
每进行一次操作 1 ,即将 ax修改成 v,此时用O(n)遍历数组中的各个元素,如果某一个元素 i 满足在数组插入排序后原本在 x 之前,现在在 x 之后(仍然用两条判定),因为 i 与数组中其他数的相对位置没有变化,使 bi 增加一, bx 减少一即可。另外一种情况同理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值