循环数142857问题 java_循环数 - 一贱书生的个人空间 - OSCHINA - 中文开源技术交流社区...

题目描述

142857是一个六位数,我们发现:

142857*1=142857

142857*2=285714

142857*3=428571

142857*4=571428

142857*5=714285

142857*6=857142

即用1到6的整数去乘142857,会得到一个将原来的数首尾相接循环移动若干数字再在某处断开而得到的数字。

也就是说,如果把原来的数字和新的数字都首尾相接,他们得到的环是相同的。只是两个数的起始数字不一定相同。

请写一个程序,判断给定的数不是循环数。

1.1 输入描述:

输入包括多组数据。

每组数据包含一个正整数n,n是2到60位的正整数,并且允许前缀0。即001也是合法的输入数据。

1.2 输出描述:

对应每一组数据,如果是循环数,则输出“Yes”;否则,输出“No”。

1.3 输入例子:

142857

012345

1.4 输出例子:

Yes

No

2 解题思路

假设所求的数字为n,其为m位,则n可以表示为x_(m-1) x_(m-2)…x_0,因为n的位数比较多,可以使用一个整形数组num来表示输入的值int数组的大小为m,num[i]= x_i。num中的每一个下标代表一个数位。

要求n是否为循环数,只要判断num乘以2、3、4、5、6是否都为循环数即可。假设num乘以k(k=2、3、4、5、6)后得到时的结果是ret,如果ret得到的结果的数据位数比num多,则num不是循环数,如果数据位数一样多,则对ret进行循环移动,找出移动后的数字ret^’使用得ret^’与num相等,如果不存说明num不是循环数。当k取完所有的数字后都相等,则num是循环数。等价n是循环数,否则n不是循环数。

3 算法实现

import java.util.Scanner;

/**

* Declaration: All Rights Reserved !!!

*/

public class Main {

public static void main(String[] args) {

Scanner scanner = new Scanner(System.in);

// Scanner scanner = new Scanner(Main.class.getClassLoader().getResourceAsStream("data.txt"));

while (scanner.hasNext()) {

String n = scanner.next();

System.out.println(cycleNumber(n));

}

scanner.close();

}

/**

* 判断n是不是循环数

*

* @param n 数字字符串

* @return No:不是循环数,Yes:是循环数

*/

private static String cycleNumber(String n) {

int[] num = new int[n.length()];

// 数字字符串转换成数组表示的数字

// 下标由小到大表示低位到高位,

for (int i = 0; i < n.length(); i++) {

num[i] = n.charAt(n.length() - 1 - i) - '0';

}

//System.out.println(n + Arrays.toString(num));

for (int i = 2; i <= 6; i++) {

if (!check(num, i)) {

return "No";

}

}

return "Yes";

}

/**

* 检查用数组表示的数字与n相乘,是否是循环数

*

* @param num 数组表示的数字,标由小到大表示低位到高位

* @param n 数字

* @return true:是循环数,false:不是循环数

*/

private static boolean check(int[] num, int n) {

// 来自低位的进位

int carry = 0;

// 结果数组

int[] ret = new int[num.length];

int t;

for (int i = 0; i < num.length; i++) {

t = carry + num[i] * n;

ret[i] = t % 10;

carry = t / 10;

}

// 计算之后还有一位说明相乘后结果多出一位,一定不循环数

if (carry != 0) {

return false;

}

// 将ret中的数字进行循环,构造新的ret,看是否与num相等

for (int i = 0; i < ret.length; i++) {

// 找要进行循环移动的位置

if (num[0] == ret[i]) {

exchange(ret, i);

if (equal(ret, num)) {

return true;

}

// 如果不相等就还原

exchange(ret, ret.length - i);

}

}

return false;

}

/**

* 将数组arr循环移动num位

* num将数组arr分成[0, num-1]、[num, arr.length-1]两部分,先将[0, num-1]翻转

* 再将[num, arr.length-1]翻转,最后将[0, arr.length-1]翻转可得结果

*

* @param arr 数组

* @param num 移动的位数

*/

private static void exchange(int[] arr, int num) {

if (arr == null || num < 1 || num > arr.length) {

return;

}

exchange(arr, 0, num - 1);

exchange(arr, num, arr.length - 1);

exchange(arr, 0, arr.length - 1);

}

/**

* 将级数arr中从from到to位置的所有元素进行翻转

*

* @param arr 数组

* @param from 开始位置

* @param to 结束位置

*/

private static void exchange(int[] arr, int from, int to) {

if (arr == null || from < 0 || from > arr.length || to < 0 || to > arr.length) {

return;

}

int t;

while (from < to) {

t = arr[from];

arr[from] = arr[to];

arr[to] = t;

from++;

to--;

}

}

/**

* 比较两个数组的内容是否相等

*

* @param a 数组

* @param b 数组

* @return true:相等,false:不相等

*/

private static boolean equal(int[] a, int[] b) {

if (a == null && b == null) {

return true;

}

if (a == null || b == null || a.length != b.length) {

return false;

}

for (int i = 0; i < a.length; i++) {

if (a[i] != b[i]) {

return false;

}

}

return true;

}

}

4 测试结果

99de554beea8c8d98c2aa18efa58c77e.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值