首页 > 其他分享 >P1102 A-B 数对

P1102 A-B 数对

时间:2024-01-22 23:11:56浏览次数:36  
标签:arr P1102 leq int 复杂度 数对 long ++

1.题目介绍

A-B 数对

题目背景

出题是一件痛苦的事情!

相同的题目看多了也会有审美疲劳,于是我舍弃了大家所熟悉的 A+B Problem,改用 A-B 了哈哈!

题目描述

给出一串正整数数列以及一个正整数 \(C\),要求计算出所有满足 \(A - B = C\) 的数对的个数(不同位置的数字一样的数对算不同的数对)。

输入格式

输入共两行。

第一行,两个正整数 \(N,C\)。

第二行,\(N\) 个正整数,作为要求处理的那串数。

输出格式

一行,表示该串正整数中包含的满足 \(A - B = C\) 的数对的个数。

样例 #1

样例输入 #1

4 1
1 1 2 3

样例输出 #1

3

提示

对于 \(75\%\) 的数据,\(1 \leq N \leq 2000\)。

对于 \(100\%\) 的数据,\(1 \leq N \leq 2 \times 10^5\),\(0 \leq a_i <2^{30}\),\(1 \leq C < 2^{30}\)。

2017/4/29 新添数据两组

2.题解

注意这里由于每组两两组合都可能满足条件,所以这里的结果记录变量 ans 可能是 N^2 大小,这里使用long long 记录!!!

2.1 哈希表

思路

类似 1.两数之和

代码

#include<iostream>
#include<vector>
#include<unordered_map>
using namespace std;
int main(){
	int N, C;
	cin >> N >> C;
	vector<int> arr(N); 
	unordered_map<int,int> mp(N); 
	for(int i = 0; i < N; i++) {
		cin >> arr[i];
		mp[arr[i]]++; 
	}
	long long ans = 0;
	for(int i = 0; i < N; i++){
		if(mp.count(arr[i] - C)) ans += mp[arr[i] - C];
	}
	
	cout <<  ans;
} 

复杂度分析

1.空间复杂度:数组,哈希表 O(N)
2.时间复杂度: 哈希表 O(1), 数组遍历 O(N). 总体O(N)

2.2 双指针法

思路

先排序,之后利用单调性,使用双指针构建结果区域,求得具体长度。
比如像目前数为s[i],我要找到s[i]-C。l为s[l]首个大于等于s[i]-C的数,r为s[r]首个大于s[i]-C的数,由二者共同构建的区域长度即为所需长度。

代码

#include <bits/stdc++.h>
using namespace std;
int main(){
	int N, C;
	cin >> N >> C;
	vector<int> arr(N); 
	for(int i = 0; i < N; i++) {
		cin >> arr[i];
	}
	// sort(arr, arr + N); //这是对于数组的排序,不是对于vector的排序,vector用迭代器即可
	sort(arr.begin(), arr.end()); 
	long long ans = 0;
	int l = 0, r = 0;
	/*1. 这里为何l,r不需要回溯?因为arr[i]-C在不断增大,l和r必定是右移
	*2.由于跳出while循环条件是第一个不满足该条件的情况出现,所以这里l指向区域首部第一个节点,r指向区域尾部节点的下一个节点,长度即r-l*/
	for(int i = 0; i < N; i++){
		while(arr[l] < arr[i] - C && l < N) l++;
		while(arr[r] <= arr[i] - C && r < N) r++;
		if(s[i] - s[l] == c) ans += r - l; //这里的判断并不是非常有必要?
	}
	cout <<  ans;
} 

复杂度分析

1.空间复杂度: 使用了数组 O(n)
2.时间复杂度:由排序决定O(nlogn){后面的数组遍历,由于l和r无需回溯所以就是O(n)}

2.3

标签:arr,P1102,leq,int,复杂度,数对,long,++
From: https://www.cnblogs.com/trmbh12/p/17979465

