首页 > 其他 > 详细

LeetCode(一)关于GrayCode的实现

时间:2014-03-21 23:55:39      阅读:582      评论:0      收藏:0      [点我收藏+]

在LeetCode上面有一道题,是关于Gray Code的实现的。

GrayCode是这样一种编码:

1 位Gray Code :

0
1
2 位Gray Code:

先添加一个镜像,如下:

0
1
1
0
然后,在原来的编码前面添加“0”,在镜像码前面添加“1”,如下:

00
01
11
10
而从2位变化到3位的Gray Code的实现方式跟从1位变化到2位的过程是一样的, 都是先添加镜像码,再分别添加 “0” 和 “1”,然后,。。。

题目的要求是:

Q:给出一个n

A:输出格雷码序列的十进制,上面2位的格雷码,有00,01,11,10,那么就要输出0,1,3,2。

第一想法就是,先照着Gray Code的定义写出来,再将其变成十进制不就好了吗,于是写吧。

	private ArrayList<Integer> grayCode(int n) {
		if (n == 0) {
			ArrayList<Integer> templist = new ArrayList<>(1);
			templist.add(0);
			return templist;
		}
		int length = 1 << n;
		ArrayList<Integer> list = new ArrayList<>(length);
		String[] numbers = new String[length];
		int k = 2;
		numbers[0] = "0";
		numbers[1] = "1";
		while (k <= n) {
			int mid = 1 << (k - 1);
			for (int i = 0; i < mid; i++) {
				numbers[mid + i] = "1" + numbers[mid - i - 1];
				numbers[mid - i - 1] = "0" + numbers[mid - i - 1];
			}
			k++;
		}
		for (CharSequence number : numbers) {
			int value = 0;
			int len = number.length();
			for (int i = 0; i < len; i++) {
				int val = (int) (number.charAt(i) - ‘0‘);
				value += val * (1 << (len - i - 1));
			}
			list.add(value);
		}
		return list;
	}

简单说一下思路:

1)如果是0,那么就直接返回一个包含“0”的list。

2)如果大于0,那么先将数组的前两个元素先置成0,1,这是最基本的两个元素。

3)接下来,按照GrayCode的定义,为这两个元素添加镜像码,本来应该是下面这样的变成镜像码:

numbers[mid + i] = numbers[mid - i - 1];
不过考虑到镜像码前面会添加“1”,那么就直接在这一步实现就好了,所以就是上面的代码了。

4)格雷码实现好了,将它变成十进制的数值,然后放到list,返回。


提交,是可以通过的。不过接着仔细一想,自己果然是傻的呀,题目的要求只是要输出十进制的数值,干嘛要用字符串呢,于是琢磨着琢磨着,就想出了下面的方法:

	private ArrayList<Integer> grayCode2(int n) {
		int length = 1 << n;
		ArrayList<Integer> list = new ArrayList<>(length);
		list.add(0);
		if (n > 0) {
			list.add(1);
			int k = 2;
			while (k <= n) {
				int mid = 1 << (k - 1);
				for (int i = 0; i < mid; i++) {
					list.add(mid + i, mid + list.get(mid - i - 1));
				}
				k++;
			}
		}
		return list;
	}

在这个方法中,不用字符串,直接用整数来处理。

1)第一步,先添加一个0,然后判断n的值,如果n > 0,则继续下去算。

2)如果n > 1,那么第二个Gray Code的值一定是1,所以就直接添加1。

3)对于n >=2 的情况,则可以分成两种情况,每次添加镜像码的时候,在镜像码前面添加“1”,而在原来的Gray Code前面添加"0",那么可以发现,前半部分的格雷码,它们的值其实是不变的!其实根本没有必要处理它!我这是多么笨才想到这的啊!

而对于后半部分,前面添加1,其实就是加上一个相对应位数的值,比如3(n)位Gray Code的第 2(k)位(从低往高数),就是添加2 ^ (k-1) = 2,它的另外的实现方式则是 

1 <<(k - 1),而这个值,刚好在上面其实就算出来了,所以就直接用就好了,这样一来,就直接将GrayCode的十进制数值给求出来了。

一测试,通过!

但是,后来在网上看到别人的实现,如下:

	private ArrayList<Integer> grayCode3(int n) {
		int length = 1 << n;
		ArrayList<Integer> list = new ArrayList<>(length);
		for (int i = 0; i < length; i++) {
			list.add((i >> 1) ^ i);
		}
		return list;
	}

突然发现,自己真的是不适合干这行,太笨了!

LeetCode(一)关于GrayCode的实现,布布扣,bubuko.com

LeetCode(一)关于GrayCode的实现

原文:http://blog.csdn.net/linmiansheng/article/details/21746827

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!