Problem A,B,C:

简单的模拟,注意A中p mod q时对q=0特殊处理(注意范围)


Problem D:

Brief Intro:

给定长度为N的数组A,将A中所有连续子序列分成最少的组,使得每组任意一对数的积均为完全平方数

求最终分成组数为K的子序列个数,K属于[1,N]

Algorithm:

能推出的性质:若P,Q两数积为完全平方数,则任意一个质因子的次幂的奇偶性必然相同

那么想判断P,Q是否满足条件,只要保留每个质因子的次幂为0或1,再判断P,Q是否相同即可

下面只要考虑如何O(N^2)地判断

为了能O(1)判断新加入的数是否已经出现过,需要预处理出每一个数的上一个“自己”出现的位置

由于数的范围过广,使用map记录一个数在检索到k时最后的位置

Code:

#include <bits/stdc++.h>

using namespace std;

const int MAXN=+;
int n,dat[MAXN],res[MAXN],pre[MAXN]; vector<int> prime;
map<int,int> mp; bool isprime(int x)
{
int up_limit=sqrt(x);
for(int i=;i<=up_limit;i++)
if(x%i==) return false;
return true;
} void init()
{
int up_limit=sqrt(1e8+);
for(int i=;i<=up_limit;i++)
if(isprime(i)) prime.push_back(i);
} int trans(int x)
{
int up_limit=sqrt(abs(x)),ret=x;
for(int i=;i<prime.size();i++)
{
if(prime[i]>up_limit || ret== || ret==-) break; int t=ret,cnt=;
while(t%prime[i]==) t/=prime[i],cnt++;
if(cnt%==) ret=t*prime[i];
else ret=t;
}
return ret;
} int main()
{
cin >> n;
for(int i=;i<=n;i++)
cin >> dat[i]; init(); for(int i=;i<=n;i++) //质因数分解
dat[i]=trans(dat[i]); for(int i=;i<=n;i++) //预处理pre
if(mp.count(dat[i]))
{
pre[i]=mp[dat[i]];
mp[dat[i]]=i;
}
else
{
pre[i]=-;
mp[dat[i]]=i;
} for(int i=;i<=n;i++)
{
bool f=true;int cnt=;
for(int j=i;j<=n;j++)
{
if(dat[j]) f=false;
if(pre[j]<i && dat[j]) cnt++,mp[dat[j]]=true; //O(1)判断
if(!f) res[cnt]++;
else res[]++;
}
} for(int i=;i<=n;i++) cout << res[i] << " ";
return ;
}

Review:

1、特解:0

在看到数据范围后,总要考虑特解。

除非一段全部为0,否则忽略当前遇到的0

2、积为完全平方数的性质:

我当时只想到了传导性,反而忽略了每个质因子次幂奇偶性相同这一性质

只考虑奇偶性   到    转化后判断相等的方法值得借鉴

3、求解一串数中不同数的个数的预处理:

求出每一个数前一次出现的位置     常用的预处理方式

Problem E:

一棵树中有N个点,每个点的权值为2^N

要舍去K个点,使得这K个点的权值和最小,且剩下的点连通

Algorithm:

显而易见的贪心策略:

反向求解,寻找n-k个要选的点

由于第n个点的权值 > 1~n-1的权值和,所以从第n个点开始贪心选取即可

为了将复杂度控制在 O(NlogN) ,使用树上倍增查找路径终点

Code:

#include <bits/stdc++.h>

using namespace std;

const int MAXN=1e6+;
vector<int> G[MAXN],res;
int n,k,f[MAXN][],vis[MAXN],dep[MAXN]; inline int read()
{
char ch;int f=,num;
while(!isdigit(ch=getchar())) f|=(ch=='-');
num=ch-'';
while(isdigit(ch=getchar())) num=num*+ch-'';
return f?-num:num;
} void dfs(int cur,int anc) //初始化
{
dep[cur]=dep[anc]+;f[cur][]=anc;
for(int i=;i<=;i++) f[cur][i]=f[f[cur][i-]][i-]; for(int i=;i<G[cur].size();i++)
{
int v=G[cur][i];
if(v==anc) continue;
dfs(v,cur);
}
} int main()
{
n=read();k=read();
for(int i=;i<n;i++)
{
int x=read(),y=read();
G[x].push_back(y);G[y].push_back(x);
} dfs(n,); memset(vis,,sizeof(vis));
vis[n]=vis[]=;k=n-k-;
for(int i=n-;i>=;i--)
{
if(vis[i]) continue; int t=i;
for(int j=;j>=;j--) //倍增找路径
if(!vis[f[t][j]]) t=f[t][j]; if(dep[i]-dep[t]+<=k)
{
k-=(dep[i]-dep[t]+);
t=i;
while(!vis[t]) vis[t]=,t=f[t][];
}
else res.push_back(i);
} sort(res.begin(),res.end());
for(int i=;i<res.size();i++) cout << res[i] << " ";
return ;
}

Review:

1、当正向贪心难以实现时,可以尝试反向贪心

