0%

题目大意

题目描述

给定平面上 nn 个圆形区域,第 ii 个圆的圆心坐标为 (pi,qi)(p_i, q_i),半径为 rir_i(保证 ri30r_i \le 30)。

给定 mm 次询问,每次询问给出一个四边平行于坐标轴的矩形区域,指定了其左下角顶点坐标 (xi,1,yi,1)(x_{i,1}, y_{i,1}) 和右上角顶点坐标 (xi,2,yi,2)(x_{i,2}, y_{i,2})。对于每次询问,需要计算出在给定的 nn 个圆中,有多少个圆与该矩形区域存在至少一个公共点。

数据范围

测试用例数量 tt 满足 1t10001 \le t \le 1000
每次测试中的圆的个数 nn 满足 1n1.5×1061 \le n \le 1.5 \times 10^6,询问的个数 mm 满足 1m1041 \le m \le 10^4
对于全部测试用例,保证 n3×106\sum n \le 3 \times 10^6m2×104\sum m \le 2 \times 10^4
坐标与半径范围:1pi,qi1091 \le p_i, q_i \le 10^91ri301 \le r_i \le 301xi,1xi,21091 \le x_{i,1} \le x_{i,2} \le 10^91yi,1yi,21091 \le y_{i,1} \le y_{i,2} \le 10^9

样例输入

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
2
4 1
1 1 1
7 1 2
5 3 2
1 2 1
2 2 6 4
8 3
5 6 4
7 7 4
3 3 2
1 8 5
10 1 2
1 8 1
7 2 2
5 4 5
9 1 10 5
3 1 6 3
5 1 9 5

样例输出

1
2
3
4
3
4
4
7

样例解释

在第一组测试数据中,共有 44 个圆和 11 次询问。询问给定的矩形左下角坐标为 (2,2)(2, 2),右上角坐标为 (6,4)(6, 4)。经过判断,该矩形与输入的第 22 个圆、第 33 个圆以及第 44 个圆(总计三个圆形)存在公共点,因此该次询问的答案为 33。第二组测试数据同理,分别输出对应矩形内包含公共点的圆的数量。

image

思路讲解

这道题目应该是一道扫描线+树状数组,应该是不需要用到这个 K-D 树这种比较高级的东西的。

呃,那我们怎么做这道题目呢?首先就是用这个数据结构,快速求解出大矩形中的这个点,所谓的大矩形,就是矩形和圆的 minkowski 和(这个 minkowski 和是一个圆角矩形),当然,圆角矩形让我们用数据结构处理实在是,我们忽略其圆角,得到的就是所谓的“大矩形”(即图中黑色框框住的部分)。

小羊-2025-12-04-M. Ahmad’s Dish(圆和正多边形的这个 minkowski 和)(minkowski 和的交换不变性)

image

2025 United Kingdom and Ireland Programming Contest (UKIEPC 2025) 2025 英国 ICPC——B. Brickwork(判断砖块是否重叠)(扫描线的本质就是遍历一个出一个)(维护一个当前生效的集合)(把二维的问题变为一维的问题来做)

okay,我们上面的这个对扫描线总结的比较好。扫描线的重点就是维护一个当前生效的这个集合。按照一顺序排序后,那么我们其实就是要维护一个目前生效的圆的集合

首先呢,就是这道题目,因为半径比较少,我们不妨把这个不同半径的圆分开处理

然后因为我们目前把不同的半径分开处理,我们目前可以假设,所有的圆的这个半径都是相同的。我们就是像上面一样,把这个矩形扩大这个半径(👆想上面那个黑色的矩形一样)。

然后我们把矩形先按照 x 的右端点排序,再按照 x 的左端点排序。不过,我们发现,无论你怎么排序这个矩形和这个点(也就是这个圆心嘛),都很难维护一个严格生效的这个点的集合,主要也是矩形的是一个这个区间范围,无论怎么排都比较麻烦。

不过我们采用差分思想?其实就是如下方法:

