首页 > 其他分享 >绍兴集训Day1

绍兴集训Day1

时间:2022-10-04 14:12:23浏览次数:55  
标签:rt int 样例 Day1 绍兴 edges && 集训 dp

Bad Luck Island

题面翻译

给三种人,分别是r,s,p个;

在孤岛上,每两个不同种人相遇则互吃,r吃s,s吃p,p吃r

求最后留下单一种族的概率

题目描述

The Bad Luck Island is inhabited by three kinds of species: $ r $ rocks, $ s $ scissors and $ p $ papers. At some moments of time two random individuals meet (all pairs of individuals can meet equiprobably), and if they belong to different species, then one individual kills the other one: a rock kills scissors, scissors kill paper, and paper kills a rock. Your task is to determine for each species what is the probability that this species will be the only one to inhabit this island after a long enough period of time.

输入格式

The single line contains three integers $ r $ , $ s $ and $ p $ ( $ 1<=r,s,p<=100 $ ) — the original number of individuals in the species of rock, scissors and paper, respectively.

输出格式

Print three space-separated real numbers: the probabilities, at which the rocks, the scissors and the paper will be the only surviving species, respectively. The answer will be considered correct if the relative or absolute error of each number doesn't exceed $ 10^{-9} $ .

样例 #1

样例输入 #1

2 2 2

样例输出 #1

0.333333333333 0.333333333333 0.333333333333

样例 #2

样例输入 #2

2 1 2

样例输出 #2

0.150000000000 0.300000000000 0.550000000000

样例 #3

样例输入 #3

1 1 3

样例输出 #3

0.057142857143 0.657142857143 0.285714285714

解答

采用倒推的方式,设f[r][s][p]为三种生物各有r,s,p时的概率
具体看代码

#include<bits/stdc++.h>
#define rt register int
using namespace std;
int r,s,p;
double f[105][105][105];
double ans1,ans2,ans3;
int main(){
	scanf("%d%d%d",&r,&s,&p);
	f[r][s][p]=1.0;//可以理解成答案的起点,也可以直接理解成概率为1,毕竟这是给出的明确的初始状态。两种方式殊途同归。
	int sum;
	for(rt i=r;i>=0;--i)
		for(rt j=s;j>=0;--j)
			for(rt k=p;k>=0;--k)
			{
				sum=i*j+j*k+i*k;//r吃s,s吃p·······,一共有这么多种可能性
				if(i&&k)f[i-1][j][k]+=1.0*f[i][j][k]*i*k/sum;//当i和k大于0时,p可以吃r
				if(j&&i)f[i][j-1][k]+=1.0*f[i][j][k]*i*j/sum;//同理
				if(j&&k)f[i][j][k-1]+=1.0*f[i][j][k]*j*k/sum;//这里解释为什么是+=,因为从别的分支语句也能转移到这里,从定义上理解确实是同步的。
				if(i&&!j&&!k)ans1+=f[i][j][k];//当j变成0时的可能性累加
				if(!i&&j&&!k)ans2+=f[i][j][k];
				if(!i&&!j&&k)ans3+=f[i][j][k];
			}
	printf("%.9lf %.9lf %.9lf",ans1,ans2,ans3);
	return 0;
}

[USACO12MAR]Cows in a Skyscraper G

题目描述

A little known fact about Bessie and friends is that they love stair climbing races. A better known fact is that cows really don't like going down stairs. So after the cows finish racing to the top of their favorite skyscraper, they had a problem. Refusing to climb back down using the stairs, the cows are forced to use the elevator in order to get back to the ground floor.

The elevator has a maximum weight capacity of W (1 <= W <= 100,000,000) pounds and cow i weighs C_i (1 <= C_i <= W) pounds. Please help Bessie figure out how to get all the N (1 <= N <= 18) of the cows to the ground floor using the least number of elevator rides. The sum of the weights of the cows on each elevator ride must be no larger than W.

给出n个物品,体积为w[i],现把其分成若干组,要求每组总体积<=W,问最小分组。(n<=18)

输入格式

* Line 1: N and W separated by a space.

* Lines 2..1+N: Line i+1 contains the integer C_i, giving the weight of one of the cows.

输出格式

* A single integer, R, indicating the minimum number of elevator rides needed.

one of the R trips down the elevator.

样例 #1

样例输入 #1

4 10 
5 
6 
3 
7

样例输出 #1

3

提示

There are four cows weighing 5, 6, 3, and 7 pounds. The elevator has a maximum weight capacity of 10 pounds.

We can put the cow weighing 3 on the same elevator as any other cow but the other three cows are too heavy to be combined. For the solution above, elevator ride 1 involves cow #1 and #3, elevator ride 2 involves cow #2, and elevator ride 3 involves cow #4. Several other solutions are possible for this input.

