zoukankan      html  css  js  c++  java
  • 第四场膜你赛总结

    前言

    这场比之前的能做多了QwQ


    题目+题解

    T1 原题:NKOI 3539 移棋子游戏[6月月赛题A]

    【题意】

    明明和亮亮在玩一个游戏。桌面上一行有n个格子,一些格子中放着棋子。明明和亮亮轮流选择如下方式中的一种移动棋子(图示中o表示棋子,*表示空着的格子):
    1)当一枚棋子的右边是空格子的话,可以将这枚棋子像右移动一格。
    **o*** -> ***o**
    2)当一枚棋子的右边连续两个都有棋子,并且这个棋子往右边数第3格没有棋子,那么可以将这个棋子可以跳过去那两个棋子
    **ooo* -> ***oo*
    当任何一枚棋子到达最右边的格子时,这枚棋子自动消失。当一方不能移动时,这方输。假设明明和亮亮都采取最优策略,明明先走,谁将取胜?


    50%的点n是小于20的,但是我可能先做了第二题,记错了范围啊,以为是100,暴力也没打。就YY了一种方法,其实跟正解有点类似:我的想法就是按棋堆(就是一段连续的棋子)来看,不是看每个棋子。算到每个棋堆到最右边的步数和,再判奇偶。

    正解就直接贴了..

    考虑每个棋子到最右边格子的距离。把所有棋子这样的距离的总和计为s。我们发现不管选择两种操作中的一种操作,每走一步,s的奇偶性都会发生一次变化。所以说,如果第一次轮到明明时,s是奇数,那么每次轮到明明时s都是奇数。而当s是奇数时,s肯定>0,这时明明总可以走最右边的棋子。也就是说当s为奇数时,总有棋子可以走。所以说,一开始若s为奇数,则明明必胜。同理,若一开始s为偶数,则当亮亮走的时候s总是奇数,所以明明必败。


    T2

    【题意】

    有一块n *n 的土地上,明明和亮亮站在(1,1)处。每块地上写有一个数字a(i, j)。现在他们决定玩一个游戏,每一秒钟,他们俩走向相邻且坐标变大的格子(从(x,y)到(x+1,y)或者从(x,y)到(x,y+1)),他们俩可以按照不同方式来走,最后经过2n-1步到达(n,n)处。明明和亮亮每一秒钟计算他们站的两个位置上数字的差的绝对值,他们希望这些差值的和最大,请问这个最大的和是多少?


    这个刚看完题目和范围,写了写就马上决定先跳过第一题打第二题先了。经典的DP。设f[k][i][j]表示已经走了k步,明明走到的横坐标为i,亮亮走到的横坐标为j.当然还有其他的状态表示我就不说了。这样我们就可以通过这个表示出两人走了k步后分别所在的坐标:(i,k+1-i);(j,k+1-j) -> 设为(x1,y1);(x2,y2)。

    那么状态转移的方程就是f[k][i][j]=max(f[k-1][i-1][j-1],f[k-1][i][j-1],f[k-1][i-1][j],f[k-1][i][j])+abs(a[x1][y1]-a[x2][y2]).

    时间复杂度是O(n^3).


    T3

    【题意】

    有一块N*M的草地组成。显然有的地方草长的好,有的地方长的不好,于是每一块草都有一个舒服度 F。
    现在明明和亮亮要选定一个a*b的草场作为晚会的地点,小朋友们就坐在上面,显然他希望小朋友们坐的最舒服!
    不过别急,篝火晚会怎么能少了篝火呢,篝火需要占用c*d的草地,当然,篝火必须严格放置在选定的草地的内部,也就是说,篝火的边界不能和选定操场的边界有公共部分。
    给定 N*M 大草地每一块的舒服度,寻找一个 a*b 的草地,除去一个严格内部的 c*d 的子草地,使得总的舒服度最大。


    这道题比赛花的时间最多了。前后打了大概四五个版本(因为怕错,所以我的代码是一步一步越改越快的..),互相对拍= =也是醉。

    在打暴力的时候我们会发现,如果能预处理出当我选择某处的草地时严格内部最小的子草地的值的话,我的时间复杂度就能从O(n^4)降到O(n^2)--一个能接受的时间复杂度。

    而由子草地的大小是固定的,就很容易想到滑动的窗口啊,学单调队列的第一道题!然后直接用啊就好了..

    一开始像GDXB那样想过二维树状数组,也像hyc那样想过用RMQ来维护,但是我都没有打(因为想到用单调队列了,感觉这个好打很多啊。


    T4 原题-hdu4812

    【题意】


    嗯点分治啊,我不会啊。。所以要学啊。。

    设d[x]为x到重心root的路径上点权的乘积,那么满足条件的点对(x,y)就满足d[x]*d[y]/d[root]≡k(mod 10^6+3)。也就是d[x]≡k*d[root]*d[y]^(-1) (mod 10^6+3)。因为模数比较小,就可以直接开个数组存这些k*d[root]*d[y]^(-1)值,于是找其他子树的时候只需要看看那个数组里有没有和v[x]一样的值就好了。

    然后去做点分治的题了

    #pragma comment(linker,"/STACK: 102400000,102400000")
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    typedef long long LL;
    #define maxn 100010
    #define MX 1000010
    
    const LL mod=1000003;
    struct node
    {
    	int y,next;
    }a[maxn*2];
    int len,first[maxn];
    int siz[maxn];//以某点为根的子树大小
    int sta[maxn],tp;
    int s[maxn],sl;//这个就是用来方便清空w数组 加速而已
    bool mark[maxn];//标记重心 那些点成为过重心
    LL v[maxn],d[maxn];//v如题意 d为到重心的路径上点权的乘积
    LL w[MX],ny[MX];//w[x]就表示存在x这个值而更新w[x]这个值的点的编号为w[x] ny就是逆元啊
    LL kk;int ans1,ans2;
    void ins(int x,int y)
    {
    	a[++len].y=y;
    	a[len].next=first[x];first[x]=len;
    }
    LL qk(LL x,LL t)
    {
    	LL ret=1;
    	while (t)
    	{
    		if (t&1) ret=(ret*x)%mod;
    		x=(x*x)%mod;t>>=1;
    	}return ret;
    }
    void dfs_root(int x,int fa,int sum,int &root)//找重心
    {
    	siz[x]=1;bool bk=1;
    	for (int k=first[x];k!=-1;k=a[k].next)
    	{
    		int y=a[k].y;
    		if (y==fa || mark[y]) continue;
    		dfs_root(y,x,sum,root);
    		if (siz[y]*2>sum) bk=0;
    		siz[x]+=siz[y];
    	}
    	if (2*(sum-siz[x])>sum) bk=0;
    	if (bk) root=x;
    }
    void dfs2(int x,int fa)//扫一遍子树
    {
    	d[x]=d[fa]*v[x]%mod;
    	siz[x]=1;sta[++tp]=x;
    	if (w[d[x]])//更新答案
    	{
    		int as1,as2;
    		if (w[d[x]]>x) as1=x,as2=w[d[x]];
    		else as1=w[d[x]],as2=x;
    		if (!ans1) {ans1=as1;ans2=as2;} 
    		else if (as1<ans1) {ans1=as1;ans2=as2;}
    		else if (as1==ans1) ans2=(ans2>as2)?as2:ans2;
    	}
    	for (int k=first[x];k!=-1;k=a[k].next)
    	{
    		int y=a[k].y;
    		if (y==fa || mark[y]) continue;
    		dfs2(y,x);siz[x]+=siz[y];
    	}
    }
    void dfs(int x,int sum)
    {
    	dfs_root(x,0,sum,x);
    	d[x]=v[x];sl=0;s[++sl]=kk;
    	w[kk]=x;mark[x]=1;//因为重心的话k*d[root]*d[root]^(-1)≡k
    	for (int k=first[x];k!=-1;k=a[k].next)
    	{
    		int y=a[k].y;
    		if (mark[y]) continue;
    		dfs2(y,x);
    		while (tp)
    		{
    			int p=sta[tp];//把刚扫完的子树更新到w数组里
    			LL du=(kk*ny[d[p]]%mod)*d[x]%mod;
    			if (!w[du] || p<w[du]) w[du]=p;
    			s[++sl]=du;
    			tp--;
    		}
    	}
    	while (sl) w[s[sl--]]=0;
    	//分治 - 每个子树重复上述内容..
    	for (int k=first[x];k!=-1;k=a[k].next)
    	{
    		int y=a[k].y;
    		if (mark[y]) continue;
    		dfs(y,siz[y]);//dfs(y,(siz[y]*2>sum)?sum-siz[y]:siz[y]);
    	}
    }
    int main()
    {
    	int n,i,j,x,y;
    	ny[1]=1;
    	for (i=2;i<mod;i++) ny[i]=(mod-mod/i)*ny[mod%i]%mod;//快好多!!!
    	// for (i=2;i<mod;i++) ny[i]=qk(i,mod-2);
    	scanf("%d%lld",&n,&kk);
    	len=0;memset(first,-1,sizeof(first));
    	for (i=1;i<=n;i++)
    	{
    		scanf("%lld",&v[i]);
    		v[i]%=mod;
    	}
    	for (i=1;i<n;i++)
    	{
    		scanf("%d%d",&x,&y);
    		ins(x,y);ins(y,x);
    	}
    	ans1=ans2=0;
    	dfs(1,n);
    	if (!ans1) printf("No solution
    ");
    	else printf("%d %d
    ",ans1,ans2);
    	return 0;
    }
    


  • 相关阅读:
    代码命名,代码里的命名规则:错误的和正确的对比 命名方法总结 “自我描述的源代码”用代码表达出你的思想,让其他人通过代码能明白你的意图。
    201508整理:前端工程师的(现在主要的问题、工作选择、技术前景未来)
    java小程序,用java在指定目录或以及子目录中找出同名文件,java File过滤文件名后找同名文件
    excel中的数据粘贴不全到plsql中,excel 粘贴后空白,Excel复制粘贴内容不全
    漫画的由来,卡通漫画、暴走漫画、先锋漫画、漫画与前卫艺术之间的边缘艺术形式、实用漫画、讽刺漫画、幽默漫画的定义
    人性漫画:一个人成功前和成功后赤裸裸的区别 人成功前后对比 成功人发展由来前后结果
    解决IE弹框提示“是否停止运行此脚本”问题
    用UltraEdit判断打开文件的编码类型 用UltraEdit或notepad记事本查看文件编码格式 用UltraEdit查看当前文件编码
    Oracle pl/sql导入sql文件,插入更新数据,中文乱码问题解决方案
    毕胜自述:我是如何把奢侈品打到白菜价的 制造业的电子商务中国零售 小米启示录
  • 原文地址:https://www.cnblogs.com/Euryale-Rose/p/6527822.html
Copyright © 2011-2022 走看看