% Day1 Solution

% Wearry

% Stay determined!

养花

  

考虑当 kkk 确定的时候如何求答案,

显然对于所有形如 [ak,(a+1)k)[ak, (a+1)k)[ak,(a+1)k) 的值域区间, 最大值一定是最优的.

  

进一步观察发现, 这样的区间总个数只有 kln⁡kk \ln kklnk 个.

考虑分块, 那么我们可以在 O(n+kln⁡k)O(n + k \ln k)O(n+klnk) 的时间复杂度内处理出一个块对于任意 kkk 的答案.

询问时复杂度是 O(mS)O(mS)O(mS) 的, 取 S=kln⁡kS = \sqrt{k \ln k}S=klnk​ 可以达到最优复杂度 O(nkln⁡k)O(n \sqrt{k \ln k})O(nklnk​).

#include <bits/stdc++.h>
using namespace std;
const int MAXN = 100000;
const int B = 1000;
int n, m, A[MAXN+5], ans[MAXN/B+5][MAXN+5], b[MAXN+5];
int main ()
{
scanf("%d%d", &n, &m);
for(int i = 0; i < n; i++)
scanf("%d", &A[i]);
int blks = (n-1)/B+1;
for(int i = 0; i < blks; i++)
{
memset(b, 0, sizeof b);
for(int j = i*B; j < B*(i+1) && j < n; j++) b[A[j]] = A[j];
for(int j = 1; j <= MAXN; j++) if(!b[j]) b[j] = b[j-1];
for(int j = 1; j <= MAXN; j++) for(int k = 0; k <= MAXN; k += j) ans[i][j] = max(ans[i][j], b[min(k+j-1,MAXN)]-k);
}
int x, y, z, a, b, Ans;
while(m--)
{
scanf("%d%d%d", &x, &y, &z), x--, y--;
a = x/B + 1, b = y/B, Ans = 0;
if(a > b) for(int i = x; i <= y; i++) Ans = max(Ans, A[i]%z);
else
{
for(int i = x; i < a*B; i++) Ans = max(Ans, A[i]%z);
for(int i = b*B; i <= y; i++) Ans = max(Ans, A[i]%z);
for(int i = a; i < b; i++) Ans = max(Ans, ans[i][z]);
}
printf("%d\n", Ans);
}
}

折射

  

若将所有点按照 yiy_iyi​ 的顺序转移, 有上界和下界两个限制, 难以优化.

  

考虑按照 xix_ixi​ 排序转移, 并记录 fi,0/1f_{i, 0/1}fi,0/1​ 表示以第 iii 个点为顶端接下来向左或向右的折线方案数.

从左到右加点, 考虑前 iii 个点构成的包含 iii 点的折线. 由于新点横坐标最大, 所以只可能在折线的第一位或第二位:

  1. ∀yj&lt;yi,fi,0←fj,1\forall y_j &lt; y_i, f_{i, 0} \leftarrow f_{j, 1}∀yj​<yi​,fi,0​←fj,1​

  2. ∀yj&gt;yi,fj,1←fk,0∣xk&gt;xj&ThinSpace;&ThinSpace;and&ThinSpace;&ThinSpace;yk&lt;yi\forall y_j &gt; y_i, f_{j, 1} \leftarrow f_{k, 0} \mid x_k &gt; x_j \,\, \mathrm{and} \,\, y_k &lt; y_i∀yj​>yi​,fj,1​←fk,0​∣xk​>xj​andyk​<yi​

  

第二种情况可以前缀和优化, 复杂度 O(n2)O(n^2)O(n2).

#include <bits/stdc++.h>
using namespace std;
#define file(x) freopen(x".in", "r", stdin);freopen(x".out", "w", stdout);
#define pii pair<int,int>
namespace IO
{
inline void read(int &num)
{
char ch; int flag = 1;
while(!isdigit(ch=getchar()))if(ch=='-')flag=-flag;
for(num=ch-'0';isdigit(ch=getchar());num=num*10+ch-'0');
num *= flag;
}
}
using namespace IO;
const int MAXN = 100005;
const int mod = 1e9+7;
int n, dp[MAXN][2];
pii p[MAXN];
int main ()
{
read(n);
for(int i = 1; i <= n; i++)
read(p[i].first), read(p[i].second);
sort(p + 1, p + n + 1);
int Ans = 0;
for(int i = 1; i <= n; i++)
{
dp[i][0] = dp[i][1] = 1;
for(int j = i-1; j; j--)
if(p[j].second < p[i].second)
(dp[i][0] += dp[j][1]) %= mod;
else (dp[j][1] += dp[i][0]) %= mod;
}
for(int i = 1; i <= n; i++) (Ans += (dp[i][0] + dp[i][1]) % mod) %= mod;
printf("%d\n", (Ans+mod-n)%mod);
}

