【差分数组】【图论】【分类讨论】【整除以2】100213按距离统计房屋对数目

这篇具有很好参考价值的文章主要介绍了【差分数组】【图论】【分类讨论】【整除以2】100213按距离统计房屋对数目。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

作者推荐

【动态规划】【数学】【C++算法】18赛车

本文涉及知识点

差分数组 图论 分类讨论 整除以2

LeetCode3017按距离统计房屋对数目

给你三个 正整数 n 、x 和 y 。
在城市中,存在编号从 1 到 n 的房屋,由 n 条街道相连。对所有 1 <= i < n ,都存在一条街道连接编号为 i 的房屋与编号为 i + 1 的房屋。另存在一条街道连接编号为 x 的房屋与编号为 y 的房屋。
对于每个 k(1 <= k <= n),你需要找出所有满足要求的 房屋对 [house1, house2] ,即从 house1 到 house2 需要经过的 最少 街道数为 k 。
返回一个下标从 1 开始且长度为 n 的数组 result ,其中 result[k] 表示所有满足要求的房屋对的数量,即从一个房屋到另一个房屋需要经过的 最少 街道数为 k 。
注意,x 与 y 可以 相等 。
示例 1:
输入:n = 3, x = 1, y = 3
输出:[6,0,0]
解释:让我们检视每个房屋对

  • 对于房屋对 (1, 2),可以直接从房屋 1 到房屋 2。
  • 对于房屋对 (2, 1),可以直接从房屋 2 到房屋 1。
  • 对于房屋对 (1, 3),可以直接从房屋 1 到房屋 3。
  • 对于房屋对 (3, 1),可以直接从房屋 3 到房屋 1。
  • 对于房屋对 (2, 3),可以直接从房屋 2 到房屋 3。
  • 对于房屋对 (3, 2),可以直接从房屋 3 到房屋 2。
    示例 2:
    输入:n = 5, x = 2, y = 4
    输出:[10,8,2,0,0]
    解释:对于每个距离 k ,满足要求的房屋对如下:
  • 对于 k == 1,满足要求的房屋对有 (1, 2), (2, 1), (2, 3), (3, 2), (2, 4), (4, 2), (3, 4), (4, 3), (4, 5), 以及 (5, 4)。
  • 对于 k == 2,满足要求的房屋对有 (1, 3), (3, 1), (1, 4), (4, 1), (2, 5), (5, 2), (3, 5), 以及 (5, 3)。
  • 对于 k == 3,满足要求的房屋对有 (1, 5),以及 (5, 1) 。
  • 对于 k == 4 和 k == 5,不存在满足要求的房屋对。
    示例 3:
    输入:n = 4, x = 1, y = 1
    输出:[6,4,2,0]
    解释:对于每个距离 k ,满足要求的房屋对如下:
  • 对于 k == 1,满足要求的房屋对有 (1, 2), (2, 1), (2, 3), (3, 2), (3, 4), 以及 (4, 3)。
  • 对于 k == 2,满足要求的房屋对有 (1, 3), (3, 1), (2, 4), 以及 (4, 2)。
  • 对于 k == 3,满足要求的房屋对有 (1, 4), 以及 (4, 1)。
  • 对于 k == 4,不存在满足要求的房屋对。

分类讨论