相关文章

  • STL—函数对象
    函数对象概念1、重载函数调用操作符的类,其对象常称为函数对象2、函数对象使用重载的()时,行为类似函数调用,也叫仿函数本质函数对象(仿函数)是一个类,不是一个函数函数对象的使用特点:1、函数对象在使用时,可以像普通函数那样调用,也可以有参数,可以有返回值2、函数对象超出普通函数的概念,函......
  • 【C++】STL 算法 ② ( foreach 循环中传入 函数对象 / Lambda 表达式处理元素 | forea
    文章目录一、foreach循环中传入函数对象/Lambda表达式处理元素1、foreach循环算法2、foreach循环中传入函数对象处理元素3、foreach循环中传入Lambda表达式处理元素4、Lambda表达式-匿名函数对象/仿函数一、foreach循环中传入函数对象/Lambda表达式处理......
  • STL-函数对象和谓词
    2STL-函数对象2.1函数对象2.1.1函数对象概念概念:重载函数调用操作符的类,其对象常称为函数对象函数对象使用重载的()时,行为类似函数调用,也叫仿函数本质:函数对象(仿函数)是一个类,不是一个函数2.1.2函数对象使用特点:函数对象在使用时,可以像普通函数那样调用,可以有参数,可以有返......
  • python高级之函数对象与闭包函数
    函数对象和闭包函数函数对象1,什么是函数对象?函数对象简单理解就是将函数当变量来使用。如下图所示:定义一个函数可以简单的理解为:func=函数体内存地址函数名+()–>调用函数函数名-->函数对象,函数名不加括号此时的函数名就是函数对象函数用于赋值将函数赋值给某个变......
  • 【python基础之函数对象和闭包】 --- 函数对象与闭包
    title:【python基础之函数对象和闭包】---函数对象与闭包date:2023-12-1119:20:00updated:2023-12-1119:20:00description:cover:https://home.cnblogs.com/u/dream-ze/【一】函数对象函数对象指的是函数可以被当做数据来处理具体可以分为四......
  • 函数/名称空间/作用域/闭包函数/函数对象
    函数【一】函数的定义函数的使用必须遵循先定义,后调用的原则def函数名(参数1,参数2,...):函数体return返回值函数名()(1)空函数函数体为pass代表什么都不做,称之为空函数定义空函数通常是有用的,因为在程序设计的开始,往往是先想好程序都需要完成什么功能,然后把......
  • # [省选联考 2021 B 卷] 数对
    题目描述给定\(n\)个正整数\(a_i\),请你求出有多少个数对\((i,j)\)满足\(1\lei\len\),\(1\lej\len\),\(i\nej\)且\(a_i\)是\(a_j\)的倍数。输入格式第一行,一个整数\(n\),表示数字个数。第二行,\(n\)个整数,表示\(a_i\)。输出格式输出一行,一个整数,表示答......
  • 基于FPGA的图像指数对比度增强算法实现,包括tb测试文件和MATLAB辅助验证
    1.算法运行效果图预览      2.算法运行软件版本Vivado2019.2 matlab2022a 3.算法理论概述3.1图像指数对比度增强概述     图像指数对比度增强是一种常见的图像处理方法,主要是通过改变图像的像素值来增强图像的对比度。具体来说,它通常通过将原始图像......
  • P1102 A-B 数对的三种解法
    1.利用map实现速查,优点是代码简洁,缺点是速度慢,内存大#include<bits/stdc++.h>usingnamespacestd;inta[200005]={0};intmain(){intn,c;scanf("%d%d",&n,&c);map<int,int>maps;for(inti=1;i<=n;i++){scanf("......
  • c++ 为什么引入函数对象?
    C++引入函数对象主要是因为函数对象具有以下优势:函数对象可以有自己的状态:我们可以在类中定义状态变量,这样一个函数对象在多次的调用中可以共享这个状态。但是函数调用没这种优势,除非它使用全局变量来保存状态。函数对象有自己特有的类型,而普通函数无类型可言:这种特性对......