解答

我们在维护时需要两个数据,当前分组数,当前空间剩余数。考虑到DP,得出当前分组的最小数量和当前剩下的最大空间。联系背包问题,这种方式能通过记忆化搜索实现。
有了状态,考虑如何转移。
分类讨论:

  1. 当前最大空间能再放下一个
  2. 当前空间无法再放下一个
    注意初始状态的设置。
#include<bits/stdc++.h>
#define rt register int
using namespace std;
int n,W,w[20],dpv[270000],dpcnt[270000];
int main(){
	scanf("%d%d",&n,&W);
	for(rt i=1;i<=n;++i)
		scanf("%d",&w[i]);
	int chose=(1<<n)-1;
	memset(dpcnt,0x3f,sizeof(dpcnt));
	dpcnt[0]=1;
	memset(dpv,0,sizeof dpv);
	dpv[0]=W ;
	for(rt i=0;i<=chose;++i)
		for(rt j=1;j<=n;++j){
			if(i&(1<<(j-1)))continue;
			if(dpv[i]>=w[j]){
				dpcnt[i|(1<<(j-1))]=min(dpcnt[i|(1<<(j-1))],dpcnt[i]);
				dpv[i|(1<<(j-1))]=max(dpv[i|(1<<(j-1))],dpv[i]-w[j]);
//				printf("1:dpcnt[%d|(1<<(%d-1))]=%d\n",i,j,dpcnt[i|(1<<(j-1))]);
//				printf("1:dpv[%d|(1<<(%d-1))]=%d\n",i,j,dpv[i|(1<<(j-1))]);
			}else{
				dpcnt[i|(1<<(j-1))]=min(dpcnt[i|(1<<(j-1))],dpcnt[i]+1);
				dpv[i|(1<<(j-1))]=max(dpv[i|(1<<(j-1))],W -w[j]);
//				printf("2:dpcnt[%d|(1<<(%d-1))]=%d\n",i,j,dpcnt[i|(1<<(j-1))]);
//				printf("2:dpv[%d|(1<<(%d-1))]=%d\n",i,j,dpv[i|(1<<(j-1))]);
			}		
		}
//	for(rt i=1;i<=chose;++i)
		printf("%d",dpcnt[chose]);
	return 0;
}

Choosing Capital for Treeland

题面翻译

题目描述

Treeland国有n个城市,这n个城市连成了一颗树,有n-1条道路连接了所有城市。每条道路只能单向通行。现在政府需要决定选择哪个城市为首都。假如城市i成为了首都,那么为了使首都能到达任意一个城市,不得不将一些道路翻转方向,记翻转道路的条数为k。你的任务是找到所有满足k最小的首都。

输入输出格式

输入格式

输入包含多个测试点。对于每个测试点,每个测试点的第一行为一个正整数n(2<=n<=2e5)。接下来n-1行,每行两个正整数ai,bi,表示城市a到城市b有一条单向通行的道路。输入以空格分隔,以EOF结尾。

输出格式

对于每个测试点,第一行输出k,第二行升序输出所有满足条件的首都的编号。

题目描述

The country Treeland consists of $ n $ cities, some pairs of them are connected with unidirectional roads. Overall there are $ n-1 $ roads in the country. We know that if we don't take the direction of the roads into consideration, we can get from any city to any other one.

The council of the elders has recently decided to choose the capital of Treeland. Of course it should be a city of this country. The council is supposed to meet in the capital and regularly move from the capital to other cities (at this stage nobody is thinking about getting back to the capital from these cities). For that reason if city $ a $ is chosen a capital, then all roads must be oriented so that if we move along them, we can get from city $ a $ to any other city. For that some roads may have to be inversed.

Help the elders to choose the capital so that they have to inverse the minimum number of roads in the country.

输入格式

The first input line contains integer $ n $ ( $ 2<=n<=2·10^{5} $ ) — the number of cities in Treeland. Next $ n-1 $ lines contain the descriptions of the roads, one road per line. A road is described by a pair of integers $ s_{i},t_{i} $ ( $ 1<=s_{i},t_{i}<=n; s_{i}≠t_{i} $ ) — the numbers of cities, connected by that road. The $ i $ -th road is oriented from city $ s_{i} $ to city $ t_{i} $ . You can consider cities in Treeland indexed from 1 to $ n $ .

输出格式

In the first line print the minimum number of roads to be inversed if the capital is chosen optimally. In the second line print all possible ways to choose the capital — a sequence of indexes of cities in the increasing order.

样例 #1

样例输入 #1

3
2 1
2 3

样例输出 #1

0
2

样例 #2

样例输入 #2

4
1 4
2 4
3 4

样例输出 #2

2
1 2 3

解答