左查询:在 X=Xmin1X = X_{min} - 1 的时刻,查询树状数组中 [Ymin,Ymax][Y_{min}, Y_{max}] 区间的点数,记为 AA,从答案中减去 AA。(记录权重为 1-1

右查询:在 X=XmaxX = X_{max} 的时刻,查询树状数组中 [Ymin,Ymax][Y_{min}, Y_{max}] 区间的点数,记为 BB,加到答案中。(记录权重为 +1+1

采用如上👆方法,我们只需要维护一个 [INF,idx][-INF,idx] 的生效点集合就可以了,可以说是非常轻松惬意了,我们也不需要删除。

然后我们要进行这个几何点的这个扣除:

image

只要像上面👆一样,不断的枚举这个 dx,就可以排除掉所有的这个不合法的点。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115

struct Solve {
ll N, M;
Compress_<ll> li_y;
vector<vector<X_y> > r_xy_ls;
vector<Event> events_backup;
vector<int> Ans;

vector<Event> gen_event(ll r) {
// vector<Event> tmp_events;
vector<Event> events;
// tmp_events.reserve(SZ(r_xy_ls[r]));
for (auto [x,y]: r_xy_ls[r]) {
events.push_back({x, y, Insert});
}
for (auto e: events_backup) {
e.y1 -= r;
e.y2 += r;
if (e.type == Left) {
e.x -= r;
} else if (e.type == Right) {
e.x += r;
}
events.push_back(e);
}
sort(all(events));
return events;
}

void solve_single_r(ll r) {
BIT tr(li_y.size());
sort(all(r_xy_ls[r]));
auto events = gen_event(r);
for (auto &e: events) {
if (e.type == Insert) {
tr.add(li_y.get_i(e.y1), 1);
} else if (e.type == Left) {
Ans[e.id] -= tr.query(li_y.get_i(e.y1), li_y.get_i(e.y2));
Ans[e.id] -= redundant_circle_cnt(r,e.x+r,e.y1+r,-1,-1);
Ans[e.id] -= redundant_circle_cnt(r,e.x+r,e.y2-r,-1,1);
} else {
Ans[e.id] += tr.query(li_y.get_i(e.y1), li_y.get_i(e.y2));
Ans[e.id] -= redundant_circle_cnt(r,e.x-r,e.y1+r,1,-1);
Ans[e.id] -= redundant_circle_cnt(r,e.x-r,e.y2-r,1,1);
}
}
}

// 查找以这个 x 为首的,在范围 [ly,ry] 之间的这个点的数量
ll query_r_x_l_r_cnt(ll r, ll x, ll ly, ll ry) {
if (ly > ry) {
return 0;
}
auto &xy_ls = r_xy_ls[r];
auto itl = lower_bound(all(xy_ls), X_y{x, ly});

if (itl == xy_ls.end()) {
return 0;
}
if (itl->x != x) {
return 0;
}
if (itl->y > ry) {
return 0;
}
auto itr = upper_bound(all(xy_ls), X_y{x, ry});
ll num = itr - itl;
return num;
}

// 计算边缘的,可能被多算的这个点。
ll redundant_circle_cnt(ll r, ll x, ll y, ll dirx, ll diry) {
ll res = 0;
for (int dx = 1; dx <= r; ++dx) {
ll tox = x + dirx * dx;
ll disy = sqrtl(r * r - dx * dx);
ll toy = y + diry * disy;
ll ly = min(toy + diry, y + diry * r);
ll ry=max(toy + diry, y + diry * r);
res += query_r_x_l_r_cnt(r, tox, ly, ry);
}
return res;
}

Solve() : r_xy_ls(32) {
cin >> N >> M;
li_y.reserve(N + M);
// for (int i = 1; i <= 30; ++i) r_xy_ls[i].reserve(N);
for (int i = 1; i <= N; ++i) {
ll x, y, d;
cin >> x >> y >> d;
r_xy_ls[d].push_back({x, y});
li_y.add(y);
}
events_backup.reserve(N + 30 * M);
Ans.resize(M);
for (int i = 0; i < M; ++i) {
ll x1, y1, x2, y2;
cin >> x1 >> y1 >> x2 >> y2;
for (int r = 0; r <= 30; ++r) {
li_y.add(y1 - r, y2 + r);
}
events_backup.push_back({x1, y1, Left, i, y2});
events_backup.push_back({x2, y1, Right, i, y2});
}
sort(all(events_backup));
li_y.init();
for (int i = 1; i <= 30; ++i) {
solve_single_r(i);
}
for (int i = 0; i < M; ++i) {
cout << Ans[i] << "\n";
}
}
};

AC代码

AC
https://acm.hdu.edu.cn/contest/view-code?cid=1198&rid=19570

心路历程(WA,TLE,MLE……)

也是成功一次 AC 了,当然,在写一些模块的时候,ai 也帮助我进行了一些东西。

题目大意

题目描述
给定一个长度为 nn、仅由小写字母组成的字符串 SS 以及一个非负整数 kk
需要计算 SS 的本质不同的非空 kk-松散子序列的数量,并将结果对 998244353998244353 取模。

相关定义

  • kk-松散子序列:如果子序列 TT 在原字符串 SS 中对应的下标序列为 pos1,pos2,,posmpos_1, pos_2, \dots, pos_m,且满足对于所有的 i[1,m1]i \in [1, m-1],都有 posi+1posi>kpos_{i+1} - pos_i > k,那么 TT 就是一个 kk-松散子序列。即子序列中原串相邻提取字符的原始位置距离必须严格大于 kk

  • 本质不同:两个子序列被认为是不同的,当且仅当它们的长度不同,或者它们在某个对应位置上的字符不相同(即只对比提取出的字符构成的字符串本身是否一致)。

输入格式
第一行包含一个整数 TT1T1061 \le T \le 10^6),表示测试用例数。
对于每组测试用例:
第一行包含两个整数 n,kn, k1n106,0kn1 \le n \le 10^6, 0 \le k \le n)。
第二行包含一个长度为 nn 的字符串 SS,保证仅包含小写字母。
保证所有测试用例的 n106\sum n \le 10^6

