bzoj1081 [SCOI2005]超级格雷码

Description

著名的格雷码是指2n个不同n位二进制数(即0~2n-1,不足n位在前补零)的一个排列,这个排列满足相邻的两个二进制数的n位数字中最多只有一个数字不同(例如003和001就有一个数位不同,而003和030有两个数位不同,不符合条件)。例如n=2时,(00,01,11,10)就是一个满足条件的格雷码。 所谓超级格雷码就是指Bn个不同的n位B进制数的排列满足上面的条件。 任务:给出n和B(2≤B≤36, 1≤Bn≤65535),求一个满足条件的格雷码。对于大于9的数位用A~Z表示(10~35)。

Input

只有一行,为两个整数n和B。

Output

一共Bn个行,每行一个B进制数,表示你所求得的符合条件的排列

Sample Input

2 2

Sample Output

00
01
11
10

这题好蛋疼啊好蛋疼

显然这是要找个规则构造方案的问题(怎么跟cf的题那么像)

随便找几组数据写一写就发现了

以2 4为例

我的输出是

00,10,20,30,     31,21,11,01,     02,12,22,32,     33,23,13,03

为什么要这样断开?看的更清楚了

我们枚举第二位的数字的时候从小到大递增,然而枚举第一位的时候却要一增一减。这就给了我们启示

先正着枚举第i位的数字,然后在i%2==0的时候正着枚举第i-1位的数字,在i%2==1的时候倒着枚举第i-1位的数字。

当倒着枚举第i位的数字的时候同理,只不过要反过来而已。

这样问题就解决了。

实际上这就是个找规律爆搜题。只要记录下当前枚举到哪一位、这一位要正着枚举还是倒着枚举就好了

代码很短

#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while(ch<‘0‘||ch>‘9‘){if(ch==‘-‘)f=-1;ch=getchar();}
    while(ch>=‘0‘&&ch<=‘9‘){x=x*10+ch-‘0‘;ch=getchar();}
    return x*f;
}
int n,m;
int a[100];
inline void out()
{
    for (int i=1;i<=n;i++)
      if (a[i]<10)printf("%d",a[i]);
      else printf("%c",a[i]-10+‘A‘);
    printf("\n");
}
inline void work(int len,int opr)//opr==1:up  opr==0:down
{
    if (len==0)
    {
        out();
        return;
    }
    if (opr==1)
    for (int i=0;i<m;i++)
    {
        a[len]=i;
        if (i&1)work(len-1,0);else work(len-1,1);
    }
    else if (opr==0)
    for (int i=m-1;i>=0;i--)
    {
        a[len]=i;
        if (i&1)work(len-1,1);else work(len-1,0);
    }
}
int main()
{
    n=read();m=read();
    work(n,1);
}
时间: 2024-12-20 09:51:32

bzoj1081 [SCOI2005]超级格雷码的相关文章

[BZOJ 1081] [SCOI2005] 超级格雷码 【找规律】

题目链接:BZOJ - 1081 备注:此题BZOJ上貌似没有 spj ,要把一般顺序的每个格雷码倒着输出...比如 0102 输出为 2010 题目分析 就是按照 Gray 码的生成方法写前几个出来找找规律就好了. 生成方法:以 3 位 3 进制为例 0 0 0 0 0 1 0 0 2 0 1 2 //中位写 1 ,后面镜像复制 0 1 1 0 1 0 0 2 0 //中位写 2,后面镜像复制 0 2 1 0 2 2 1 2 2 //高位写 1 ,后面镜像复制 1 2 1 ........ 代

BZOJ1081超级格雷码

著名的格雷码是指2n个不同n位二进制数(即0~2n-1,不足n位在前补零)的一个排列,这个排列满足相邻的两个二进制数的n位数字中最多只有一个数字不同(例如003和001就有一个数位不同,而003和030有两个数位不同,不符合条件).例如n=2时,(00,01,11,10)就是一个满足条件的格雷码. 所谓超级格雷码就是指Bn个不同的n位B进制数的排列满足上面的条件. 任务:给出n和B(2≤B≤36, 1≤Bn≤65535),求一个满足条件的格雷码.对于大于9的数位用A~Z表示(10~35). 题解

