19-11-12-Aftern-℘

我饿死了,于是写写博客安慰一下即将退役的自己。

ZJ:

T1。

三种颜色,想到一道神奇的‘天空龙’。

于是觉得此题可做。

那好了。

于是切掉,还拿了一个暴力对拍。疯狂A。

啊dfs慢的要死了

T2一眼拓扑裸题。

但是读入很××。

于是拿出 getline(cin,st); 但是死了。

因为输入太……

后来一番倒饬过了样例。

T3丢了一个大暴力,然后发现根本不是图论题,是序列题=。=

然后就不会了。

最后T2就死了

为什么考输入啊????

30
Miemeng 100
00:00:03
0
00:00:04
20
00:00:05
120
00:00:05

TJ解:

T1:

这题我是贪心思路,但是可以证明是正确的。(但是我不想,所以大家感性理解吧)

首先将三种颜色一个一个一个卡在一起。

非常棒

 

然后一定会剩两种颜色(或是一种)

那么如果有两种,就用多的少的$2:1$加入锅中组合

好好看

然后就考虑剩下的了。

如果剩下的是比较多的那种,那么就可以每三个多的和一个三元的交换来使答案$++$

如果是比较少的那种,就可以将每三个和前面匹配的两种交换。

有个细节,如果多的那个剩一个,可以和下面的两个少的相结合也挺好的

//decorate

#include <iostream>
#include <cstring>
#include <cstdio>

using namespace std;

int r,b,y;
void getsort(int &x,int &y,int &z){
	if(x<y)swap(x,y);
	if(y<z)swap(y,z);
	if(x<y)swap(x,y);
}
int main(){
#ifndef LOCAL
	freopen("decorate.in" ,"r",stdin);
	freopen("decorate.out","w",stdout);
#endif
//	while(cin>>r>>y>>b,getsort(r,y,b),cout<<r<<" "<<y<<" "<<b<<endl);
	int T;
	cin>>T;
	while(T--){
		scanf("%d%d%d",&r,&b,&y);
		getsort(r,b,y);
//		cout<<"Start:"<<r<<" "<<b<<" "<<y<<endl;

		int n_3=y;
		r-=n_3,b-=n_3,y-=n_3;
//		cout<<"3th_:"<<r<<" "<<b<<" "<<y<<endl;

		int n_2=min(r/2,b);
		r-=n_2*2,b-=n_2;

//		cout<<"2th_:"<<r<<" "<<b<<" "<<y<<endl;
		int swn=0;
		if(b==0){
			swn=min(n_3,r/3);
		}
		else if(r==1){
			if(b>=2){
				b-=2,r--;
				swn++;
				if(b>0){
					swn+=min(n_2+n_3,b/3);
				}
			}
		}
		else{//b!=0 && r==0.
			swn=min(n_2+n_3,b/3);
		}
//		cout<<n_3<<"+"<<n_2<<"+"<<swn<<endl;
		cout<<n_3+n_2+swn<<endl;
	}
}

T2

只考输入的题。

//dependency

#include <iostream>
#include <cstring>
#include <cstdio>
#include <string>
#define N 333333
#define M 3333333

using namespace std;
const int Mod=5131111,Upbit=2411;//5477777
int id[Mod];
string st;
char dat[20];

struct SR{
	int f,t,next;
}rs[M];
int pcnt,ecnt,fl[N],pn;

