蓝桥杯-递增三元组(三种解法,二分, 双指针, 前缀和)

给定三个整数数组

A=[A1,A2,…AN],
B=[B1,B2,…BN],
C=[C1,C2,…CN],

请你统计有多少个三元组 (i,j,k)
满足:

1≤i,j,k≤N
Ai<Bj<Ck

输入格式

第一行包含一个整数 N。

第二行包含 N 个整数 A1,A2,…AN。

第三行包含 N 个整数 B1,B2,…BN。

第四行包含 N 个整数 C1,C2,…CN。

输出格式

一个整数表示答案。

数据范围

1≤N≤105,
0≤Ai,Bi,Ci≤105

输入样例:

3
1 1 1
2 2 2
3 3 3

输出样例:

27

题解:

这题有五种写法~

  • 超时的写法: 纯暴力和mini版的暴力
  • ac的写法: 二分或者前缀和或双指针

纯暴力代码👇, 时间复杂度O(n^3)

#include <bits/stdc++.h>
using namespace std;
const int N = 1e5 + 10;
int a[N], b[N], c[N];
int as[N], cs[N];
int cnt[N];int main()
{int n; cin >> n;for (int i = 0; i < n; i ++) cin >> a[i];for (int i = 0; i < n; i ++) cin >> b[i];for (int i = 0; i < n; i ++) cin >> c[i];int res = 0;for (int i = 0; i < n; i ++)for (int j = 0; j < n; j ++ )for (int k = 0; k < n; k ++){if (a[i] < b[j] && b[j] < c[k]) res ++;}cout << res << endl;return 0;
}

这里我们进行一些优化

  • 我们只枚举b数组, 然后看 a 中比当前b[j]小的个数cnt1 和 c中比当前b[j]大的个数cnt2, 那么此时能跟当前b[j]组成满足题目要求的三元组的个数就是 cnt1 * cnt2

mini版的暴力👇, 时间复杂度O(n^2)

#include <bits/stdc++.h>
using namespace std;
const int N = 1e5 + 10;
int a[N], b[N], c[N];
int as[N], cs[N];
int cnt[N];int main()
{int n; cin >> n;for (int i = 0; i < n; i ++) cin >> a[i];for (int i = 0; i < n; i ++) cin >> b[i];for (int i = 0; i < n; i ++) cin >> c[i];int res = 0;for (int j = 0; j < n; j ++){int cnt1 = 0, cnt2 = 0;for (int i = 0; i < n; i ++) if (a[i] < b[j]) cnt1 ++;for (int k = 0; k < n; k ++) if (c[k] > b[j]) cnt2 ++;res += cnt1 * cnt2;}cout << res << endl;return 0;
}

ac写法 -> 二分版本 时间复杂度O(nlogn)
ac代码👇

  • 我们可以用二分优化mini版暴力中查找个数这个步骤
