1221. 四平方和--(暴力,二分,哈希表)

题目:

1221. 四平方和 - AcWing题库

 思路1:暴力

暴力枚举
1.枚举顺序为从a到c,依次增大。
2.t=n-a*a-b*b-c*c,求得d=sqrt(t)
3.判断求出的d是否成立。d要求:d*d==t&&d>=c

#include<iostream>
#include<cmath>
using namespace std;
const int N=2300;
int n,a,b,c,d;
int main()
{
    cin>>n;
    for(a=0;a*a<n;a++)
        for(b=a;a*a+b*b<n;b++)
            for(c=b;a*a+b*b+c*c<n;c++){
                int t=n-a*a-b*b-c*c;
                d=sqrt(t);
                if(d*d==t&&d>=c){
                    cout<<a<<" "<<b<<" "<<c<<" "<<d;
                    return 0;
                }
                
            }
}

思路2:二分

1.以空间换取时间的思路。先枚举c,d的情况,将所以可能存入结构体Sum中。再枚举a,b的情况。我们若对Sum.s进行从小打到的排序,就可以用二分寻找满足条件的Sum。

2.在枚举a,b的过程中寻找Sum,此时已经可以确定a,b满足字典序,为保证c,d也为字典序,我们需要对结构体进行自定义排序,不仅仅要按照Sum.s从小到大的顺序排序,同时还要兼顾Sum.c和Sum.d。因此,这里我们需要用到自定义排序或者减号运算符重载。

 自定义排序:
#include<iostream>
#include<algorithm>
using namespace std;
const int N = 9 * 1e6;
int a, b, c, d, n, m;
struct Sum
{
    int s;
    int c;
    int d;
}sum[N];

bool comp(struct Sum sum1,struct Sum sum2)//自定义输出
{
    if (sum1.s != sum2.s)return sum1.s < sum2.s;
    else if (sum1.c != sum2.c)return sum1.c < sum2.c;
    else return sum1.d < sum2.d;
}

int main()
{
    cin >> n;
    //先枚举c,d,将平方和以及c,d存入结构体Sum(以空间换取时间)O(n3)->O(n2)
    for (c = 0; c * c < n; c++)
        for (d = c; c * c + d * d <= n; d++) {//存入结构体
            sum[m].s = c * c + d * d;
            sum[m].c = c;
            sum[m].d = d;
            m++;
        }
    sort(sum, sum + m, comp);//自定义输出(先后按照结构体内s,c,d从小到大顺序排序)

    //枚举a、b,同时二分查找符合条件的c、d
    for(a=0;a*a<n;a++)
        for (b = a; a * a + b * b < n; b++) {
            int L = 0, R = m-1;//对下标二分
            int t = n - a * a - b * b;
            while (L < R) {
                int mid = L + R >> 1;
                if (sum[mid].s >= t)R = mid;
                else L = mid + 1;
            }
            if (sum[L].s == t) {
                cout << a << " " << b << " " << sum[L].c << " " << sum[L].d;
                return 0;
            }
        }
}
减号运算符重载:
#include<iostream>
#include<algorithm>
using namespace std;
const int N = 9 * 1e6;
int a, b, c, d, n, m;
struct Sum
{
    int s;
    int c;
    int d;
    bool operator<(const Sum& t)const//重载减号运算符,实现自定义排序
    {
        //不同情况下减号赋予不同含义,返回值也不一样
        if (s != t.s)return s < t.s;
        else if (c != t.c)return c < t.c;
        else return d < t.d;
    }
}sum[N];


int main()
{
    cin >> n;
    //先枚举c,d,将平方和以及c,d存入结构体Sum(以空间换取时间)O(n3)->O(n2)
    for (c = 0; c * c <= n; c++)
        for (d = c; c * c + d * d <= n; d++) //存入结构体
            sum[m++] = { c * c + d * d,c,d };//结构体与类不同,无需构造函数

    sort(sum, sum + m);//自定义输出(先后按照结构体内s,c,d从小到大顺序排序)

    //枚举a、b,同时二分查找符合条件的c、d
    for (a = 0; a * a < n; a++)
        for (b = a; a * a + b * b < n; b++) {
            int L = 0, R = m - 1;//对下标二分
            int t = n - a * a - b * b;
            while (L < R) {
                int mid = L + R >> 1;
                if (sum[mid].s >= t)R = mid;
                else L = mid + 1;
            }
            if (sum[L].s == t) {
                cout << a << " " << b << " " << sum[L].c << " " << sum[L].d;
                return 0;
            }
        }
}

思路3:哈希表

1.依旧是以空间换时间,变O(n3)为O(n2) 。但存储方式改变。与前面定义结构体不同的是采用了哈希表存储,将c*c+d*d的值t作为哈希表的下标键,将c,d以pair的形式作为键t对于的值且只存储第一个键t对应的值(确保了每个键对应的值都是最小的字典序)。

#include<iostream>
#include<unordered_map>

using namespace std;
#define x first
#define y second

typedef pair <int, int>PII;
unordered_map<int, PII>S;//前为下标后为值(两个)

const int N = 9 * 1e6;
int a, b, c, d, n, m;

int main()
{
    cin >> n;
    //先枚举c,d,将平方和以及c,d存入结构体Sum(以空间换取时间)O(n3)->O(n2)
    for (c = 0; c * c <= n; c++)
        for (d = c; c * c + d * d <= n; d++) { //存入结构体
            int t = c * c + d * d;
            if (S.count(t) == 0)//无键t
                S[t] = { c,d };//键值匹配
        }
            

    //枚举a、b,同时二分查找符合条件的c、d
    for (a = 0; a * a < n; a++)
        for (b = a; a * a + b * b < n; b++) {
            int t = n - a * a - b * b;
            if(S.count(t)){
                cout << a << " " << b << " " << S[t].x << " " << S[t].y;
                return 0;
            }
        }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hz2.0

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值