画作

  

不难证明猜到一个这样的结论: 存在一种最优方案使得每次操作的区域是上一次的子集且颜色与上一次相反.

  

考虑归纳证明, 记 SSS 为当前所有操作区域的并, TTT 为接下来一步的操作区域, 我们有:

  1. TTT 与 SSS 有交的情况一定可以转化成 TTT 被 SSS 包含的情况.

  2. TTT 与 SSS 交集为空时, 可以找一个连接 SSS 和 TTT 的集合 MMM 并操作 S∪T∪MS \cup T \cup MS∪T∪M,

    并将之前的所有操作连接到更外的层以及外层的连接部分同时操作, 特殊处理最外层和第二层的情况.

  3. TTT 被 SSS 包含时, TTT 落在某个完整区域内时等价于情况二,

    否则一定连接若干个同色块, 这些块可以同时处理, 步数一定不会更劣.

  

知道这个结论就比较好做了, 我们可以枚举最后被修改的区域,

这时答案就是将同色边边权当作 000, 异色边边权当作 111 后距离这个点最远的黑色点的距离, 对所有点取最小值即可.

#include <bits/stdc++.h>
using namespace std;
const int MAXN = 55, MAXM = 55;
#define pii pair<int,int>
#define mp make_pair
#define X first
#define Y second
int dx[4] = { 0, 0, 1, -1 };
int dy[4] = { 1, -1, 0, 0 };
char s[MAXN][MAXM];
int dis[MAXN][MAXM], n, m;
deque<pii>q; inline bool chkmin(int &A, int B) { return B < A ? A = B, 1 : 0; }
inline bool chkmax(int &A, int B) { return B > A ? A = B, 1 : 0; } int bfs(int u, int v)
{
memset(dis, -1, sizeof dis);
int ret = -1;
dis[u][v] = 0; q.push_back(mp(u, v));
while(!q.empty())
{
pii tp = q.front(); q.pop_front();
if(s[tp.X][tp.Y] == '1') chkmax(ret, dis[tp.X][tp.Y]);
for(int i = 0; i < 4; i++)
if((u=tp.X+dx[i]) >= 1 && (v=tp.Y+dy[i]) >= 1 && u <= n && v <= m && dis[u][v] == -1)
{
if(s[tp.X][tp.Y] == s[u][v])
dis[u][v] = dis[tp.X][tp.Y], q.push_front(mp(u, v));
else dis[u][v] = dis[tp.X][tp.Y] + 1, q.push_back(mp(u, v));
}
}
return ret;
} int main()
{
scanf("%d%d", &n, &m);
for(int i = 1; i <= n; i++)
scanf("%s", s[i]+1);
int Ans = 0x7f7f7f7f;
for(int i = 1; i <= n; i++)
for(int j = 1; j <= m; j++)
chkmin(Ans, bfs(i, j));
printf("%d\n", Ans+1);
}