2、当要在树上O(logN)搜寻路径时,使用树上倍增法

Codeforces #480 Tutorial的更多相关文章

  1. [Codeforces #172] Tutorial

    Link: Codeforces #172 传送门 A: 一眼看上去分两类就可以了 1.每个矩形只有两条边相交,重合的形状为菱形 2.每个矩形四条边都有相交 对于情况1答案为$h*h/sin(a)$ ...

  2. [Codeforces #514] Tutorial

    Link: Codeforces #514 传送门 很简单的一场比赛打崩了也是菜得令人无话可说…… D: 一眼二分,发现对于固定的半径和点,能包含该点的圆的圆心一定在一个区间内,求出区间判断即可 此题 ...

  3. [Codeforces #210] Tutorial

    Link: Codeforces #210 传送门 A: 贪心,对每个值都取最大值,不会有其他解使答案变优 #include <bits/stdc++.h> using namespace ...

  4. [Codeforces #196] Tutorial

    Link: Codeforces #196 传送门 A: 枚举 #include <bits/stdc++.h> using namespace std; #define X first ...

  5. [Codeforces #174] Tutorial

    Link: Codeforces #174 传送门 A: 求原根的个数,有一条性质是原根个数为$\phi(\phi(n))$,多了一个不会证的性质 如果要确定哪些是原根的话还是要枚举,不过对于每个数不 ...

  6. [Codeforces #190] Tutorial

    Link: Codeforces #190 传送门 A: 明显答案为$n+m-1$且能构造出来 #include <bits/stdc++.h> using namespace std; ...

  7. [Codeforces #211] Tutorial

    Link: Codeforces #211 传送门 一套非常简单的题目,但很多细节都是错了一次才能发现啊…… 还是不能养成OJ依赖症,交之前先多想想corner case!!! A: 模拟,要特判0啊 ...

  8. [Codeforces #192] Tutorial

    Link: Codeforces #192 传送门 前两天由于食物中毒现在还要每天挂一天的水 只好晚上回来随便找套题做做找找感觉了o(╯□╰)o A: 看到直接大力模拟了 但有一个更简便的方法,复杂度 ...

  9. #2 codeforces 480 Parcels

    题意: 就是有一个用来堆放货物的板,承重力为S.现在有N件货物,每件货物有到达的时间,运走的时间,以及重量,承重,存放盈利.如果这件货物能再运达时间存放,并在指定时间取走的话,就能获得相应的盈利值.货 ...

随机推荐

  1. POJ1637:Sightseeing tour(混合图的欧拉回路)

    Sightseeing tour Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 10581   Accepted: 4466 ...

  2. MongoDB加索引

    1.登陆MongoDB, 命令行在MongoDB主目录下执行:mongo -port 27017 2.切换至需要添加索引的db 并授权 use MeetingBooking db.auth({&quo ...

  3. Linux shell命令无法使用

    配置java环境变量时,把PATH=$PATH:$HOME/bin中的冒号“:”错写成分号“;”.导致变量PATH配置错误,不能正确找到shell命令的文件位置.把/etc/profile 中java ...

  4. vue2学习篇一 $mount()手动挂载

    $mount()手动挂载 //当Vue实例没有el属性时,则该实例尚没有挂载到某个dom中: //假如需要延迟挂载,可以在之后手动调用vm.$mount()方法来挂载.例如: new Vue({ // ...

  5. Spring 中 AbstractExcelView 支持根据模板生成Excel文件. 通过设置 view 的 URL 属性指定模板的路径

     注意:1. 模板需放在 WEB-INF 目录下2. 指定模板路径时不需要添加扩展名, Spring将自动添加 .xls 到URL 属性中.3. 在指定URL前需先设置 view 的 Applicat ...

  6. bzoj1862: [Zjoi2006]GameZ游戏排名系统

    Description GameZ为他们最新推出的游戏开通了一个网站.世界各地的玩家都可以将自己的游戏得分上传到网站上.这样就可以看到自己在世界上的排名.得分越高,排名就越靠前.当两个玩家的名次相同时 ...

  7. MySQL实现分页查询

    imit 基本实现方式 一般情况下,客户端通过传递 pageNo(页码).pageSize(每页条数)两个参数去分页查询数据库中的数据,在数据量较小(元组百/千级)时使用 MySQL自带的 limit ...

  8. [转]使用 LDAP OU 限制访问

    使用 LDAP OU 限制访问 http://www-01.ibm.com/support/knowledgecenter/api/content/SSEP7J_10.2.2/com.ibm.swg. ...

  9. 【hdu3080】01背包(容量10^7)

    [题意]n个物品,有wi和vi,组成若干个联通块,只能选取一个联通块,问得到m的价值时最小要多少空间(v).n<=50,v<=10^7 [题解] 先用并查集找出各个联通块. 这题主要就是v ...

  10. bzoj 2659 几何

    首先考虑(0, 0)到(p, q)这条直线. y = q / p * x. sum{k = 0 to (p - 1) / 2} [q / p * k] 就是直线下方的点数.sum{k = 0 to ( ...