P4460 [CQOI2018]解锁屏幕
算是我比较擅长的类型,自己想想就会了。普通小状压,状态傻子都能想出来。一开始裸的枚举T了,30.后来与处理之后跑的飞起,就是不对,还是30分。后来看讨论版。。。mod竟然是1e8+7!!!这不有毒吗。。。
题干:
题目背景
使用过Android 手机的同学一定对手势解锁屏幕不陌生。Android 的解锁屏幕由3X3 个点组成,手指在屏幕上画一条线,将其中一些点连接起来,即可构成一个解锁图案。如下面三个例子所示:
题目描述
画线时还需要遵循一些规则:
连接的点数不能少于4 个。也就是说只连接两个点或者三个点会提示错误。
两个点之间的连线不能弯曲。
每个点只能“使用”一次,不可重复。这里的“使用”是指手指划过一个点,该点变绿。
两个点之间的连线不能“跨过”另一个点,除非那个点之前已经被“使用”过了。
对于最后一条规则,参见下图的解释。左边两幅图违反了该规则; 而右边两幅图(分别为2->--- 和6->-->->-) 则没有违反规则,因为在“跨过”点时,点已经被“使用”过了。
现在工程师希望改进解锁屏幕,增减点的数目,并移动点的位置,不再是一个九宫格形状,但保持上述画线的规则不变。请计算新的解锁屏幕上,一共有多少满足规则的画线方案。
输入输出格式
输入格式:
输入文件第一行,为一个整数n,表示点的数目。
接下来n 行,每行两个空格分开的整数xix_ixi 和yiy_iyi,表示每个点的坐标。
输出格式:
输出文件共一行,为题目所求方案数除以100000007 的余数。
30
#include<iostream>
#include<cstdio>
#include<cmath>
#include<ctime>
#include<queue>
#include<algorithm>
#include<cstring>
using namespace std;
#define duke(i,a,n) for(register int i = a;i <= n;++i)
#define lv(i,a,n) for(register int i = a;i >= n;--i)
#define clean(a) memset(a,0,sizeof(a))
const int INF = << ;
typedef long long ll;
typedef double db;
template <class T>
void read(T &x)
{
char c;
bool op = ;
while(c = getchar(), c < '' || c > '')
if(c == '-') op = ;
x = c - '';
while(c = getchar(), c >= '' && c <= '')
x = x * + c - '';
if(op) x = -x;
}
template <class T>
void write(T x)
{
if(x < ) putchar('-'), x = -x;
if(x >= ) write(x / );
putchar('' + x % );
}
const int mod = 1e9 + ;
const int N = ;
int dp[ << N][N];
int x[N],y[N],n,ans;
/*void dfs(int now,int u,int len)//zhuangtai.xianzaidedian,changdu
{
if(dp[now][u]) return dp[now][u];
for(int i = 1;i <= n;i++)
{
if(now & (1 << (i - 1)) == 0)
{
int ok = 0;
duke(j,1,n)
{
if(now & (1 << (i - 1)) == 0)
{
if(check(u,j,i) == false)
{
ok = 1;
break;
}
}
}
if(ok == 0)
{
dfs(now | (1 << (i - 1)),i,len + 1);
dp[now][u] += dp[now | (1 << (i - 1))]
}
}
}
}*/
bool check(int a,int b,int c)
{
/*if(a == 1 && c == 3 && b == 2)
printf("%d %d %d\n",a,b,c);*/
if((y[b] - y[a]) * (x[c] - x[b]) != (y[c] - y[b]) * (x[b] - x[a]))
return true;
if(x[a] > x[b] && x[c] > x[b])
return true;
if(x[a] < x[b] && x[c] < x[b])
return true;
if(y[a] > y[b] && y[c] > y[b])
return true;
if(y[a] < y[b] && y[c] < y[b])
return true;
return false;
}
void p(int now)
{
duke(i,,n)
{
if((now & ( << (i - ))) != )
printf("");
else
printf("");
}
}
int main()
{
read(n);
duke(i,,n)
read(x[i]),read(y[i]);
duke(i,,n - )
{
dp[ << i][i + ] = ;
}
for(int now = ;now <= ( << n) - ;now++)
{
duke(i,,n)
{
if((now & ( << (i - ))) != )
{
duke(j,,n)
{
if((now & ( << (j - ))) != && i != j)
{
int ok = ;
duke(k,,n)
{
if((now & ( << (k - ))) == )
if(check(i,k,j) == false)
{
ok = ;//cout<<i<<" "<<j<<" "<<k<<endl;
break;
}
}
if(ok == )
{
dp[now][i] += dp[now - ( << (i - ))][j];
// p(now),printf(" ");cout<<i<<" "<<j;printf(" ");p(now - (1 << (i - 1)));
// puts("");
dp[now][i] %= mod;
}
}
}
}
}
int tot = ;
duke(i,,n - )
{
if((now & ( << i)) != )
tot++;
}
if(tot >= )
{
duke(i,,n)
{
if((now & ( << (i - ))) != )
ans += dp[now][i],ans %= mod;
}
}
}
// ans += 1;
// ans /= 2;
/*duke(i,0,(1 << n) - 1)
{
duke(j,1,n)
if((i & (1 << (j - 1))) != 0)
p(i),printf(" "),cout<<dp[i][j]<<endl;
}*/
printf("%d\n",ans);
return ;
}
AC代码:
// luogu-judger-enable-o2
#include<iostream>
#include<cstdio>
#include<cmath>
#include<ctime>
#include<queue>
#include<algorithm>
#include<cstring>
using namespace std;
#define duke(i,a,n) for(register int i = a;i <= n;++i)
#define lv(i,a,n) for(register int i = a;i >= n;--i)
#define clean(a) memset(a,0,sizeof(a))
const int INF = << ;
typedef long long ll;
typedef double db;
template <class T> void read(T &x)
{
char c;
bool op = ;
while(c = getchar(), c < '' || c > '')
if(c == '-') op = ;
x = c - '';
while(c = getchar(), c >= '' && c <= '')
x = x * + c - '';
if(op) x = -x;
}
template <class T>
void write(T x)
{
if(x < ) putchar('-'), x = -x;
if(x >= ) write(x / );
putchar('' + x % );
}
const int mod = 1e8 + ;
const int N = ;
int dp[ << N][N];
int x[N],y[N],n,ans;
bool check(int a,int b,int c)
{
/*if(a == 1 && c == 3 && b == 2)
printf("%d %d %d\n",a,b,c);*/
if((y[b] - y[a]) * (x[c] - x[b]) != (y[c] - y[b]) * (x[b] - x[a]))
return true;
if(x[a] > x[b] && x[c] > x[b])
return true;
if(x[a] < x[b] && x[c] < x[b])
return true;
if(y[a] > y[b] && y[c] > y[b])
return true;
if(y[a] < y[b] && y[c] < y[b])
return true;
return false;
}
void p(int now)
{
duke(i,,n)
{
if((now & ( << (i - ))) != )
printf("");
else
printf("");
}
}
vector <int> mp[N][N];
void init(int x,int y)
{
for(int i = ;i <= n;i++)
{
if(check(x,i,y) == false)
{
mp[x][y].push_back(i);
}
}
}
int main()
{
read(n);
duke(i,,n)
read(x[i]),read(y[i]);
duke(i,,n - )
{
dp[ << i][i + ] = ;
}
duke(i,,n)
duke(j,,n)
init(i,j);
for(int now = ;now <= ( << n) - ;now++)
{
duke(i,,n)
{
if((now & ( << (i - ))) != )
{
duke(j,,n)
{
if((now & ( << (j - ))) != && i != j)
{
int ok = ;
duke(f,,(int)mp[i][j].size() - )
{
int k = mp[i][j][f];
if((now & ( << (k - ))) == )
{
ok = ;//cout<<i<<" "<<j<<" "<<k<<endl;
break;
}
}
if(ok == )
{
dp[now][i] += dp[now - ( << (i - ))][j];
// p(now),printf(" ");cout<<i<<" "<<j;printf(" ");p(now - (1 << (i - 1)));
// puts("");
dp[now][i] %= mod;
}
}
}
}
}
int tot = ;
duke(i,,n - )
{
if((now & ( << i)) != )
tot++;
}
if(tot >= )
{
duke(i,,n)
{
if((now & ( << (i - ))) != )
ans += dp[now][i],ans %= mod;
}
}
}
// ans += 1;
// ans /= 2;
/*duke(i,0,(1 << n) - 1)
{
duke(j,1,n)
if((i & (1 << (j - 1))) != 0)
p(i),printf(" "),cout<<dp[i][j]<<endl;
}*/
printf("%d\n",ans);
return ;
}
代码:
P4460 [CQOI2018]解锁屏幕的更多相关文章
- [Luogu] P4460 [CQOI2018]解锁屏幕
题目背景 使用过Android 手机的同学一定对手势解锁屏幕不陌生.Android 的解锁屏幕由3X3 个点组成,手指在屏幕上画一条线,将其中一些点连接起来,即可构成一个解锁图案.如下面三个例子所示: ...
- bzoj5299: [Cqoi2018]解锁屏幕
题目链接 bzoj 5299: [Cqoi2018]解锁屏幕 题解 很水的装压dp,相信没人需要看题解.... dp[i][j]表示状态为i最后一个到的点为j,然后转移就很好写了 不过 我读入优化没读 ...
- BZOJ5299:[CQOI2018]解锁屏幕(状压DP)
Description 使用过Android手机的同学一定对手势解锁屏幕不陌生.Android的解锁屏幕由3x3个点组成,手指在屏幕上画一条 线将其中一些点连接起来,即可构成一个解锁图案.如下面三个例 ...
- [CQOI2018]解锁屏幕
嘟嘟嘟 这题感觉真的很简单-- \(O(n ^ 2 logn)\)的做法特别好理解,但得开O2. 看数据范围,肯定是状压dp.但刚开始我没想通状压啥,因为点与点之间还有顺序问题.但后来发现这个顺序是子 ...
- 【[CQOI2018]解锁屏幕】
状压这个东西好像没有什么能优化的高级东西,像什么斜率优化,单调队列在状压的优化上都很少见 而最常见的状压优化就是预处理优化了, 这道题就预处理一下所有点对之间连线上的点,之后压成状态就能做到\(O(2 ...
- BZOJ5299 [Cqoi2018]解锁屏幕 【状压dp】
题目链接 BZOJ5299 题解 就一个毒瘤卡常题..写了那么久 设\(f[i][s]\)表示选了集合\(s\)中的点,最后一个是\(i\),进行转移 要先预处理出两点间的点,然后卡卡常就可以过了 # ...
- BZOJ 5299: [Cqoi2018]解锁屏幕
状压DP #include<cstdio> using namespace std; const int mod=1e8+7; int F[1000005][25],dis[25][25] ...
- bzoj 5299: [Cqoi2018]解锁屏幕 状压dp+二进制
比较简单的状压 dp,令 $f[S][i]$ 表示已经经过的点集为 $S$,且最后一个访问的位置为 $i$ 的方案数. 然后随便转移一下就可以了,可以用 $lowbit$ 来优化一下枚举. code: ...
- 【BZOJ5299】【CQOI2018】解锁屏幕(动态规划,状态压缩)
[BZOJ5299][CQOI2018]解锁屏幕(动态规划,状态压缩) 题面 BZOJ 洛谷 Description 使用过Android手机的同学一定对手势解锁屏幕不陌生.Android的解锁屏幕由 ...
随机推荐
- [Python3网络爬虫开发实战] 1.2.6-aiohttp的安装
之前介绍的Requests库是一个阻塞式HTTP请求库,当我们发出一个请求后,程序会一直等待服务器响应,直到得到响应后,程序才会进行下一步处理.其实,这个过程比较耗费资源.如果程序可以在这个等待过程中 ...
- 性能测试培训day1
测试本质: 1构造测试数据和期望结果 2执行 3验证 自动化测试: 写完代码,单元测试测代码逻辑,单元测试搞清楚代码逻辑就行了(白盒测试)先静态,运行前用工具扫描BUG例如(a==11写成a=11), ...
- day21 05 员工信息表
day21 05 员工信息表 假设有一个员工信息表,里面有每个员工的名字,id,年龄,电话,还有他们所作的工作,而有时候我们并不需要所有的信息,而想根据某些条件,寻找符合条件即可,即筛选, 比如想要筛 ...
- CSRF之Ajax请求
A:Ajax提交数据是,携带的CSRF在data中: <form method="POST" action="/csrf.html"> {% csr ...
- 10 行 Python 代码,批量压缩图片 500 张,简直太强大了
本文原创并首发于公众号[Python猫],未经授权,请勿转载. 原文地址:https://mp.weixin.qq.com/s/5hpFDgjCpfb0O1Jg-ycACw 熟悉 "Pyth ...
- Qt——信号与槽用法总结(未完待续)
1.设计模式中信号与槽编辑选项卡 2.右键组件,转到槽,写函数 void LoginDialog::on_loginBtn_clicked() { accept(); } 3.信号与槽编辑模式 按下F ...
- Python条件判断(if)
Python条件判断(if) 一.基本介绍 1.Python 编程中 if 语句用于控制程序的执行,基本形式为: if 判断条件: 执行语句…… 需要注意的是,Python没有像其他大多数语言一样使用 ...
- P1072 Hankson的趣味题
#include<iostream> #include<cstdio> #include<cstring> #include<cmath> #inclu ...
- NOIP2018普及游记
我好弱啊,今年又是考pj啊 今年GD的又是在我们学校有考点(gzez) 考前其实还是蛮紧张的,毕竟考砸了就AFO了.我dp是真的弱,模拟赛连最长下降子序列都不会写,心想要是T3是dp就咕咕咕了.去年那 ...
- Linux下汇编语言学习笔记66 ---
这是17年暑假学习Linux汇编语言的笔记记录,参考书目为清华大学出版社 Jeff Duntemann著 梁晓辉译<汇编语言基于Linux环境>的书,喜欢看原版书的同学可以看<Ass ...