#include <bits/stdc++.h>
using namespace std;
#define int long long  // int 会爆掉
const int N = 1e5 + 10;
int a[N], b[N], c[N];
int as[N], cs[N];
int cnt[N];signed main()
{int n; cin >> n;for (int i = 0; i < n; i ++) cin >> a[i];for (int i = 0; i < n; i ++) cin >> b[i];for (int i = 0; i < n; i ++) cin >> c[i];sort(a, a + n); sort(c, c + n);int res = 0;for (int j = 0; j < n; j ++){int cnt1 = 0, cnt2 = 0;int l = 0, r = n - 1;while (l < r){int mid = l + r + 1 >> 1;if (a[mid] < b[j]) l = mid;  // 找到的是a小于b[j]的最后一个数的下标 (因为找到的是下标, 所有后面乘的时候 要+1)else r = mid - 1;}if (a[l] >= b[j]) cnt1 = -1;  // 不满足条件的 后面+1的时候刚好0else cnt1 = l;l = 0, r = n - 1;while (l < r){int mid = l + r >> 1;if (c[mid] > b[j]) r = mid;  // 找到的是c大于b[j]的第一个数的下标, 所以c中大于 b[j]的个数就是 (n - 下标)else l = mid + 1;}if (c[l] <= b[j]) cnt2 = 0;  // 不满足条件的 等于0else cnt2 = n - l;res += (cnt1 + 1) * cnt2;  // a c有任意一个不存在满足条件的都不能算在答案中}cout << res << endl;return 0;
}

ac写法 -> 前缀和版本 时间复杂度O(n)
ac代码👇

  • 代码中的每个a,b,c之所以都 加1是因为 它们的最小值包含0, 求前缀和下标0要空出来
  • as[t] 表示的是数组a中的元素小于等于t的个数的总和
  • cs[t] 表示的是数组c中的元素小于等于t的个数的总和
#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N = 1e5 + 10;
int a[N], b[N], c[N];
int as[N], cs[N];
int cnt[N];signed main()
{int n; cin >> n;for (int i = 1; i <= n; i ++){cin >> a[i]; a[i] ++;as[a[i]] ++;}for (int i = 1; i <= n; i ++) cin >> b[i], b[i] ++;for (int i = 1; i <= n; i ++) {cin >> c[i]; c[i] ++;cs[c[i]] ++;}for (int i = 1; i <= N; i ++) as[i] += as[i - 1];  // 求前缀和for (int i = 1; i <= N; i ++) cs[i] += cs[i - 1];int res = 0;for (int j = 1; j <= n; j ++)res += as[b[j] - 1] * (cs[N - 1] - cs[b[j]]);  // a 要小于b[j]所以是 as[b[j] - 1]; cs[b[j]]表示c中小于等于b[j]的元素个数, c一共有n个, 大于b[j]的就有 n - cs[b[j]].cout << res << endl;return 0;
}

ac写法 ->双指针版本 时间复杂度O(n)

  • 两个while循环在在整个代码运行过程中最多运行n次, 整个代码的时间复杂度是 O(n)

ac代码👇

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N = 1e5 + 10;
int a[N], b[N], c[N];
int as[N], cs[N];
int cnt[N];signed main()
{int n; cin >> n;for (int i = 1; i <= n; i ++) cin >> a[i];for (int i = 1; i <= n; i ++) cin >> b[i];for (int i = 1; i <= n; i ++) cin >> c[i];sort(a + 1, a + n + 1); sort(b + 1, b + n + 1), sort(c + 1, c + n + 1);int res = 0, ai = 1, ci = 1;for (int j = 1; j <= n; j ++){while (ai <= n && a[ai] < b[j]) ai ++;  while (ci <= n && c[ci] <= b[j]) ci ++;res += (ai - 1) * (n - ci + 1);}cout << res << endl;return 0;
}

最终提交运行时间对比, 前缀和的是最快的
觉得写的不错的话, 点个赞吧~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/10526.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【图像畸变校正】

接上篇文章&#xff1a;【鱼眼&#xff0b;普通相机】相机标定 附代码&#xff1a; 方法一&#xff1a; 使用cv2.undistort """Create May 11, 2024author Wang Jiajun """import cv2 import numpy as npdef correct(img,camera_fileE:/cali…

怎么使用远程桌面传输文件?

微软提供的远程桌面功能是一项强大的工具&#xff0c;可让您在同一网络下远程访问和管理其他计算机。除了远程控制&#xff0c;它还支持文件传输功能&#xff0c;为Windows用户提供了极大的便利。在接下来的内容中&#xff0c;我们将介绍如何使用远程桌面传输文件。 如何从远程…

PADS:生成自交叉平面区域

根据板外形铺铜方法&#xff1a; pads根据板外形铺铜_铺铜如何根据板子形状改变-CSDN博客 根据板外形创建平面区域出现问题&#xff1a; 解决方法&#xff1a;去找结构&#xff0c;让他把出图之前把线合并了

【数据结构】顺序栈

顺序栈 一、相关概念 栈和队列是操作受限的线性表&#xff0c;是限定性的数据结构&#xff1b;栈分为顺序栈和链式栈栈只能在一端进行操作&#xff08;插入、删除&#xff09;栈是限定仅在表尾进行插入或删除操作的线性表&#xff0c;因此&#xff0c;对栈来说&#xff0c;表…

https免费证书获取

获取免费证书的网址&#xff1a; Certbot 1. 进入你的linux系统&#xff0c;先安装snapd&#xff0c; yum install snapd 2. 启动snapd service snapd start 3.安装 Certbot snap install --classic certbot 注意如下出现此错误时&#xff0c;需要先建立snap 软连接后&am…

山东大学软件学院创新项目实训开发日志——第11周

山东大学软件学院创新项目实训开发日志——第11周 项目名称&#xff1a;ModuFusion Visionary&#xff1a;实现跨模态文本与视觉的相关推荐 -------项目目标&#xff1a; 本项目旨在开发一款跨模态交互式应用&#xff0c;用户可以上传图片或视频&#xff0c;并使用文本、点、…

Golang | Leetcode Golang题解之第84题柱状图中最大的矩形

题目&#xff1a; 题解&#xff1a; func largestRectangleArea(heights []int) int {n : len(heights)left, right : make([]int, n), make([]int, n)for i : 0; i < n; i {right[i] n}mono_stack : []int{}for i : 0; i < n; i {for len(mono_stack) > 0 &&am…

SQLite索引名称重复(index already exists)

文章目录 概述报错信息解决方案 概述 SQLite中创建单列索引的方式&#xff0c;跟MySQL类似&#xff1a; CREATE INDEX index_name ON table_name (column_name);但是也有不同的地方&#xff1a; MySQL中索引名称在表内部不重复即可。 SQLite中索引名称在整个库中必须是不重复…

整理项目中经常用到的正则

目录 1、手机号码 2、Email 邮箱 3、QQ 号码 4、非零正整数 5、URL 地址 6、身份证号 项目中难免会经常使用到表单&#xff0c;而表单项校验就需要用到正则&#xff0c; 所以整理总结一下自己项目中使用比较频繁的一些正则校验逻辑。 正则表达式 是由一些具有特殊含义的…

JavaScript之数据类型(3)——object进阶

前言&#xff1a; 利用基础知识来构建对象会发现十分复杂&#xff0c;我们可以结合其他的知识点来为我们object的构建进行优化。 <1>工厂法&#xff1a; 基本格式&#xff1a; function creatObject(属性值1,属性值2,属性值3,...,属性值n) {var 对象名 new Object();对…

在IDEA中使用 Spring Initializr 新建 spring boots 项目

【在IDEA中使用 Spring Initializr 新建 spring boots 项目 - CSDN Apphttp://t.csdnimg.cn/mVs5P Spring Initializr 创建spring boots项目 添加到pom.xml <dependency> <groupId>mysql</groupId> <artifactId>mysql-connec…

Python | Leetcode Python题解之第84题柱状图中最大的矩形

题目&#xff1a; 题解&#xff1a; class Solution:def largestRectangleArea(self, heights: List[int]) -> int:n len(heights)left, right [0] * n, [n] * nmono_stack list()for i in range(n):while mono_stack and heights[mono_stack[-1]] > heights[i]:righ…

代码随想录算法训练营day21 | 513.找树左下角的值、112. 路径总和、106.从中序与后序遍历序列构造二叉树

513.找树左下角的值 迭代法比较简单&#xff0c;层序遍历&#xff0c;找到最下面一层的第一个节点。题目已经说明节点数>1了 class Solution:def findBottomLeftValue(self, root: Optional[TreeNode]) -> int:queue collections.deque()queue.append(root)result ro…

LeetCode题练习与总结:复原IP地址--93

一、题目描述 有效 IP 地址 正好由四个整数&#xff08;每个整数位于 0 到 255 之间组成&#xff0c;且不能含有前导 0&#xff09;&#xff0c;整数之间用 . 分隔。 例如&#xff1a;"0.1.2.201" 和 "192.168.1.1" 是 有效 IP 地址&#xff0c;但是 &qu…

Rust学习笔记(中)

前言 笔记的内容主要参考与《Rust 程序设计语言》&#xff0c;一些也参考了《通过例子学 Rust》和《Rust语言圣经》。 Rust学习笔记分为上中下&#xff0c;其它两个地址在Rust学习笔记&#xff08;上&#xff09;和Rust学习笔记&#xff08;下&#xff09;。 错误处理 pani…

01、什么是ip、协议、端口号知道吗?计算机网络通信的组成是什么?

声明&#xff1a;本教程不收取任何费用&#xff0c;欢迎转载&#xff0c;尊重作者劳动成果&#xff0c;不得用于商业用途&#xff0c;侵权必究&#xff01;&#xff01;&#xff01; 目录 前言 计算机网络 网络ip地址 网络协议 网络端口号 前言 最近有个项目要用到相关文章…

Android — 使用 Runtime 获取日志并保存至 download 目录

万一哪天要用找不到 使用 Runtime 获取日志并保存至 download 目录。 try {final String path Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS).getAbsolutePath() File.separator;ArrayList<String> commandLine new ArrayList&l…

蓝桥杯单片机之模块代码《多样点灯方式》

过往历程 历程1&#xff1a;秒表 历程2&#xff1a;按键显示时钟 历程3&#xff1a;列矩阵按键显示时钟 历程4&#xff1a;行矩阵按键显示时钟 历程5&#xff1a;新DS1302 历程6&#xff1a;小数点精确后两位ds18b20 历程7&#xff1a;35定时器测量频率 历程8&#xff…

大数据Scala教程从入门到精通第六篇:Scala编译结果反编译分析

一&#xff1a;Scala编译结果反编译分析 问题&#xff1a;为什么Scalac之后的生成的class文件有两个&#xff0c;一个带$的&#xff0c;一个不带$的&#xff1f; 不能直接java 执行scala编译的字节码文件。 直接运行的话就会报错&#xff0c;会报一个类没有被找到。 引入类库就…

JavaScript 防抖与节流——以游戏智慧解锁实战奥秘

&#x1f525; 个人主页&#xff1a;空白诗 文章目录 &#x1f3ae; 引言❓ 什么是防抖和节流&#x1f3f9; 防抖(Debounce) - 锁定追击&#xff0c;精确无误&#x1f4cc; 基础概念&#x1f4cc; 适用场景&#x1f4cc; 实战代码&#xff1a;防抖 应用于输入框的实时搜索 &…