struct Myqueue{
	int A[M*5];
	int f,b;
	Myqueue(){f=b=0;}
	void clear(){f=b=0;}
	bool empty(){return f==b;}
	void push(const int k){A[b++]=k;}
	void pop(){f++;}
	int front(){return A[f];}
}q;
int deg[N];
string thes;
int read(){
	getline(cin,thes);
	int d=0,n=0;
	while((thes[d]<'0' || thes[d]>'9') && d<(int)thes.length())d++;
	while(thes[d]>='0' && thes[d]<='9' && d<(int)thes.length()){
		n=n*10+thes[d]-'0';
		d++;
	}
	return n;
}
void add(int f,int t){
//	cout<<f<<"->"<<t<<endl;
	rs[ecnt].f=f;
	rs[ecnt].t=t;
	rs[ecnt].next=fl[f];
	fl[f]=ecnt++;
}
bool topsort(){
	q.clear();
	for(int i=1;i<=pcnt;i++){
		if(deg[i]==0){
			q.push(i);
		}
	}
	while(!q.empty()){
		int f=q.front();q.pop();
		for(int i=fl[f];i!=-1;i=rs[i].next){
			int t=rs[i].t;
			deg[t]--;
			if(deg[t]==0){
				q.push(t);
			}
		}
	}
	for(int i=1;i<=pcnt;i++)
		if(deg[i]!=0)return 1;
	return 0;
}
void prerun(){
	pcnt=0,ecnt=0;
	memset(id , 0,sizeof id );
	memset(deg, 0,sizeof deg);
	memset(fl ,-1,sizeof fl );
}
int main(){
#ifndef LOCAL
	freopen("dependency.in" ,"r",stdin);
	freopen("dependency.out","w",stdout);
#endif
//	freopen("2.in","r",stdin);
	ios_base::sync_with_stdio(false);
	int T;
	T=read();
	while(T--){
		prerun();
		pn=read();
		for(int i=1;i<=pn;i++){
			getline(cin,st);
			st+=" ";
			int f=0,len=st.length();
			long long hsn=0;
			for(int j=0;j<len;j++){
				if(st[j]=='\r')continue;
				if((st[j]>='A'&&st[j]<='Z')||(st[j]>='0' && st[j]<='9'))
					hsn=(hsn*Upbit+st[j])%Mod;
				else{
					if(id[hsn]==0)id[hsn]=++pcnt;
					int sid=id[hsn];
					if(f==0) f=sid;
					else {
						add(f,sid);
						deg[sid]++;
					}
					hsn=0;
				}
			}
		}
		cout<<(topsort()?"Yes":"No")<<endl;
	}
}

T3

可以证明将处理出的数组排序然后切开就是优的。

然后有个决策单调性,即转移点是从$i-\frac{i}{j}$到$i$的

那么就可以了(水题使我快乐)

//assignment

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
#include <queue>
#define N 5555
#define M 55555
#define LL long long

using namespace std;

int pn,cn,pron,edn;
LL vals[N],pre[N];
LL dp[N][N],
   siz[N];
int apppn=0;

typedef pair<LL,int> pli;
struct SR{
	int f,t,next;
	int v;
};
priority_queue<pli,vector<pli>,greater<pli> >q;
struct MAPS{
	SR rs[M];
	int fl[N];
	int cnt;
	LL dis[N];
	bool is_v[N];
	MAPS(){
		memset(fl,-1,sizeof fl);
		cnt=0;
	}
	void add(int f,int t,int v){
		rs[cnt].f=f;
		rs[cnt].t=t;
		rs[cnt].v=v;
		rs[cnt].next=fl[f];
		fl[f]=cnt++;
	}
	void dij(int st){
		memset(dis,0x3f,sizeof dis);
		dis[st]=0;
		q.push(make_pair(0,st));
		while(!q.empty()){
			int f=q.top().second;q.pop();
			if(is_v[f])continue;
			is_v[f]=1;
			for(int i=fl[f];i!=-1;i=rs[i].next){
				int t=rs[i].t;
				if(dis[t]>dis[f]+rs[i].v){
					dis[t]=dis[f]+rs[i].v;
					q.push(make_pair(dis[t],t));
				}
			}
		}
	}
}am,bm;

int main(){
#ifndef LOCAL
	freopen("assignment.in" ,"r",stdin);
	freopen("assignment.out","w",stdout);
#endif
	int a,b,v;
	scanf("%d%d%d%d",&pn,&cn,&pron,&edn);
	for(int i=1;i<=edn;i++){
		scanf("%d%d%d",&a,&b,&v);
		am.add(a,b,v);
		bm.add(b,a,v);
	}
	am.dij(cn+1);
	bm.dij(cn+1);
	for(int i=1;i<=cn;i++){
		vals[i]=am.dis[i]+bm.dis[i];
//		cout<<vals[i]<<" ";
	}
//	cout<<endl;
	sort(vals+1,vals+cn+1);
	for(int i=1;i<=cn;i++)
		pre[i]=pre[i-1]+vals[i];
	memset(dp,0x3f,sizeof dp);
	for(int i=1;i<=cn;i++){
		dp[i][1]=pre[i]*(i-1);
	}
	for(int i=1;i<=cn;i++){
		for(int j=2;j<=pron;j++){
			for(int k=i-i/j;k<i;k++){
				dp[i][j]=min(dp[i][j],dp[k][j-1]+(pre[i]-pre[k])*(i-k-1));
			}
		}
	}
//	cout<<dp[cn-1][pron-1]<<endl;
	cout<<dp[cn][pron]<<endl;
}
posted @ 2019-11-12 20:17  Miemeng_麦蒙  阅读(337)  评论(0编辑  收藏  举报

小麦在雨中,蒙蒙的雾气

麦蒙不想有人骚扰他,如果有必要 联系 QQ:1755601414

如果你嫌广告大,那就喷我吧,不是博客园的锅。