算法学习3_枚举

枚举算法又称穷举算法
枚举算法的核心思想 : 有序的尝试每一种可能


题一3 * 6528 = 3 * 8256 ,在空格里面填入两个相同的数字保持等式成立?
这是最简单的枚举算法

    for (int i = 0; i < 10; i ++) {
        if ((i * 10 + 3)*6528 == (3*10+i)*8256) {
            printf("空格填入的数值是: %d \n",i);
        }
    } 

题二、_ _ _ + _ _ _ = _ _ _ ,将1-9 分别填入空格,,每一个数字只可以使用一次,
例如 173 + 286 = 459; ,还有 286 + 173 = 459 和之前的算一种可能,有多少种可能

因为每一个数字只能使用一次使得等式成立,所以我使用一个book[10]数组来标记该数字是否有出现,出现了标记为book[i] = 1,默认为book[i] = 0
下面的 为什么要 total/2,是因为 173 + 286 = 459和 286 + 173 = 459算一种可能

// 这个打印当前时间的一个方法
char* getDateTime()
{
    static char nowtime[20];
    time_t rawtime;
    struct tm* ltime;
    time(&rawtime);
    ltime = localtime(&rawtime);
    strftime(nowtime, 20, "%Y-%m-%d %H:%M:%S", ltime);
    return nowtime;
}
int main(int argc, const char * argv[]) {
 int a [10], i,total = 0 ,book[10], sum;  // a[i] 用来存储 9个数 ,使用book[10] 来存储数字出现的次数
    // 因为是 1- 9 ,所以,重1开始计算
    char * time = getDateTime();
    printf("时间 ====%s \n",time);
    // 时间复杂度 O(a^n)  时间复杂度太高
    for (a[1] = 1; a[1] < 10 ; a[1]++) {
        for (a[2] = 1; a[2] < 10 ; a[2]++) {
            for (a[3] = 1; a[3] < 10 ; a[3]++) {
                for (a[4] = 1; a[4] < 10 ; a[4]++) {
                    for (a[5] = 1; a[5] < 10 ; a[5]++) {
                        for (a[6] = 1; a[6] < 10 ; a[6]++) {
                            for (a[7] = 1; a[7] < 10 ; a[7]++) {
                                for (a[8] = 1; a[8] < 10 ; a[8]++) {
                                    for (a[9] = 1; a[9] < 10 ; a[9]++) {
                                        
                                        for (i = 1; i < 10; i ++) {  // 初始化book数组
                                            book[i] = 0;
                                        }
                                        
                                        for (i = 1; i < 10; i ++) {
                                            book[a[i]] = 1;
                                        }
                                        
                                        // 统计出现了多少个不同得数
                                        sum = 0;
                                        for (i = 1; i < 10; i ++) {
                                            sum += book[i];
                                        }
                                        
                                        // 如果出现了 9个数,并且满足等式 条件,这输出
                                        if (sum == 9 && (a[1]* 100 + a[2] * 10 + a[3]) + (a[4]* 100 + a[5] * 10 + a[6]) ==
                                            (a[7]* 100 + a[8] * 10 + a[9])) {
                                            
                                            printf("%d%d%d + %d%d%d = %d%d%d \n" ,a[1],a[2],a[3],a[4],a[5],a[6],a[7],a[8],a[9]);
                                            total ++;
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    }
    
    printf("total === %d \n",total/2);
    time = getDateTime();
    printf("时间 ====%s \n",time);
    getchar();
return;
}

题三、 假如你只有一个炸弹,这颗炸弹威力超强,可以炸死范围里面的所有敌人,放在那里,可以炸死更多的敌人
地图模型化
墙有两种,一种可以炸,一种不可以炸,当前只有一颗炸弹,
所以
成墙用 # 表示,炸弹不能穿墙,
敌人用 G 表示,
空地用 . 表示,炸弹只能放在空地上

 具体地图模型如下
 13 * 13
 #############
 #GG.GGG#GGG.#
 ###.#G#G#G#G#
 #.......#..G#
 #G#.###.#G#G#
 #GG.GGG.#.GG#
 #G#.#G#.#.###
 ##G...G.....#
 #G#.#G###.#G#
 #...G#GGG.GG#
 #G#.#G#G#.#G#
 #GG.GGG#G.GG#
 #############

分析:首先我们需要一个二维数组来存储这个地图,至于炸弹放在哪个位置炸死的敌人最多,需要一个个尝试,炸弹的爆炸方向是 上下左右 所以,炸弹的爆炸统计如下
假如炸弹在坐标x ,y上,炸弹的涉及范围的坐标变化如下
具体消灭如下所示

            (x - 1, y)
 (x, y - 1) (x   ,  y) (x, y + 1)
            (x + 1, y)

具体demo如下

int main(int argc, const char * argv[]) {
    char a[20][20]; // 假设地图大小不超过20*20
    int i ,j, sum ,map = 0,p = 0,q = 0,x,y,n,m;
    printf("输入,n, m, 代表行和列:");
    scanf("%d %d",&n, &m); //  n 代表多少行, m代表多少列
    printf("n = %d , m = %d \n",n,m);
    
    // 读入n行字符
    for (i = 0; i < n; i ++) {
        printf("输入该行参数 %d",i);
        scanf("%s",a[i]);
    }
    // 利用两重循环便利枚举里面的每一个点
    for (i = 0; i < n; i ++) {
        
        for (j = 0; j < m ; j ++) {
            
            // 首先判断这个点是不是平底,是平底才可以防止炸弹
            
            if (a[i][j] == '.') {
                
                sum = 0; // sum 用来计数(可以消灭的敌人数),所以初始化需要为0
                
                // 将当期坐标 i , j 复制到两个新变量 x , y 中, 以便上下左右四个方向统计可以消灭的敌人数
                
                // 向上统计可以消灭的敌人数
                x = i ; y = j;   // 这里的坐标不是x,y 代表的意思是 第x行, 第y列
                while (a[x][y] != '#') {
                    if (a[x][y] == 'G') {
                        sum ++;
                    }
                    x --; // x --继续向上统计可以消灭的敌人数
                }
                // 向下统计可以消灭的敌人数
                x = i ; y = j;   // 这里的坐标不是x,y 代表的意思是 第x行, 第y列
                while (a[x][y] != '#') {
                    
                    if (a[x][y] == 'G') {
                        sum ++;
                    }
                    x ++; // x ++继续向下统计可以消灭的敌人数
                    
                }
                
                // 向左统计可以消灭的敌人数
                x = i ; y = j;   // 这里的坐标不是x,y 代表的意思是 第x行, 第y列
                while (a[x][y] != '#') {
                    
                    if (a[x][y] == 'G') {
                        sum ++;
                    }
                    y --; //
                    
                }
                // 向右统计可以消灭的敌人数
                x = i ; y = j;   // 这里的坐标不是x,y 代表的意思是 第x行, 第y列
                while (a[x][y] != '#') {
                    
                    if (a[x][y] == 'G') {
                        sum ++;
                    }
                    y ++;
                }
                // 更新map 的值
                if (sum > map) {
                    map = sum;
                    
                    p = i; // 记录行
                    q = j; // 记录列
                }
            }
        }
    }
    printf("炸弹坐标为(%d ,%d),可以炸死最多的敌人%d \n",p,q,map); // 结果 (9,9) 消灭8 个
    getchar();
return;
}

本人也是刚刚学习,如有错误,请多多指正。
本文部分内容参考 【啊哈!算法】这本书
代码例子

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容