输出格式
对于每组测试用例,输出一行一个整数,表示本质不同的非空 kk-松散子序列的数量对 998244353998244353 取模的结果。

样例

1
2
3
4
5
6
7
8
9
10
11
12
13
输入:
3
4 1
aabb
5 2
abcab
7 3
abcdece

输出:
3
6
10

样例解释

  • 对于第一个测试用例(S=aabbS = \text{aabb}k=1k = 1),下标位置差必须严格大于 11。满足条件的本质不同的子序列有 abab

  • 对于第二个测试用例(S=abcabS = \text{abcab}k=2k = 2),下标位置差必须严格大于 22。满足条件的本质不同的子序列有 abcaaabbb

思路讲解

呃,这个我们需要想办法做到求本质不同的这个所有子序列

我们转移的时候,就是应该以 s[i] 开头的串,在我们所有的这个的计数之中,应该以 s[i] 开头的串只应该真的以 i 开头的时候被计算,因此后面的应该被减去。(当然,我们是从后往前遍历)

具体而言,这个转移式子是:

next[i]存在: dp[i]2×dp[i+1](dp[next[i]+1]+1)+1dp[i]2×dp[i+1]dp[next[i]+1]next[i]不存在: dp[i]2×dp[i+1]+1next[i] 存在:\ dp[i]\gets 2\times dp[i+1]-(dp[next[i]+1]+1)+1 \\ dp[i]\gets 2\times dp[i+1]-dp[next[i]+1] \\ next[i] 不存在:\ dp[i]\gets 2\times dp[i+1]+1

注意,为什么是减去 dp[next[i]+1]+1dp[next[i]+1]+1 呢?因为其实这个才是含有 next[i]next[i] 的子序列的所有数量。

当然,这个是求本质不同的这个子序列的时候,我们要计算 k-松散子序列。

计算 k-松散子序列:

