【编程马拉松】【007-循环数】
2016-06-01 08:01
447 查看
【编程马拉松算法目录>>>】
【007-循环数】【工程下载>>>】
1 题目描述
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; /** * Author: 王俊超 * Time: 2016-05-09 06:56 * CSDN: http://blog.csdn.net/derrantcm * Github: https://github.com/Wang-Jun-Chao * 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 测试结果
5 其它信息
因为markddow不好编辑,因此将文档的图片上传以供阅读。Pdf和Word文档可以在Github上进行【下载>>>】。相关文章推荐
- deeplab script---python
- Asp.Net底层解析
- 【编程马拉松】【006-统计一】
- PPTP服务器配置选项详解
- Bluemix云端数据库服务ClearDB MySQL使用示例———Python开发投票程序
- 编程小练习
- ASP.NET Core管道深度剖析(4):管道是如何建立起来的?
- 编程小练习
- Python3.x和Python2.x的区别
- 用wxpython 5行代码显示一个对话框
- C++设计模式浅识外观模式
- Java常量的申明和使用
- Java算术运算符和关系运算符详解
- Java逻辑运算符和位运算符详解
- C++设计模式浅识模板方法模式
- Java移位运算符和赋值运算符
- Java三目运算符和逗号运算符
- Java运算符的优先级
- Java分支结构-if语句
- Java循环结构-for语句、while语句详解