这道题犇犇说是裸的换根DP。
1.第一次深搜初始化
2.第二次深搜转移
设dp[i]为以i点作为首都时需要翻转的道路数。
这里有个对应技巧:

  1. 虽然是单向边,但我们可以按照无向图来存图,这里0表示顺,1表示逆,这样处理有利于后期计算
  2. 神奇的深搜
  3. 换根时,如果首都从u到v转移,而存在一条从u到v的边。那么相当于dp[v]=dp[u]-1(思考为什么-1,打字比较麻烦),如果存在的是一条从v到u的边,dp[v]=dp[u]+1。
#include<bits/stdc++.h>
#define rt register int
using namespace std;
struct edge{
	int to,nxt,w;
}edges[500005];	
int idx,h[200005],n,dp[200005],ans=0x3f3f3f3f;
inline void add(int u,int v,int w){
	edges[++idx].to=v;edges[idx].w=w;edges[idx].nxt=h[u];h[u]=idx;
}
inline void dfs1(int u,int fa){
	for(rt i=h[u];i;i=edges[i].nxt){
		int v=edges[i].to;
		if(v==fa)continue;
		dfs1(v,u);
		dp[u]+=dp[v]+edges[i].w;
	}
}
inline void dfs2(int u,int fa){
	for(rt i=h[u];i;i=edges[i].nxt){
		int v=edges[i].to;
		if(v==fa)continue;
		dp[v]=dp[u]+(edges[i].w?-1:1);
		dfs2(v,u);
	}
}
int main(){
	while(~scanf("%d",&n))
	{
		idx=0;
		memset(h,0,sizeof h);
		int u,v;
		for(rt i=1;i<n;++i)
		{
			scanf("%d%d",&u,&v);
			add(u,v,0);add(v,u,1);
		}
		memset(dp,0,sizeof dp);
		dfs1(1,-1);
		dfs2(1,-1);
		ans=0x3f3f3f3f;
		for(rt i=1;i<=n;++i)
			ans=min(ans,dp[i]);
		printf("%d\n",ans);
		for(rt i=1;i<=n;++i)
			if(dp[i]==ans)printf("%d ",i);
		puts("");
	}
	return 0;
}

标签:rt,int,样例,Day1,绍兴,edges,&&,集训,dp
From: https://www.cnblogs.com/zychh/p/16753691.html

相关文章

  • day13:leetcode:150,239,347
    leetcode150.逆波兰表达式求值本题的思路和之前的括号配对,两数相消的思想一样,利用栈先进后出的特性,三个字符相消到遍历到的是数字时,直接push,当遇到运算符,直接pop出两个......
  • 代码随想录day11 ● 232.用栈实现队列 ● 225. 用队列实现栈 ● 20. 有效的括号 ●
    232.用栈实现队列1classMyQueue{2public:34//初始化入队栈和出队栈5stack<int>stIn;6stack<int>stOut;78MyQueue(){9......
  • 国庆集训
    简称国集Day3说实话今天做题时感觉就像昨天磕了药或者今天没吃药一样T1题目描述题目描述小A和小B都是51nod的用户。有一天小A打开了程序挑战排行榜,发现小......
  • 代码随想录day10 ● 459.重复的子字符串 ● 字符串总结 ● 双指针回顾
    459.重复的子字符串1classSolution{2public:3boolrepeatedSubstringPattern(strings){4stringt=s+s;5//掐头去尾6......
  • day11leetcode232,225,20,1047
    225.用队列实现栈利用两个栈来实现队列的基本操作一个负责进栈一个负责出栈classMyQueue{Stack<Integer>stackIn;Stack<Integer>stackOut;publi......
  • ciscn2019华北赛区Day1-Web1dropbox-md
    title:ciscn2019华北赛区Day1Web1dropbox.mddate:2022-06-2620:56:10tags:进去之后呢看到一个登录框然后注册一个账号进去看到有上传文件的东西然后试试上传一个......
  • P5934 [清华集训2012]最小生成树
    简要题意给你一个\(N\)个点,\(M\)条边的无向连通带权图。给定一条边\((u,v,L)\),请问需要在原图中删除多少条边,使得将\((u,v,L)\)插入图后,它既可能在最小生成树上,又......
  • day10
    双指针总结数组移除某一个元素快慢一起向后遍历,遇到要删除的元素。快先走,然后快指针直接覆盖要删除的元素classSolution{publicintremoveElement(int[]n......
  • day1
    markdown学习 二级标题 #+空格字体HEllo引用帅气分割线图片超链接点击跳转列表无序列表表格名字别生日张三男19970101代码​  ......
  • Day14
    异常异常错误代码publicclassDemo01{  publicstaticvoidmain(String[]args){/*   newDemo01().a();  }  publicvoida(){    b()......