next[i]不存在: dp[i]2×dp[i+k+1]+1next[i]存在: dp[i]2×dp[i+k+1]+1(dp[next[i]+k+1]+1)dp[i]2×dp[i+k+1]dp[next[i]+k+1]next[i] 不存在:\ dp[i]\gets 2\times dp[i+k+1]+1 \\ next[i] 存在:\ dp[i]\gets 2\times dp[i+k+1]+1-(dp[next[i]+k+1]+1) \\ dp[i]\gets 2\times dp[i+k+1]-dp[next[i]+k+1]

AC代码

心路历程(WA,TLE,MLE……)

杭电 OJ 暂不支持调整栈空间大小。如果使用 G++ 提交代码,并且代码中含有递归,递归层数较深,会返回 Runtime Error (STACK_OVERFLOW)。添加如下代码可以开大栈空间:

1
2
3
4
5
6
7
int main() {
int size(256<<20); // 256M
__asm__ ( "movq %0, %%rsp\n"::"r"((char*)malloc(size)+size));
// 关同步流代码
// 主程序代码
exit(0); // 一定要添加这一条,不然会返回 Runtime Error (ACCESS_VIOLATION)
}

题目大意

题目描述

给定二维平面第一象限上的 nn 个弹幕发射器。第 ii 个发射器位于坐标 (xi,yi)(x_i, y_i),并向四个方向之一发射一条射线(激光):

  • di=0d_i = 0:向上发射

  • di=1d_i = 1:向右发射

  • di=2d_i = 2:向下发射

  • di=3d_i = 3:向左发射

射线的照明路径包含发射器本身所在的位置。题目保证任意两个发射器不会同时在对方的照明路径上。

现在可以在平面上放置障碍物(允许放置在发射器所在的坐标上),只要障碍物位于某条射线上,就能阻挡该射线的照射。求最少需要放置多少个障碍物,才能使得所有 nn 个发射器的照射路径上都至少包含一个障碍物。

输入格式

第一行包含一个整数 tt1t1001 \le t \le 100),表示测试用例的数量。

每个测试用例第一行包含一个整数 nn1n3001 \le n \le 300),表示发射器的数量。
接下来 nn 行,每行包含三个整数 xi,yi,dix_i, y_i, d_i1xi,yi1091 \le x_i, y_i \le 10^9di{0,1,2,3}d_i \in \{0, 1, 2, 3\}),表示发射器的坐标与发射方向。

保证所有测试用例中 nn 的总和不超过 30003000

输出格式

对于每个测试用例,输出一行一个整数,表示最少需要放置的障碍物数量。

样例输入

1
2
3
4
5
6
7
8
9
10
2
6
5 5 0
3 7 1
9 9 2
9 1 3
5 2 0
1 20 3
1
1 1 0

样例输出

1
2
3
1

样例解释

在第一个测试用例中,有 66 个发射器。一种最优方案是放置 33 个障碍物,具体位置如下:

  1. 在坐标 (5,7)(5, 7) 处放置一个障碍物:该点覆盖了第 11 束(从 (5,5)(5,5) 向上)、第 22 束(从 (3,7)(3,7) 向右)以及第 55 束(从 (5,2)(5,2) 向上)激光。

  2. 在坐标 (9,1)(9, 1) 处放置一个障碍物:该点覆盖了第 33 束(从 (9,9)(9,9) 向下)以及第 44 束(从 (9,1)(9,1) 向左)激光。

  3. 在坐标 (2025,20)(-2025, 20) 处放置一个障碍物:该点覆盖了第 66 束(从 (1,20)(1,20) 向左)激光。

该方案共使用了 33 个障碍物,且可以证明不存在比 33 更少的放置方案。

思路讲解

一种错误的图论建模方式是把激光当作左边的点,把这个障碍物当成右边的点,但是这样子建图是做不出来这道题目的,因为我们不是要让激光和障碍物一一匹配,这个非常容易就能做到。

P3386 【模板】二分图最大匹配

那么要做这道题目,最重要的就是我们要把障碍物看成连边,左边的点指的是这个方向为上下的点,右边的点指的是这个方向为左右的点

