【HDU 5528】Count a * b(推导)

Problem Description
Marry likes to count the number of ways to choose two non-negative integers a and b less than m to make a×b a × b mod m0 m ≠ 0 .

Let’s denote f(m) f ( m ) as the number of ways to choose two non-negative integers a and b less than m to make a×b a × b mod m0 m ≠ 0 .

She has calculated a lot of f(m) for different m, and now she is interested in another function g(n)=m|nf(m) g ( n ) = ∑ m | n f ( m ) . For example, g(6)=f(1)+f(2)+f(3)+f(6)=0+1+4+21=26 g ( 6 ) = f ( 1 ) + f ( 2 ) + f ( 3 ) + f ( 6 ) = 0 + 1 + 4 + 21 = 26 . She needs you to double check the answer.
这里写图片描述
Give you n n . Your task is to find g(n) modulo 264 2 64 .
Input
The first line contains an integer T indicating the total number of test cases. Each test case is a line with a positive integer n.

1T20000 1 ≤ T ≤ 20000
1n109 1 ≤ n ≤ 10 9

Output
For each test case, print one integer s, representing g(n) modulo 264.

Sample Input
2
6
514

Sample Output
26
328194

Source
2015ACM/ICPC亚洲区长春站-重现赛(感谢东北师大)

先推一下 f f :

f(m)=m2i=1mj=1mm|ij=m2i=1mj=1mm(m,i)|i(m,i)j

可以发现 m(m,i)i(m,i) m ( m , i ) 和 i ( m , i ) 已经是互质了,所以要整除m, j j 必须是m(m,i)的倍数,其数量显然是 (m,i) ( m , i ) 个,则:

f(m)=m2i=1m(m,i) f ( m ) = m 2 − ∑ i = 1 m ( m , i )

这里是老套路,去枚举gcd:最终就有:
f(m)=m2d|mdϕ(md) f ( m ) = m 2 − ∑ d | m d ∗ ϕ ( m d )

来看 g g ,所以:

g(n)=m|nm2m|nd|mdϕ(md)

把d拿到最外层:

g(n)=m|nm2d|ndd|m,m|nϕ(md)=g(n)=m|nm2d|ndk|ndϕ(k)=g(n)=m|nm2d|nn g ( n ) = ∑ m | n m 2 − ∑ d | n d ∑ d | m , m | n ϕ ( m d ) = g ( n ) = ∑ m | n m 2 − ∑ d | n d ∑ k | n d ϕ ( k ) = g ( n ) = ∑ m | n m 2 − ∑ d | n n

到这里就是去枚举n的所以因子了,用唯一分解来做,然后搜索枚举因子即可。
代码:

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define ull  unsigned long long
#define ll long long
#define ul unsigned int
#define maxn 40000
using namespace std;
bool isP[maxn];
int prime[maxn];
int cnt;
void init()
{
    for(int i=2;i<maxn;i++)
    {
        if(!isP[i])prime[cnt++]=i;
        for(int j=0;j<cnt&&i*prime[j]<maxn;j++)
        {
            isP[i*prime[j]]=true;
            if(i%prime[j]==0)break;
        }
    }
}
int fac[50];
int e[50];
int k;
void getFac(int x)
{
    k=0;
    for(int i=0;prime[i]*prime[i]<=x;i++)
    {
        if(x%prime[i]==0)
        {
            fac[k]=prime[i];
            e[k]=0;
            while(x%prime[i]==0)
            {
                x/=prime[i];
                ++e[k];
            }
            ++k;
        }
    }
    if(x>1){fac[k]=x;e[k++]=1;}
}
ull ans=0;
void dfs(int cur,ull temp)
{
    if(cur==k)
    {
        ans+=temp*temp;
        return;
    }
    dfs(cur+1,temp);
    ull now=temp;
    for(int i=1;i<=e[cur];i++)
    {
        now*=fac[cur];
        dfs(cur+1,now);
    }
}
ull mod=0;
int main()
{
    mod=~mod;
    init();
    int t;
    cin>>t;
    int n;
    while(t--)
    {
        scanf("%d",&n);
        int total=1;
        getFac(n);
        ans=0;
        dfs(0,1);
        for(int i=0;i<k;i++)total*=e[i]+1;
        ull _n=n;
        _n*=total;
        _n=mod-_n+1;
        cout<<ans+_n<<endl;
    }
    return 0;
}
HDU 2034 是一道经典的 A-B Problem 题目,通常涉及简单的数学运算或者字符串处理逻辑。以下是对此类问题的分析以及可能的解决方法。 ### HDU 2034 的题目概述 该题目要求计算两个数之间的差值 \(A - B\) 并输出结果。需要注意的是,输入数据可能存在多种情况,因此程序需要能够适应同的边界条件和特殊情况[^1]。 #### 输入描述 - 多组测试数据。 - 每组测试数据包含两行,分别表示整数 \(A\) 和 \(B\)。 #### 输出描述 对于每组测试数据,输出一行表示 \(A - B\) 的结果。 --- ### 解决方案 此类问题的核心在于正确读取多组输入并执行减法操作。以下是实现此功能的一种常见方式: ```python while True: try: a = int(input()) b = int(input()) print(a - b) except EOFError: break ``` 上述代码片段通过循环断接收输入直到遇到文件结束符 (EOF),适用于批量处理多组测试数据的情况。 --- ### 特殊考虑事项 尽管基本思路简单明了,在实际编码过程中仍需注意以下几点: 1. **大数值支持**:如果题目中的 \(A\) 或 \(B\) 可能非常大,则应选用可以容纳高精度的数据类型来存储这些变量。 2. **负数处理**:当 \(B>A\) 导致结果为负时,确保程序会因符号错误而失效。 3. **异常捕获**:为了防止运行期间由于非法字符或其他意外状况引发崩溃,建议加入必要的错误检测机制。 --- ### 示例解释 假设给定如下样例输入: ``` 5 3 7 2 ``` 按照以上算法流程依次完成各步操作后得到的结果应当分别为 `2` 和 `5`。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值