模拟赛20181031 雅礼 Wearry 养花 折射 画作的更多相关文章

  1. 模拟赛20181101 雅礼 Wearry 施工 蔬菜 联盟

    % Day2 Solution % Wearry % Stay determined! 施工    记 fif_{i}fi​ 表示考虑前 iii 个建筑, 并且第 iii 个建筑的高度不变的答案, 每 ...

  2. 雅礼集训 Day1 T3 画作 解题报告

    画作 题目描述 小\(\mathrm{G}\)的喜欢作画,尤其喜欢仅使用黑白两色作画. 画作可以抽象成一个\(r\times c\)大小的\(01\)矩阵.现在小\(\mathrm{G}\)构思好了他 ...

  3. NOIP2018 模拟赛(二十二)雅礼NOI

    Preface 这次的题目都是NOI+的题,所以大家的分数都有点惨烈. 依靠T1大力骗分水到Rank2 所以想看正解的话看这里吧 A. 「雅礼NOI2018模拟赛(一) Day1」树 看一眼题目感觉十 ...

  4. 雅礼集训 Day1 T1 养花

    养花 题目描述 小\(C\)在家种了\(n\)盆花,每盆花有一个艳丽度\(a_i\). 在接下来的\(m\)天中,每天早晨他会从一段编号连续的花中选择一盆摆放在客厅, 并在晚上放回. 同时每天有特定的 ...

  5. 雅礼集训1-9day爆零记

    雅礼集训1-9day爆零记 先膜一下虐爆我的JEFF巨佬 Day0 我也不知道我要去干嘛,就不想搞文化科 (文化太辣鸡了.jpg) 听李总说可以去看(羡慕)各路大佬谈笑风声,我就报一个名吧,没想到还真 ...

  6. [NOIP2018模拟赛10.22]咕咕报告

    闲扯 这是篇咕咕了的博客 考场上码完暴力后不知道干什么,然后忽然发现这个T1好像有点像一道雅礼集训时讲过的CF题目 Rest In Shades ,当时那道题还想了挺久不过思路比较妙,于是我就也\(y ...

  7. NOIP模拟赛20161022

    NOIP模拟赛2016-10-22 题目名 东风谷早苗 西行寺幽幽子 琪露诺 上白泽慧音 源文件 robot.cpp/c/pas spring.cpp/c/pas iceroad.cpp/c/pas ...

  8. NOI模拟赛 Day1

    [考完试不想说话系列] 他们都会做呢QAQ 我毛线也不会呢QAQ 悲伤ING 考试问题: 1.感觉不是很清醒,有点困╯﹏╰ 2.为啥总不按照计划来!!! 3.脑洞在哪里 4.把模拟赛当作真正的比赛,紧 ...

  9. NOIP第7场模拟赛题解

    NOIP模拟赛第7场题解: 题解见:http://www.cqoi.net:2012/JudgeOnline/problemset.php?page=13 题号为2221-2224. 1.car 边界 ...

随机推荐

  1. Java基础之(四)HashMap(jdk10)

    JDK1.7以前的HashMap jdk1.7中,当冲突时,在冲突的地址上生成一个链表,将冲突的元素的key,通过equals进行比较,相同即覆盖,不同则添加到链表上,此时如果链表过长,效率就会大大降 ...

  2. 影响MySQL的性能(一)磁盘的选择

    一.磁盘的选择也是影响MySQL的性能的重大因素之一 1.使用传统的机器硬盘读取数据的过程 2.如何选择传统机器硬盘的因素 二.使用RAID增加传统机器硬盘的性能 1.什么是RAID技术 2.常见的R ...

  3. 关于类视图选择继承APIView还是工具视图(ListAPIView、CreateAPIView等等)

    APIView使用方法,直接继承APIView,get或者post请求.方法很简单1.先获取到要操作的数据,然后把数据放到serializer中序列化或者反序列化,最后return返回值(记得.dat ...

  4. AVR单片机教程——按键状态

    好久没更新了,今天开始继续,争取日更. 今天我们来讲按键.开发板的右下角有4个按键,按下会有明显的“咔嗒”声.如何检测按键是否被按下呢?首先要把按键或直接或间接地连接到单片机上.与之前使用的4个LED ...

  5. Linux远程管理命令

    关机\重启 shutdown 选项 时间 参数 -r 重启 例子: shutdown 1分钟后关机 shutdown now 立刻关机 shutdown –r now 立即重启 shutdown 20 ...

  6. js指定日期时间加一天 ,判断指定时间是否为周末

    function dateAdd(startDate) { startDate = new Date(startDate); startDate = +startDate + ***; startDa ...

  7. mybatis 多个中间表查询映射

    最近项目用到中间表,则遇到如何联查映射的问题,之前一直都是一个表头,多个明细或者一对一这样的关系,没遇到这样的问题,所以趁机找了下资料解决了这个问题. 表结构设计如下: 主表: CREATE TABL ...

  8. @PostConstruct使用总结

    https://blog.csdn.net/qq_37636695/article/details/84791468 https://www.jianshu.com/p/aba99a49a459 @P ...

  9. Linux sftp命令

    sftp是Secure File Transfer Protocol的缩写,安全文件传送协议.可以为传输文件提供一种安全的网络的加密方法.sftp 与 ftp 有着几乎一样的语法和功能.SFTP 为 ...

  10. 未检测到.NET CORE SDK 或者 新建项目没有.NET CORE 3.0选择项

    终于解决了 首先先看自己的VS2019版本 由于楼主下载的 .NET CORE SDK 3.0.100-preview8-013656 焕然大悟 原来是版本不符合,需要用vs 2019 preview ...