[bzoj1081]超级格雷码

手动模拟一组样例,可以理解大致应该如何操作具体实现中,记录每一位当前应该+1还是-1,然后操作最低的位并将后面的+1和-1全部取反即可 1 #include<bits/stdc++.h> 2 using namespace std; 3 int n,b,a[105],flag[105]; 4 void write(){ 5 for(int i=1;i<=n;i++) 6 if (a[i]<10)printf("%d",a[i]); 7 else printf(&

连环锁 格雷码知识+大数

连环锁 题目抽象:某个格雷码到另一个格雷码最少需要的转换步数. 思路:直接求出对应得十进制,相减取绝对值. 当n=2时变换的序列为00,01,11,10 当n=3时变换的序列为000,001,011,010,110,111,101,100 …… 仔细观察发现这正是格雷码,相邻两个状态只有一位不同(实际上题目描述已经说了这个条件)! 十进制数 自然二进制数 格雷码 0 0000 0000 1 0001 0001 2 0010 0011 3 0011 0010 4 0100 0110 5 0101

【格雷码】

/* 格雷码 说明: Gray Code是一个数列集合 ,每个数使用二进位来表示 ,假设使用n位元来表示每个数好了 ,任两个数之间只有一个位元值不同, 例如以下为3位元的Gray Code: 000 001 011 010 110 111 101 100 由定义可以知道,Gray Code的顺序并不是唯一的,例如将上面的数列反过来写,也是一组GrayCode: 100 101 111 110 010 011 001 000 Gray Code是由贝尔实验室的Frank Gray在1940年代提出

FPGA学习笔记之格雷码、边沿检测、门控时钟

一.格雷码 格雷码的优点主要是进位时只有一位跳变,误码率低. 1.二进制转格雷码 我们观察下表: 二进制码 格雷码 00 00 01 01 10 11 11 10 二进制码表示为B[],格雷码表示为G[],则有 G(i) = B(i),i为最高位 G(i-1) = B(i) xor B(i-1),i非最高位 用verilog可以这样写 1 reg [WIDTH-1:0] bin; 2 reg [WIDTH-1:0] gray; 3 parameter WIDTH = 8; 4 always @(

格雷码的实现

问题:产生n位元的所有格雷码. 格雷码(Gray Code)是一个数列集合,每个数使用二进位来表示,假设使用n位元来表示每个数字,任两个数之间只有一个位元值不同. 例如以下为3位元的格雷码: 000 001 011 010 110 111 101 100 . 如果要产生n位元的格雷码,那么格雷码的个数为2^n. 假设原始的值从0开始,格雷码产生的规律是:第一步,改变最右边的位元值:第二步,改变右起第一个为1的位元的左边位元:第三步,第四步重复第一步和第二步,直到所有的格雷码产生完毕(换句话说,已

生成格雷码 转自leetcode

在一组数的编码中,若两个相邻的代码中只有一个二进制数不同,则称这种编码为格雷码. 下面尝试使用递归的方法来生成格雷码 vector<int> GrayCode(int n){//n表示格雷码的位数 if(n==0) return vector<int>(1)//返回只有一个0的容器 vector<int> res=GrayCode(n-1);//采用递归方法 for(int i=res.size()-1 ; i >=0 ;i-- )//采用从后往前的方式 res.

格雷码

*/--> pre.src {background-color: Black; color: White;} 格雷码 void gray_create(int n) { int a[n]; for (int i = 0; i < (i<<n); i++) { a[i] = i^(i>>1); } for (int i = 0; i < n; i++) { printf("%d\n", a[i]); } } Date: 2015-01-23 00