随后,小明对每一个硬币分别进行一次 Q 操作。
对第x行第y列的硬币进行 Q 操作的定义:将所有第 i*x 行,第 j*y 列的硬币进行翻转。
其中i和j为任意使操作可行的正整数,行号和列号都是从1开始。
当小明对所有硬币都进行了一次 Q 操作后,他发现了一个奇迹——所有硬币均为正面朝上。
小明想知道最开始有多少枚硬币是反面朝上的。于是,他向他的好朋友小M寻求帮助。
聪明的小M告诉小明,只需要对所有硬币再进行一次Q操作,即可恢复到最开始的状态。然而小明很懒,不愿意照做。于是小明希望你给出他更好的方法。帮他计算出答案。
对于20%的数据,n、m <= 10^7;
对于40%的数据,n、m <= 10^15;
对于10%的数据,n、m <= 10^1000(10的1000次方)。
这道题是Java编程者的福利,用java做比用c语言做简单很多,
java中有BigInter大数,可以直接计算长的数字
本来以为还是一道模拟题,聪(bai)明(chi)的小M也是too young too simple,其实,看到题目的数据范围我们应该已经感觉到,模拟绝对没戏,接下来我们仔细分析一下这道题的解法。
1.很容易得出,如果一枚硬币被翻了奇数次,那么它原来的状态肯定是反面朝上,所以,我们要找的就是被翻了奇数次的硬币
2. Q 操作的定义:将所有第 i*x 行,第 j*y 列的硬币进行翻转。正向看可能不好想,那么我们反向看一下,对于一个横坐标为N的硬币,在翻哪些硬币(横坐标x)的时候会翻到它呢?其实就是这个数N所有的约数,比如横坐标为4的硬币,那么,在翻横坐标为1,2,4的硬币时都会翻到它,纵坐标的情况是一样的。
3.对于一个硬币,我们必须同时考虑其横坐标x和纵坐标y,假如横坐标被翻了a次,纵坐标被翻了b次,则这个硬币总共被翻了a*b次,若想要这个硬币被翻奇数次,a和b必须都得是奇数,即x和y都有奇数个约数
4.那么问题来了:哪些数有奇数个约数呢?不管你知不知道,反正现在你知道了,完全平方数有奇数个约数。那么什么又是完全平方数呢,简单的说就是n^2,n为自然数,也就是0,2,4,9……
5.问题又来了,怎么求完全平方数的个数呢,首先,我们已经知道了这个矩阵式n*m的,而且是从1开始编号的,对于n,我们可以求sqrt(n),然后取整,容易想出,在1-n的范围内的完全平方数的个数为(int)(sqrt(n))个,而sqrt(n)*sqrt(m)就是所有的横纵坐标都是完全平方数的硬币的个数。
/**
*
*/
package 历届试题;
import java.math.BigInteger;
import java.util.Arrays;
import java.util.Scanner;
import javax.script.Bindings;
/**
* @author Administrator
*
*问题描述
小明先把硬币摆成了一个 n 行 m 列的矩阵。
随后,小明对每一个硬币分别进行一次 Q 操作。
对第x行第y列的硬币进行 Q 操作的定义:将所有第 i*x 行,第 j*y 列的硬币进行翻转。
其中i和j为任意使操作可行的正整数,行号和列号都是从1开始。
当小明对所有硬币都进行了一次 Q 操作后,他发现了一个奇迹——所有硬币均为正面朝上。
小明想知道最开始有多少枚硬币是反面朝上的。于是,他向他的好朋友小M寻求帮助。
聪明的小M告诉小明,只需要对所有硬币再进行一次Q操作,即可恢复到最开始的状态。然而小明很懒,不愿意照做。于是小明希望你给出他更好的方法。帮他计算出答案。
输入格式
输入数据包含一行,两个正整数 n m,含义见题目描述。
输出格式
输出一个正整数,表示最开始有多少枚硬币是反面朝上的。
样例输入
2 3
样例输出
1
数据规模和约定
对于10%的数据,n、m <= 10^3;
对于20%的数据,n、m <= 10^7;
对于40%的数据,n、m <= 10^15;
对于10%的数据,n、m <= 10^1000(10的1000次方)。
*/
public class 矩阵翻硬币 {
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner sc=new Scanner(System.in);
String n=sc.next();
String m=sc.next();
BigInteger ans1=BigSqrt(n);
BigInteger ans2=BigSqrt(m);
BigInteger ans=ans1.multiply(ans2);
System.out.println(ans);
}
private static BigInteger BigSqrt(String s) {
int mlen=s.length();//被开方数的长度
int len;//开方后的长度
BigInteger beSqrtNum=new BigInteger(s);//被开方数
BigInteger sqrtOfNum;//存储开放后的数
BigInteger sqrtOfNumMul;//开方数的平方
String sString;//存储sArray转化后的字符串
if(mlen%2==0)
len=mlen/2;
else
len=mlen/2+1;
char[] sArry=new char[len];
Arrays.fill(sArry, '0');
for (int pos = 0; pos < len; pos++) {
//从最高开始遍历数组,每一位都转化为开方数平方后刚好不大于被开方数的程度
for (char num = '1'; num <= '9'; num++) {
sArry[pos]=num;
sString=String.valueOf(sArry);
sqrtOfNum=new BigInteger(sString);
sqrtOfNumMul=sqrtOfNum.multiply(sqrtOfNum);
if (sqrtOfNumMul.compareTo(beSqrtNum)==1) {
sArry[pos]-=1;
break;
}
}
}
return new BigInteger(String.valueOf(sArry));
}
}