四平方和题解(二分习题)

news/2024/10/25 11:18:16/

四平方和

在这里插入图片描述
在这里插入图片描述




暴力做法

Y总暴力做法,蓝桥云里能通过所有数据

总结:暴力也分好坏,下面这份代码就是写的好的暴力
如何写好暴力:1. 按组合枚举 2. 写好循环结束条件,没必要循环那么多次

#include<iostream>
#include<cmath>using namespace std;int n;int main(){std::ios::sync_with_stdio(false);cin.tie(0);cin>>n;for(int a=0;a*a<=n;a++)for(int b=a;a*a+b*b<=n;b++)for(int c=b;a*a+b*b+c*c<=n;c++){int t=(n-a*a-b*b-c*c);int d=sqrt(t);if(d*d==t){cout<<a<<" "<<b<<" "<<c<<" "<<d<<" ";return 0;}}return 0;
}


我自己写的暴力做法,只能过87.5% o(╥﹏╥)o

#include<iostream>
#include<cmath>using namespace std;int n;int main(){std::ios::sync_with_stdio(false);cin.tie(0);cin>>n;for(int a=0;a*a<=n;a++)for(int b=0;b*b<=n;b++)for(int c=0;c*c<=n;c++)for(int d=0;d*d<=n;d++){if(a*a+b*b+c*c+d*d==n){cout<<a<<" "<<b<<" "<<c<<" "<<d;return 0;}}return 0;
}



分析

参考题解

这道题目最重要的思路是空间换时间(这是算法竞赛里一个非常重要的思想!!!)
本来,我们需要枚举 a b c d 四个数字,因为第四个数字可以计算出来,所以至少需要三重循环,即 O(n3),但是这肯定是要超时的,所以就想办法优化循环的次数

  • 重点来了:
    一个比较好的思路是把三重循环拆成两次二重循环。在第一次二重循环中,计算一下c^2+d^2,然后记录下来,在第二次对a和b的循环中可以直接使用,而不需要再次计算。如此一来,时间复杂度就被大大的简化了。

至于如何记录 c^2+d^2 ,则可以考虑使用哈希表,或者数组+二分的做法。不得不说,实在是太巧妙了!!




二分做法

//四平方和
//二分
#include<bits/stdc++.h>using namespace std;const int N = 5e6 + 10;struct Sum{int s, c, d;// 下面这个重载用来解决按字典序输出的问题// 首先根据 c*c+d*d排序,为什么按c*c+d*d排序呢?// 因为我们的c是从小到大枚举,d从c开始枚举;也就是说我们的c和d是按字典序枚举的// 所以c*c+d*d中的c和d一定是符合字典序的// 如果c*c+d*d相同,那么就比较c;如果c*c+d*d和c都相同,那么就比较d;原因应该好理解bool operator<(const Sum &t)const{if(s != t.s) return s < t.s;if(c != t.c) return c < t.c;return d < t.d;}
}record[N * 2];int n;int main()
{cin >> n;int k = 0;// 用来记录结构体数组里的元素个数(也就是说record里有多少个元素)// 预处理(其实就是打表)for(int c = 0; c * c <= n; c++){for(int d = c; c * c + d * d <= n; d ++){record[k++] = {c * c + d * d, c, d};}}// 排序sort(record, record + k);for(int a = 0; a * a <= n; a ++){for(int b = a; a * a + b * b <= n; b++){int t = n - a * a - b * b;// k表示record里的元素个数int l = 0, r = k - 1;while(l < r){int mid = l + r >> 1;if(record[mid].s >= t) r = mid;else l = mid + 1;}if(record[l].s == t){printf("%d %d %d %d\n", a, b, record[l].c, record[l].d);return 0;}}}return 0;
}



哈希做法

使用C++自带的unordered_map过不了,不过模拟哈希表却可以过。

