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

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

作者推荐

【动态规划】【数学】【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});
  }
  
  
  
}


相关文章
|
存储 缓存 安全
ConcurrentHashMap:使用方法和底层原理详解
ConcurrentHashMap:使用方法和底层原理详解
473 1
|
4月前
|
存储 人工智能 小程序
微信掌上医院是如何实现的?智慧掌上医院核心实现原理及关键技术解析
微信掌上医院(智慧医院)小程序源码,基于uni-app与Vue.js开发,集成预约挂号、在线支付、报告查询等功能,支持与医院HIS、医保系统对接,采用微服务架构与多重安全机制。
181 2
|
JavaScript 前端开发
JavaScript中的原型 保姆级文章一文搞懂
本文详细解析了JavaScript中的原型概念,从构造函数、原型对象、`__proto__`属性、`constructor`属性到原型链,层层递进地解释了JavaScript如何通过原型实现继承机制。适合初学者深入理解JS面向对象编程的核心原理。
215 1
JavaScript中的原型 保姆级文章一文搞懂
|
8月前
|
机器学习/深度学习 人工智能 自然语言处理
AI训练师入行指南(三):机器学习算法和模型架构选择
从淘金到雕琢,将原始数据炼成智能珠宝!本文带您走进数字珠宝工坊,用算法工具打磨数据金砂。从基础的经典算法到精密的深度学习模型,结合电商、医疗、金融等场景实战,手把手教您选择合适工具,打造价值连城的智能应用。掌握AutoML改装套件与模型蒸馏术,让复杂问题迎刃而解。握紧算法刻刀,为数字世界雕刻文明!
303 6
|
8月前
|
人工智能 自然语言处理 数据处理
《DeepSeek轻量级模型蒸馏技术:知识迁移损失补偿策略全解析》
在人工智能领域,大语言模型虽强大但部署困难,尤其在资源受限设备上。DeepSeek的轻量级模型蒸馏技术通过知识迁移损失补偿策略,有效解决了这一难题。该技术将大型教师模型的知识传递给小型学生模型,通过输出分布、中间特征和梯度匹配等方式最小化性能损失,实现模型轻量化。此外,动态自适应策略如温度调节和课程蒸馏进一步提升了蒸馏效果。实际应用中,轻量级模型在自然语言处理任务中表现出色,大幅降低了计算资源需求,为更广泛的应用场景提供了可能。
381 0
|
11月前
体育赛事直播系统怎么开发(足球篮球电竞)
一、通过购买现成源码(如“熊猫比分”)并进行二次开发,是一种快速启动体育直播项目的方式,适合需求变动不大、预算有限的情况。但需注意,大规模改动可能增加工作量和潜在错误。 二、全新定制开发体育直播系统,虽成本高、周期长(约2周),但能完全根据客户需求设计,提供更高的灵活性和定制化程度。开发前需详细规划功能与界面布局,确保最终产品符合预期。
体育赛事直播系统怎么开发(足球篮球电竞)
|
10月前
|
人工智能 安全 Java
微服务引擎 MSE:打造通用的企业级微服务架构
微服务引擎MSE致力于打造通用的企业级微服务架构,涵盖四大核心内容:微服务技术趋势与挑战、MSE应对方案、拥抱开源及最佳实践。MSE通过流量入口、内部流量管理、服务治理等模块,提供高可用、跨语言支持和性能优化。此外,MSE坚持开放,推动云原生与AI融合,助力企业实现无缝迁移和高效运维。
428 1
|
存储 安全 IDE
Linux网络服务——PXE网络批量装机+Kickstart无人值守安装(上)
1 系统安装介绍 1.1 系统装机的三种引导方式 硬盘 光驱 网络
568 0
|
机器学习/深度学习
R语言模型评估:深入理解混淆矩阵与ROC曲线
【9月更文挑战第2天】混淆矩阵和ROC曲线是评估分类模型性能的两种重要工具。混淆矩阵提供了模型在不同类别上的详细表现,而ROC曲线则通过综合考虑真正率和假正率来全面评估模型的分类能力。在R语言中,利用`caret`和`pROC`等包可以方便地实现这两种评估方法,从而帮助我们更好地理解和选择最适合当前任务的模型。
|
XML 监控 druid
SpringBoot整合Druid数据源并配置监控
SpringBoot整合Druid数据源并配置监控
1704 1