【2013杭州区域赛】部分题解 hdu4770—4780
1008:
题意:
有20W个数,每个数都在20W以内,现在有20W个询问,每次询问L,R区间内不会打架的数有多少个
定义两个数不互质 就会打架
解法:
脑洞很大的一道题,先要进行预处理,对每一个数预处理出 最大的,没有跟他不互质数(即这个数不会打架)的区间
显然可以通过找到这个数左边和右边的第一个跟他互质的数来求得此区间,那么这个怎么求呢,发现可以通过枚举当前数的质因数找到
我们先处理,每一个数左边的最后一次出现的坐标,这样每扫到一个数,都只需要对这个数及其质因数进行更新
找到了每个数的上述区间后,我们发现设某个数的上述区间为 [L,R],这个数自己的位置为pos,现在对于一个询问[a,b]
这个数是询问中的一个答案的条件就是 L<=a,b<=R 且 a<=pos<=b。现在可以离线进行更新和查询
我们可以在L 处将区间[pos,R]位置的值+1,并在pos处对该区间的值 -1,那么对于某个查询[a,b] 只需要在扫到a的时候,单点查询b的值即可
代码:
#include <bits/stdc++.h>
using namespace std;
int a[];
int l[];
int r[];
int pre[];
vector<int>d[];
bool np[];
void setprime(int maxn)
{
memset(np,,sizeof(np));
for(int i=; i<=maxn; i++)
{
if(np[i]==)
d[i].push_back(i);
for(int j=i+i; j<=maxn; j+=i)
{
np[j]=;
if(np[i]==)
d[j].push_back(i);
}
}
}
int n,m;
struct node
{
int type; //-1:+1 -2:-1,else ²éѯ;
int pos;
int num;
bool operator <(const node a)const
{
if(pos!=a.pos)
return pos<a.pos;
else
{
return type<a.type;
}
}
};
int ans[];
vector<node>q;
int t[];
int lowbit(int x)
{
return x&(-x);
} int getSum(int x)
{
int sum = ;
while(x)
{
sum += t[x];
x -= lowbit(x);
}
return sum;
} void add(int x , int val)
{
while(x <=)
{
t[x] += val;
x += lowbit(x);
}
}
void init()
{
memset(t,,sizeof(t));
q.clear();
memset(l,,sizeof(l));
for(int i=; i<=n; i++)
{
r[i]=n+;
}
}
int main()
{
//freopen("in.txt","r",stdin);
setprime();
while(scanf("%d%d",&n,&m),n+m)
{
init();
for(int i=; i<=n; i++)
{
scanf("%d",a+i);
}
memset(pre,,sizeof(pre));
for(int i=; i<=n; i++)
{
for(int j=; j<(int)d[a[i]].size(); j++)
{
int now=d[a[i]][j];
l[i]=max(pre[now],l[i]);
pre[now]=i;
}
l[i]++;
}
memset(pre,0x3f,sizeof(pre));
for(int i=n; i>; i--)
{
for(int j=; j<(int)d[a[i]].size(); j++)
{
int now=d[a[i]][j];
r[i]=min(pre[now],r[i]);
pre[now]=i;
}
r[i]--;
q.push_back(node {-,l[i],i});
q.push_back(node {m,i,i});
}
for(int i=; i<m; i++)
{
int u,v;
scanf("%d%d",&u,&v);
q.push_back(node {i,u,v});
}
//printf("%d\n",q.size());
sort(q.begin(),q.end());
for(int i=; i<(int)q.size(); i++)
{
node now=q[i];
if(now.type==-)
{
add(now.num,);
add(r[now.num]+,-);
}
if(now.type==m)
{
add(now.num,-);
add(r[now.num]+,);
}
if(now.type>=&&now.type<m)
{
ans[now.type]=getSum(now.num);
}
}
for(int i=; i<m; i++)
{
printf("%d\n",ans[i]);
}
}
return ;
}
1009:
题意:
有g种不同颜色的小球,b个袋子,每个袋子里面有若干个每种小球
两人轮流取袋子,当袋子里面的同色小球有s个时,会合并成一个魔法球,并被此次取袋子的人获得
成功获得魔法球的人可以再次取
求二者都进行最优策略之后两人所得魔法球个数差
解法:
数据不大,考虑搜索,发现不行,于是可以记忆化搜索,转移不难想到
代码:
#include <iostream>
#include <stdio.h>
#include<string.h>
#include<algorithm>
#include<string>
#include<ctype.h>
using namespace std;
#define MAXN 10000
typedef struct Node
{
int a,b;
}node;
node dp[];
bool vi[];
int num[][];
int p[];
int g,b,s,k,m,x;
void dfs(int state,int turn)
{
if(state==(<<b)-)
{
dp[state].a=;
dp[state].b=;
return ;
}
if(vi[state])
{
return ;
}
int ok;
node t;
node ans;
ans.a=-;
ans.b=;
int pp[];
memcpy(pp,p,sizeof(pp));
for(int i=;i<b;i++)
{
t.a=;
t.b=;
ok=;
if(state&(<<i))
continue;
int st=state|(<<i); //状态
for(int j=;j<g;j++)
{
p[j]+=num[i][j];
ok+=p[j]/s;
p[j]%=s;
}
int tur=ok?turn:!turn;//是否交换选手
dfs(st,tur);
memcpy(p,pp,sizeof(pp));
t.a+=ok;
if(tur==turn) //子状态的先手所要转移到的状态相同
{
t.a+=dp[st].a;
t.b+=dp[st].b;
}
else //选手交换了
{
t.b+=dp[st].a;
t.a+=dp[st].b;
}
if(t.a-t.b>ans.a-ans.b)
{
ans=t;
}
}
vi[state]=;
dp[state]=ans;
return ;
}
int main()
{
while(scanf("%d%d%d",&g,&b,&s),g+b+s)
{
memset(num,,sizeof(num));
for(int i=;i<b;i++)
{
scanf("%d",&k);
while(k--)
{
scanf("%d",&x);
num[i][x-]++;
}
}
memset(p,,sizeof(p));
memset(vi,,sizeof(vi));
dfs(,);
printf("%d\n",dp[].a-dp[].b);
}
return ;
}
【2013杭州区域赛】部分题解 hdu4770—4780的更多相关文章
- HDU 4745 Two Rabbits (2013杭州网络赛1008,最长回文子串)
Two Rabbits Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Tota ...
- HDU 4747 Mex (2013杭州网络赛1010题,线段树)
Mex Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Submis ...
- HDU 4741 Save Labman No.004 (2013杭州网络赛1004题,求三维空间异面直线的距离及最近点)
Save Labman No.004 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- HDU 4739 Zhuge Liang's Mines (2013杭州网络赛1002题)
Zhuge Liang's Mines Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- HDU 4738 Caocao's Bridges (2013杭州网络赛1001题,连通图,求桥)
Caocao's Bridges Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- 【2013南京区域赛】部分题解 hdu4802—4812
上周末打了一场训练赛,题目是13年南京区域赛的 这场题目有好几个本来应该是我擅长的,但是可能是太久没做比赛了各种小错误代码写的也丑各种warusn trush搞得人很不爽 全场题之一的1002也没有想 ...
- 2013杭州现场赛B题-Rabbit Kingdom
杭州现场赛的题.BFS+DFS #include <iostream> #include<cstdio> #include<cstring> #define inf ...
- 【2013长沙区域赛】部分题解 hdu4791—4801
1001: 签到题,二分一下即可 代码: #include <set> #include <map> #include <cmath> #include <c ...
- 2013 ACM区域赛长沙 C Collision HDU 4793
题意:在平面上0,0点,有一个半径为R的圆形区域,并且在0,0点固定着一个半径为RM(<R)的圆形障碍物,现在圆形区域外x,y,有一个半径 为r的,并且速度为vx,vy的硬币,如果硬币碰到了障碍 ...
随机推荐
- 解除网页右键限制和开启网页编辑状态的js代码
当访问页面右键被限制了怎么办?很好办!将以下代码添加进收藏夹,点击执行即可: javascript:alert(document.onselectstart = document.onbeforeco ...
- Android 模仿微信启动动画(转)
本文内容 环境 项目结构 演示微信启动动画 本文演示微信启动动画.请点击此处下载,自行调试. 顺便抱怨一下,实践性(与研究性质的相对)技术博的“七宗罪”: 第一宗罪,错字连篇,逻辑不清: 第二宗罪,文 ...
- Java基础知识强化之IO流笔记19:FileOutputStream的三个write方法
1. FileOutputStream的三个write方法: void write(byte[] buffer) Writes the entire contents of th ...
- 第五章:最后一步准备,1.8的Json模型、状态描述机制详解
<基于1.8 Forge的Minecraft mod制作经验分享> 1.8的所有纹理材质都需要一个Json来对其描述,这一块感觉是各大神的教程里面涉及最少最浅的,我就斗胆在这分享下我研究了 ...
- struts.enable.DynamicMethodInvocation = true 动态方法调用(转)
原文地址:http://blog.csdn.net/wfcaven/article/details/5937557 default.properties 在Struts 2的核心jar包-struts ...
- 中国剩余定理模板poj1006
#include <cstdio> #include <iostream> #include <cstring> #include <cmath> #i ...
- 下拉框上移、下移、添加、移除demo
<meta http-equiv="content-type" content="text/html; charset=UTF-8"> <sc ...
- JS 点击事件失效
有时候,会发现js失效 ,代码如果是如下的时候 <input type="button" onclick="change()"value="変更 ...
- Android中使用PullToRefreshListView遇到的问题
1.布局文件中要是设置visible属性为Gone的时候,注意了,这样会有一个bug,在代码中setVisible的时候设置为VISIBLE是不起作用的..这个应该是自身的一个小bug(本人目前没有找 ...
- android launcher3 home页简易分析
最近在修改一个问题:就是修改home页下,用户手动拖出来的APP图片下面的字体显示不全,思路比较明确,需要尽量加大整个APP控件的高度,或者缩小图片和文字之间的间隔. 跟代码发现APP整个控件的lay ...