x,y从1开始,x–,y–让其从0开始。如果x> y,交换x和y。
如果xy,直接处理:{(n-1)*2,(n-2)2…}
题目要计算的是:除了起点和终点外,经过的房屋数。
起点i和终点j相同,不统计。
起点和终点互换,需要统计。可以只统计起点<终点,最后将结果
2。
整个路径可以四个节点i,x,y,j表示。
一,各节点最多出现1次,否则有环,将环删除。
二,i,j 必定出现1次,x和y要么都出现1次,要么不出现。
三,第一个节点必定是i,最后一个节点必定是j。
理论上有六种可能,实际上只有五种:
一,i → \rightarrow j。
二,i → \rightarrow x → \rightarrow y → \rightarrow j。
三,i → \rightarrow y → \rightarrow x → \rightarrow j。这中可能不存在:
{ 到 y 之前已经到达终点 , j < = y y → x → y 这段路程是多走的 j > y \begin{cases} 到y之前已经到达终点, & j <= y \\ y \rightarrow x \rightarrow y这段路程是多走的 & j > y \\ \end{cases} {y之前已经到达终点,yxy这段路程是多走的j<=yj>y
四, x == i,y == i 结果是0。 x → \rightarrow y
五,x
i,y不等于j。 x → \rightarrow y → \rightarrow j
六,x!=i,y等于j。 i → \rightarrow x → \rightarrow y

第一种可能经过的房屋数(不包括起点终点):j-i-1
第二种情况,x==y,无需做特殊处理,会被淘汰。x < y时。
abs(x-i)+ abs(j-y) → \rightarrow abs(x-i)+abx(j-y)
{ i − x + j − y i > x , j > y 一 i − x i > x , j = = y 二 i − x + y − j x > i , j < y 三 x − i + j − y i < x , j > y 四 x − i i < x , j = = y 五 x − i + y − j i < x , j < y 六 j − y i = = x , j > y 七 0 i = = x , j = = y 八 y − j i = = x , j < y 九 \begin{cases} i-x + j- y & i > x ,j>y &一\\ i-x & i > x, j == y &二\\ i-x +y-j & x > i ,j < y &三\\ x-i + j- y & i < x,j > y &四\\ x-i & i < x,j == y & 五\\ x-i+y-j & i < x,j < y& 六\\ j - y & i == x ,j > y &七 \\ 0 & i== x ,j == y & 八\\ y - j & i==x,j < y & 九 \\ \end{cases} ix+jyixix+yjxi+jyxixi+yjjy0yji>x,j>yi>x,j==yx>i,j<yi<x,j>yi<x,j==yi<x,j<yi==x,j>yi==x,j==yi==x,j<y

一,二,七,八 可以合并,合并后 i >=x j >= y 。 四五可以合并,i < x ,j >=y 三九也可以合并。合并i >=x ,j<y
{ i − x + j − y i > = x , j > = y 一 i − x + y − j x > = i , j < y 三 x − i + j − y i < x , j > = y 四 x − i + y − j i < x , j < y 六 \begin{cases} i-x + j- y & i >= x ,j>=y &一\\ i-x +y-j & x >= i ,j < y &三\\ x-i + j- y & i < x, j >= y &四\\ x-i+y-j & i < x,j < y& 六\\ \end{cases} ix+jyix+yjxi+jyxi+yji>=x,j>=yx>=i,j<yi<x,j>=yi<x,j<y

我们枚举i,计算j,故x,y,i可以看做常数,可以求出相等的临界值的j。
情况二一:
i-x + j- y <= j - i -1 → \rightarrow 2i -x - y <= -1 → \rightarrow 2i <= x+y-1 → \rightarrow
{ i → x → y → j 2 ∗ i < = x + y − 1 i → j e l s e \begin{cases} i \rightarrow x \rightarrow y \rightarrow j & 2*i <= x+y-1\\ i \rightarrow j & else \end{cases} {ixyjij2i<=x+y1else
和j无关

情况二三:
i-x +y-j <= j - i -1 → \rightarrow 2*(i-j) -x + y <= -1 → \rightarrow -2j <= x - y - 2i -1注意除以-1,大于会变小于 → \rightarrow 2j >= y-x+2i+1 → \rightarrow j >= (y-x)/2 + i +1
{ i → x → y → j j > = ( y − x ) / 2 + i + 1 i → j e l s e \begin{cases} i \rightarrow x \rightarrow y \rightarrow j & j >= (y-x)/2 + i +1 \\ i \rightarrow j & else \end{cases} {ixyjijj>=(yx)/2+i+1else
情况二四:
要想通过x,y 必须 x-i + j- y <= j - i -1 → \rightarrow x-y <= -1 → \rightarrow x <y,恒成立。

情况二六:
要想通过x,y,必须 x-i+y-j <= j - i -1 → \rightarrow x+y-2j <= -1 → \rightarrow -2j <= -x-y-1 注意除以-1,大于会变小于 → \rightarrow 2j >= x+y+1 → \rightarrow j>=(x+y+1+1)/2
{ i → x → y → j > = ( x + y + 1 + 1 ) / 2 i → j e l s e \begin{cases} i \rightarrow x \rightarrow y \rightarrow & j>=(x+y+1+1)/2 \\ i \rightarrow j & else \end{cases} {ixyijj>=(x+y+1+1)/2else

y >= 0 整除2的逆运算

2x >= y ,如果y是偶数 等效与 x >= y/2 。如果y是奇数,等效与 x >= (y+1)/2 。两者可以统一为: x >=(y+1)/2 。
2
x > y 如果y是偶数 等效与 x > y/2 。如果y是奇数,等效与 x > y/2。两者统一为x > y/2。
2x <= y 可以统一为 x <=y/2。
2
x < y 可以统一为:x < ( y+1)/2

代码

核心代码

class Solution {
public:
	vector<long long> countOfPairs(int n, int x, int y) {		
		vector<long long> vRet(n);
		if (x == y)
		{
			vRet.clear();
			for (int i = n - 1; i >= 0; i--)
			{
				vRet.emplace_back(i * 2);
			}
			return vRet;
		}
		if (x > y )
		{
			swap(x, y);
		}
		x--;
		y--;
		int i = 0;
#define Path1(j) (j - i -1 )
		auto Path2 = [&i,&x,&y]( const int j)
		{
			return abs(i - x) + abs(j - y);
		};
		vector<long long> vDiff(n);
		auto Add = [&](int left, int len)
		{
			if (len <= 0)
			{
				return;
			}
			vDiff[left] += 2 ;
			vDiff[left+len] -= 2 ;
		};
		
		for (; i < x; i++)
		{
			//j 在[y,n)
			const int iy = max(i + 1, y);
			if (n - iy > 0)
			{
				Add(Path2(iy), n - iy);
			}	
			//j在(i,y)
			if( y - i -1 > 0 )
			{//i->x->y-j [j0,y)
				const int j0 = (x + y + 2) / 2;	
				Add(Path2(y-1), y - j0);
				//(i,j0)
				Add(0, j0 - i - 1);
			}
		}
		for (; i < n; i++)
		{
			//j在(max(y-1,i),n)
			if (2*i <= x + y-1)
			{//i->x->y-j
				Add(Path2(max(y-1, i) +1), n - max(y-1, i) -1);
			}
			else
			{
				Add(Path1(max(y-1, i) +1), n - max(y-1, i) -1);
			}
			//j在(i,y)
			if (y - i - 1 > 0)
			{
				int j0 = min(y,(2 * i + y - x + 2) / 2);
				j0 = max(j0, i + 1);
				//if (y - j0 >= 0)
				{
					//j在[j0,y) i->x->y-j 
					Add(Path2(y - 1), y - j0);
					//j在(i,j0)
					Add(0, j0 - i - 1);
				}
			}
		}
		
		long long cur=0;
		for (int i = 0; i < n; i++)
		{
			cur += vDiff[i];
			vRet[i] = cur;
		}
		return vRet;
	}
};

测试用例

template<class T>
void Assert(const T& t1, const T& t2)
{
	assert(t1 == t2);
}

template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{
	if (v1.size() != v2.size())
	{
		assert(false);
		return;
	}
	for (int i = 0; i < v1.size(); i++)
	{
		Assert(v1[i], v2[i]);
	}

}

int main()
{	
	int n,  x,  y;

	{
		Solution sln;
		n = 6, x = 1, y = 5;
		auto res = sln.countOfPairs(n, x, y);
		Assert(res, vector<long long>{ 12, 14, 4, 0, 0, 0 });
	}

	{
		Solution sln;
		n = 3, x = 2, y = 2;
		auto res = sln.countOfPairs(n, x, y);
		Assert(res, vector<long long>{4, 2, 0});
	}

	{
		Solution sln;
		n = 4, x = 1, y = 1;
		auto res = sln.countOfPairs(n, x, y);
		Assert(res, vector<long long>{6, 4, 2, 0});
	}
	{
		Solution sln;
		n = 5, x = 2, y = 4;
		auto res = sln.countOfPairs(n, x, y);
		Assert(res, vector<long long>{10, 8, 2, 0, 0});
	}
	
	
	{
		Solution sln;
		n = 3, x = 1, y = 3;		
		auto res = sln.countOfPairs(n,x,y);
		Assert(res, vector<long long>{6, 0, 0});
	}		
	
	{
		Solution sln;
		n = 2, x = 2, y = 2;
		auto res = sln.countOfPairs(n, x, y);
		Assert(res, vector<long long>{2, 0});
	}
	
	

	
}

【差分数组】【图论】【分类讨论】【整除以2】100213按距离统计房屋对数目,# 算法题,图论,算法,c++,LeetCode,差分数组,分类讨论,被2整除

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。

【差分数组】【图论】【分类讨论】【整除以2】100213按距离统计房屋对数目,# 算法题,图论,算法,c++,LeetCode,差分数组,分类讨论,被2整除文章来源地址https://www.toymoban.com/news/detail-814488.html

到了这里,关于【差分数组】【图论】【分类讨论】【整除以2】100213按距离统计房屋对数目的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【图论】树上差分(边差分)

    其实点差分和边差分区别不大。 点差分中,d数组存储的是树上的节点 边差分中,d数组存储的是当前节点到父节点的那条边的差分值。 指定注意的是:边差分中因为根连的父节点是虚点,所以遍历结果时应当忽略!       样例输入: 4 1 1 2 2 3 1 4 3 4 样例输出: 3 我们易知:

    2024年02月14日
    浏览(34)
  • 【图论】差分约束

    x1-x0=9 ; x2-x0=14 ; x3-x0=15 ; x2-x1=10 ; x3-x2=9; 求x3-x0的最大值; 联立式子2和5,可得x3-x0=23;但式子3可得x3-x0=15。所以最大值为15; 但式子多了我们就不好解了,或者说在计算机中怎么解呢? 我们可以想到,不妨把式子转为图的形式。我们令x0--x1的边表示为x1-x0=边权值。 则以上式子

    2024年02月13日
    浏览(35)
  • 常用的统计建模方法——差分分析

    差分,本质上就是下一个数值减去上一个数值,主要是消除一些波动使数据趋于平稳,非平稳序列可通过差分变换转化为平稳序列。 输入: 1个时间序列数据定量变量 输出: 经过指定阶数差分后的序列图 SPSSPRO-免费专业的在线数据分析平台 案例: 基于某杂志1995-2019年的印

    2023年04月08日
    浏览(33)
  • LC-1033. 移动石子直到连续(分类讨论)

    难度中等50 三枚石子放置在数轴上,位置分别为 a , b , c 。 每一回合,你可以从两端之一拿起一枚石子(位置最大或最小),并将其放入两端之间的任一空闲位置。形式上,假设这三枚石子当前分别位于位置 x, y, z 且 x y z 。那么就可以从位置 x 或者是位置 z 拿起一枚石子

    2024年02月02日
    浏览(35)
  • 【数学】差分数组(一维差分)

     差分数组是指对一个 一维数组 进行差分操作得到的 新数组 。差分操作是指计算原数组中相邻元素之间的差异,并将这些差异作为新数组的元素。 具体而言,对于一个长度为n的一维数组x,其差分数组diff的第i个元素可以通过以下公式计算得到: diff[i] = x[i] - x[i-1] 其中,

    2024年02月15日
    浏览(37)
  • 周赛379(排序、分类讨论、记忆化搜索(动态规划))

    简单 给你一个下标从 0 开始的二维整数数组 dimensions 。 对于所有下标 i ( 0 = i dimensions.length ), dimensions[i][0] 表示矩形 i 的长度,而 dimensions[i][1] 表示矩形 i 的宽度。 返回对角线最 长 的矩形的 面积 。如果存在多个对角线长度相同的矩形,返回面积最 大 的矩形的面积。

    2024年01月19日
    浏览(41)
  • 数组分割(2023省蓝桥杯)n种讨论 JAVA

    小蓝有一个长度为 N 的数组 A = [A0, A1,…, AN−1]。现在小蓝想要从 A 对应的数组下标所构成的集合 I = {0, 1, 2, . . . , N − 1} 中找出一个子集 R1,那么 R1在 I 中的补集为 R2。记 S1=∑r∈R1Ar,S2 =∑r∈R2Ar,我们要求 S1 和 S2 均为偶数,请问在这种情况下共有多少种不同的 R1。当 R1 或

    2024年02月11日
    浏览(32)
  • 差分数组详解

    一维差分数组 假设给你一个数组 nums ,先对区间 [a,b] 中每个元素加 3 ,在对区间 [c,d] 每个元素减 5 …… ,这样非常频繁的区间修改,常规的做法可以一个个计算。 频繁对数组的一段区间进行增加或减去同一个值,如果一个个去操作,很明显效率很差,我们可以使用差分数

    2024年02月08日
    浏览(51)
  • 差分数组 & 技巧小记

      如果两个信息“长得很像”,只要保留一个,对另一个,只要保留它们的差异,然后进行微调就行了。 差分数组: 3210,3208,3206,3211,3220,3212…… 3210 【-2】【-2】【5】【9】【-8】…… 适用场景,频繁对某个区间的元素进行增减。 对区间 nums[i…j] 的元素全部加 5: di

    2024年02月02日
    浏览(31)
  • 算法之路-------差分数组

    针对数组中连续的大量数据进行修改的问题,如果我们对每个数据都进行依次修改,对于一些少量的数据的修改(例如:1~100这些的),修改的时候我们发现速度貌似还是很快的,但是一旦修改的连续数组中的数量上万了,那么修改的速率就明显下降了。 所以:针对这样的情

    2024年02月06日
    浏览(50)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包