image

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
void Solve() {
ll N;
cin >> N;
vector<X_y_d> xyd_ls(N);
for (int i = 0; i < N; ++i) {
cin >> xyd_ls[i].x >> xyd_ls[i].y >> xyd_ls[i].d;
}
xyd_ls = unique_xyd_ls(xyd_ls);
auto g = build_graph(xyd_ls);
Bi_graph_matcher matcher(g,SZ(xyd_ls));
ll mx_match = matcher.mx_match();
ll ans = (SZ(xyd_ls) - 2 * mx_match) + mx_match;
cout << ans << "\n";
}

AC代码

AC
https://acm.hdu.edu.cn/contest/view-code?cid=1198&rid=18859

心路历程(WA,TLE,MLE……)

最好显式忽略这个其他情况:

image

使用 offset 进行 swap,最后也要 swap 回来。

image

将哨兵值设置为 -1,可以避免 0-based 造成问题。

image

题目大意

题目描述

给定一个数组 aa,定义 f(a)f(a) 为将 aa 划分成一个或多个连续子数组的方案数,需满足以下条件:

  1. 每个元素恰好属于一个子数组。

  2. 所有子数组的元素和都相等。

例如,对于 a=[1,1]a=[1,1]f(a)=2f(a)=2,因为有两种合法的划分方案:

  • [1,1][1,1],此时唯一的子数组和为 22

  • [1]+[1][1]+[1],此时两个子数组和均为 11

给定两个整数 xxyy。你需要构造一个长度为 x+yx+y 的数组 aa,其中包含 xx11yy1-1
你需要求出在所有合法的数组 aa 中,f(a)f(a)最小值,并将该最小值对 676767677676767677 取模后输出。同时,你需要输出一个能达到该最小值的数组构造方案。

注意:你需要最小化的是 f(a)f(a) 本身,然后将这个最小值对 676767677676767677 取模,而不是去最小化 f(a)mod676767677f(a) \bmod 676767677 的结果。

输入格式

第一行包含一个整数 tt (1t1041 \le t \le 10^4),表示测试用例的数量。

接下来每个测试用例包含一行,包含两个整数 xxyy (0x,y21050 \le x, y \le 2 \cdot 10^5),保证 x+y1x+y \ge 1

保证所有测试用例中 xx 的总和与 yy 的总和均不超过 21052 \cdot 10^5

输出格式

对于每个测试用例,输出两行:
第一行输出 f(a)f(a) 的最小值对 676767677676767677 取模的结果。
第二行输出一个达到该最小值的数组 aa 的元素(用空格分隔)。

样例输入

1
2
3
4
5
4
2 0
1 1
6 7
1 3

样例输出

1
2
3
4
5
6
7
8
2
1 1
1
1 -1
1
-1 1 -1 1 -1 1 -1 1 -1 1 -1 1 -1
2
-1 -1 -1 1

样例解释

在第一个测试用例中,x=2x=2y=0y=0。唯一可能的数组是 a=[1,1]a=[1,1],正如题目描述中所解释的,此时 f(a)=2f(a)=2

在第二个测试用例中,x=1x=1y=1y=1。一个能使 f(a)f(a) 最小化的合法数组是 a=[1,1]a=[1,-1]。此时 f(a)=1f(a)=1,因为唯一能使所有子数组和相等的划分方式就是不进行任何分割(即 [[1,1]][[1,-1]],子数组和为 00)。

思路讲解

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
void Solve() {
ll X, Y;
cin >> X >> Y;
vector<ll> A(X + Y + 2);
for (int i = 1; i <= X; ++i) {
// cout << 1 << " ";
A[i] = 1;
}
for (int i = 1; i <= Y; ++i) {
// cout << -1 << " ";
A[i + X] = -1;
}
ll s = llabs(X - Y);
ll ans = 1;
// 这里要采用求约数个数的方法
if (s != 0) {
ans = 0;
for (ll d = 1; d * d <= s; ++d) {
if (s % d == 0) {
++ans;
if (d * d != s) ++ans;
}
}
}
cout << ans << "\n";
for (int i = 1; i <= X + Y; ++i) {
cout << A[i] << " ";
}
cout << "\n";
// mod
}

AC代码

AC

https://codeforces.com/contest/2211/submission/368623804

心路历程(WA,TLE,MLE……)