#include<iostream>
#include<cstring>
#include<algorithm>using namespace std;// 开放地址法的N一般为原先的两倍,在本题应该为2*5e6
// 但这里我怕爆内存,只好跟原来一样了;
// 注意N得为质数,5e6的质数为5e6+11
/* 求质数的代码:for(int i=5e6;;i++){bool flag=true;for(int j=2;j*j<=i;j++){if(i%j==0){flag=false;break;}if(flag){cout<<i<<endl;break;}}}*/
const int N=5e6+11,null=0x3f3f3f3f;int h[N];typedef pair<int,int> PII;PII s[N];int n;int find(int x){int k=(x%N+N)%N;while(h[k]!=null&&h[k]!=x){k++;}return k;}int main(){cin>>n;memset(h,null,sizeof h);// 打表for(int c=0;c*c<=n;c++)for(int d=c;c*c+d*d<=n;d++){int k=find(c*c+d*d);if(h[k]==null) {h[k]=c*c+d*d;s[k].first=c,s[k].second=d;}}for(int a=0;a*a<=n;a++)for(int b=a;a*a+b*b<=n;b++){int t=n-a*a-b*b;int k=find(t);if(h[k]!=null){cout<<a<<" "<<b<<" "<<s[k].first<<" "<<s[k].second;return 0;}}return 0;
}

http://www.ppmy.cn/news/34939.html

相关文章

求最大公约数和最小公倍数,附python实现

1、求最大公约数 1&#xff09;采用辗转相除法 例如&#xff0c;需要求63和117的最大公约数 [117]1∗[63]54[63]1∗[54]9[54]6∗[9]0[117] 1*[63]54\\ [63] 1*[54]9\\ [54] 6*[9]0 [117]1∗[63]54[63]1∗[54]9[54]6∗[9]0 可知&#xff0c;最大公约数为9 验证&#xff1a…

图神经网络(GCN)

一、GCN的起源 曾经深度学习一直都是被几大经典模型给统治着&#xff0c;如CNN、RNN等等&#xff0c;它们无论再CV还是NLP领域都取得了优异的效果。 但是对于图结构的数据&#xff0c;无论是CNN还是RNN都无法解决或者效果不好。 &#xff08;1&#xff09;CV中的CNN&#xff1…

9大 HIVE SQL 最频繁被问到的面试题

SQL是用于数据分析和数据处理的最重要的编程语言之一&#xff0c;因此与数据科学相关的工作&#xff08;例如数据分析师、数据科学家和数据工程师&#xff09;在面试时总会问到关于 SQL 的问题。 SQL面试问题旨在评估应聘者的技术和解决问题的能力。因此对于应聘者来说&#x…

每个开发人员都需要掌握的10 个基本 SQL 命令

SQL 是一种非常常见但功能强大的工具&#xff0c;它可以帮助从任何数据库中提取、转换和加载数据。数据查询的本质在于SQL。随着公司和组织发现自己处理的数据量迅速增加&#xff0c;开发人员越来越需要有效地使用数据库来处理这些数据。所以想要暗恋数据领域&#xff0c;SQL是…

TCP和UDP协议的区别?

是否面向连接&#xff1a; TCP 是面向连接的传输&#xff0c;UDP 是面向无连接的传输。 是否是可靠传输&#xff1a;TCP是可靠的传输服务&#xff0c;在传递数据之前&#xff0c;会有三次握手来建立连接&#xff1b;在数据传递时&#xff0c;有确认、窗口、重传、拥塞控制机制…

验证码——vue中后端返回的图片流如何显示

目录 前言 一、p调用接口获取验证码 canvas画布渲染&#xff1f; 二、后端返回图片&#xff08;图片流&#xff09;&#xff0c;前端显示 1.blob 2.arraybuffer 总结 前言 登录界面经常会有验证码&#xff0c;验证码的实现方式也有很多&#xff0c;我目前做过以下两种&…

Android开发-Android UI与布局

01 Android UI 1.1 UI 用户界面(User Interface&#xff0c;简称 UI&#xff0c;亦称使用者界面)是系统和用户之间进行交互和信息交换的媒介&#xff0c;它实现信息的内部形式与人类可以接受形式之间的转换。软件设计可分为两个部分&#xff1a;编码设计与UI设计。 1.2 Andr…

记一次若依后台管理系统渗透

前言 最近客户开始hw前的风险排查&#xff0c;让我们帮他做个渗透测试&#xff0c;只给一个单位名称。通过前期的信息收集&#xff0c;发现了这个站点&#xff1a; 没有验证码&#xff0c;再加上这个图标&#xff0c;吸引了我注意&#xff1a; 从弱口令开始 若依默认口令为ad…