java开发系列基本:放大招!为系统开发星球大战游戏程序

眼下,我们的连串现已升高到了一定较为周到的档次,当然最好一个操演系统,它无法跟正式的windows或linux相比较,但麻雀虽小五腑俱全,我们的系列或者能支撑用户在其下边开发出许多错综复杂但又幽默的应用程序的,它所显现的效率完全可以称得上是一个鲁棒的操作系统。

我们看看第一个好玩的应用程序:不规则窗口。
不少独居特色的应用程序,一大特点之一就是颠三倒四窗口,奇形怪状的窗体形态确实容易给人眼睛一亮的痛感,不规则窗口首假设把窗体的某部分用透明色来显现,现在大家就来尝试一下,在app.c中成功以下代码:

2017年12月23日    星期六    天气晴

void main() {
    int win;
    char buf[150*70];
    win = api_openwin(buf, 150, 70, 255, "noterec");
    api_boxfilwin(win, 0, 50, 34, 69, 255);
    api_boxfilwin(win, 115, 50, 149, 69, 255);
    api_boxfilwin(win, 50, 30, 99, 49, 255);
    for (;;) {
        if (api_getkey(1) == 0x1c) {
            break;
        }
    }
    api_closewin(win);
    return;
}

余医务卫生人员出生于悬壶济世的祖传医家,大伯是远近著名的乡医,家道殷实,从小聪明绝顶,眉清目秀,子承父业成为镇卫生院经理医务卫生人员,娶贤妻得外孙女,日子过得和美。

下边代码先创制一个窗体,然后把窗体的几何个部分填充成透明色,于是窗体形态变成一个好玩的非矩形,下面代码完成后,运行起来情形如下:

不过这余大夫心术不正,花花肚肠贪恋美色,与同事妙龄少妇兰勾搭成奸,于是对家园贤妻百般挑剔,可怜这母女成了其婚外恋的散货。母女俩被扫地出门形孤影只,余医务人员只静等兰与他再结秦晋之好。

图片 1

唯恐是兰对余医师的为人处世有所顾忌,突然中途退出回归家庭。却说这余医务卫生人员依心像意,没人在他眼里瞧得上,医院里除了比她地点高的司长肯打个招呼,另外皆爱搭不理。倘诺回到村上,遇见所有人都不屑一顾,恨不得别人都低头哈腰向他行鞠躬礼。

此地写图片描述

却说余医务人员离婚成了孤独,再再次回到接受前妻女不是他生性。正在她为如此窝囊事心烦意乱、焦躁不堪时,秋成了他的及时雨。

接下去我们再看看一个更有意思的用户程序,线团火球!在app.c中添加代码如下:

要说这秋非同小可,凭着一身狐骚气,专门靠男人上位而有名。原先单位实际臭名昭著混不下去了,才成为余医师同事。

struct POINT {
    int x;
    int y;
};

void main() {
    int win;
    char buf[216*237];
    struct POINT table[18] = {
    {204,129},{195,90},{172,58},{137,38},{98,34},
    {61,46},{31,73},{15,110},{15,148},{31,185},
    {61,212},{98,224},{137,220},{172,200},{195,168},
    {204,129}
    };

    win = api_openwin(buf, 216, 237, -1, "bball");
    api_boxfilwin(win, 8, 29, 207, 228, 255);
    int i = 0, j = 0, dis = 0;
    for (i = 0; i <= 14; i++) {
      for(j = i+1; j <= 15; j++) {
          dis = j - i;
          if (dis >= 8) {
             dis = 15 - dis;
          }
          if (dis != 0) {
             api_linewin(win, table[i].x, table[i].y,
                         table[j].x, table[j].y, 8-dis);
          }
      }
    }

    api_refreshwin(win, 8, 29, 207, 228);
    for (;;) {
        if (api_getkey(1) == 0x1c) {
            break;
        }
    }
    api_closewin(win);

    return;    
}

秋是怎么样会察言观色的女士,余医务卫生人员很快成其猎物。正好填补了空档,余医务人员又得意忘形。好心人提示过他别招惹秋这样的“心机婊”,余医务人员不以为然,自以为把控秋这样的小女孩子就是区区小事。

地方代码运行后效果如下:

很快,秋与余医务卫生人员结为夫妻,心怀叵测之秋的计划始于举行。余医务卫生人员的舅舅是副镇长,仕途方面肯定帮得上余医务人员的忙。舅舅与余医师二姑同父异母平常提到一般没啥来往,此时此刻秋的身手充分表现。

图片 2

恰逢镇卫生院老司长行将退居二线,余医务人员也跃跃欲试。秋出谋划策,副乡长这样的官场资源岂可白白浪费。余医务卫生人员面露难色,秋拍拍胸脯包在她随身。

此地写图片描述

果真,余医务卫生人员夫妇登门拜访,舅舅尽弃前嫌,大人大量热情款待。秋的三寸不烂之舌功夫发挥到极致,余医务卫生人员也为了自己前程,在舅舅面前摆出最好不安的规范。

绘图出来的图画是不是这一个惊艳,分外灿烂,我们的系统平台依然蛮给力的。

话表六头。在秋的刻意操控下,夫妇俩转性似的,变得温柔起来。尤其是镇医院同事值夜班,夫妇俩还会准备好非凡的夜宵,带到医务室里供我们享用。医务人员护士们得人恩惠,对这对夫妻刮目相看。

接下去大家要放大招,开发一个诙谐的星球大战游戏。星球大战是一款像素级游戏,它的形态让自家想起早年,大概是98年,所谓电脑刚传入中华,系统或者DOS时,当时上电脑课最欢喜的实在有打闹可玩。代码相比较长,大家分段实现,首先在app.c中输入以下代码:

舅舅的暗中相助,同事们的努力推荐,余医务卫生人员称心满意坐上了镇医院委员长宝座。权杖到手,余医务卫生人员摇身一变成为秘书长大人,立马恢复生机了原来,有过之而无不及。

#define win_width 240

void putstr(int win, char *winbuf,char * charset,
 int x, int y, int col, unsigned char *s) {
    int c, x0, i = 0, j;
    char *p, *q, t[2];
    x = x*8+8;
    y = y*16+29;
    x0 = x;
    while (s[i] != 0) {
        i++;
    }

    api_boxfilwin(win, 8, y, win_width - 8 , y+15, 0);

    q = winbuf + y * win_width; 
    t[1] = 0;
    for (;;) {
        c = *s;
        if (c == 0) {
            break;
        }

        if (c != ' ') {
            if ('a' <= c && c <= 'h') {
                p = charset + 16 * (c-'a');
                q += x;
                for (i = 0; i < 16; i++) {

                    if ((p[i] & 0x80) != 0) {q[0] = col;}
                    if ((p[i] & 0x40) != 0) {q[1] = col;}
                    if ((p[i] & 0x20) != 0) {q[2] = col;}
                    if ((p[i] & 0x10) != 0) {q[3] = col;}
                    if ((p[i] & 0x08) != 0) {q[4] = col;}
                    if ((p[i] & 0x04) != 0) {q[5] = col;}
                    if ((p[i] & 0x02) != 0) {q[6] = col;}
                    if ((p[i] & 0x01) != 0) {q[7] = col;}

                    q += win_width; 
                }
                q -= win_width * 16 + x;
            } else if((c>='i' && c <= 'z') || (c>='A' && c <= 'Z')){
                t[0] = *s;
                api_putstrwin(win, x, y, col,1,t);
            }
        }

        s++;
        x += 8;
    }

    api_refreshwin(win,  8, y, win_width - 8, y+15);
}



void main() {
    int win, i, j, ix, iy, fx;
    char winbuf[win_width * 237], invstr[32*6], s[12];
    char* p = winbuf;

    char invstr0[36] = {' ' , 'a','b','c','d',' ' ,'a','b','c','d', ' ',
                         'a','b','c','d',' ',  'a','b','c','d',' ',
                           'a','b','c','d', ' ',
                         0};

    char charset[18 * 8] = {
    0x00, 0x00, 0x00, 0x43, 0x5f, 0x5f, 0x5f, 0x7f,
    0x1f, 0x1f, 0x1f, 0x1f, 0x00, 0x20, 0x3f, 0x00,

    0x00, 0x0f, 0x7f, 0xff, 0xcf, 0xcf, 0xcf, 0xff,
    0xff, 0xe0, 0xff, 0xff, 0xc0, 0xc0, 0xc0, 0x00,

    0x00, 0xf0, 0xfe, 0xff, 0xf3, 0xf3, 0xf3, 0xff,
    0xff, 0x07, 0xff, 0xff, 0x03, 0x03, 0x03, 0x00,

    0x00, 0x00, 0x00, 0xc2, 0xfa, 0xfa, 0xfa, 0xfe,
    0xf8, 0xf8, 0xf8, 0xf8, 0x00, 0x04, 0xfc, 0x00,

    0x00, 0x00, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01,
    0x01, 0x43, 0x47, 0x4f, 0x5f, 0x7f, 0x7f, 0x00,

    0x18, 0x7e, 0xff, 0xc3, 0xc3, 0xc3, 0xc3, 0xff,
    0xff, 0xff, 0xe7, 0xe7, 0xe7, 0xe7, 0xff, 0x00,

    0x00, 0x00, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80,
    0x80, 0xc2, 0xe2, 0xf2, 0xfa, 0xfe, 0xfe, 0x00,

    0x00, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18,
    0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x00
    };

    win = api_openwin(winbuf, win_width, 237, -1, "invader");
    api_boxfilwin(win, 8, 29, win_width - 6, 228, 0);
    //putstr(win, winbuf, 22, 0, 7, "HIGH:00000000");

restart:
    fx = 18;
    char fly[8] = {'e', 'f', 'g', 0};
    putstr(win, winbuf, charset, 18, 11, 6, fly);

nextgroup:
    ix =  7;
    iy = 1;
    int invline = 6;
    for (i = 0; i < 3; i++) {
        for (j = 0; j < 20; j++) {
            invstr[i * 32 + j] = invstr0[j];
        }
        putstr(win, winbuf,charset, ix, iy + i, 2, invstr + i * 32);
    }


    api_refreshwin(win, 8, 29, 207, 228);

    for(;;) {
        if (api_getkey(1) == 0x1c) {
            break;
        }
    }

    api_closewin(win);

    return;
}

嗤笑权力的娱乐,余医师在物欲横流的秋策动下狮子大开口,贪污受贿无所不可能,没多短期就钵满盆满富得流油。已经晋升副秘书长的舅舅略有耳闻分外令人担忧,主动找这么些外甥敲警钟。尤其是余大夫对前妻女不管不顾这件事,舅舅苦口婆心奉劝其至少得承担一个做小叔的权利。不过,余医务卫生人员置之度外。

putstr函数有四个成效,当输入字符在’a’到’h’之间时,它遵照点阵表charset,通过设置某个像素点的水彩来绘制图像。其中参数x,y是图像初始坐标,最终一个参数s用来决定图像绘制的办法,例如当最终的s内容为”abcd”时,该函数从charset表中取出前四行数据,其中每一个数值都在告知代码在绘制哪一个像素点,它的绘图原理与早前大家讲课的哪些绘制字符是均等的。我们先看看下面代码运行起来后的情况:

元配女真是要命,因为缺失经济来源,战表可以的闺女连读高校也只可以无可奈何遗弃。

图片 3

出来混迟早要还的。水性杨花的秋不甘寂寞,在诊所里偷偷找年轻男同事满意自己的淫欲。这么些夜晚,一值班男医务卫生人员裸着身躯虚脱在秋的床上,当班护士有时不我待病情才察觉了这丑陋一幕。余医生大为恼火,却顾及身份与脸面,只好选拔隐忍。

此处写图片描述

话说余医师依然那么耀武扬威,开着医院公车去老家探亲,结果被人举报。纪检部门低度重视,展开专项调查,一查触目惊心。

代码绘制了三排外星人和一架战斗机,在main函数中,代码:

以此时候的秋为了维护好温馨,主动找纪检人士交代事项。如此那般,余医师的题目一清二楚,所犯罪行铁证如山。经法院起诉审判,余医务卫生人员获刑六年登时执行。

restart:
    fx = 18;
    char fly[8] = {'e', 'f', 'g', 0};
    putstr(win, winbuf, charset, 18, 11, 6, fly);

看守所里的余医师尝尽红颜祸水味道的同时,也得美好检查自己的一言一行。秋那样的女孩子不用为她多操心,吃了上家吃下家,一个余医师倒下,千万个“余医务卫生人员”在等候。

绘图的就是丹青底部的战斗机,而代码片段:

最可怜是余先生亲闺女从没得到父爱,只好靠自己微薄的打工收入与姑姑相依为命……

nextgroup:
    ix =  7;
    iy = 1;
    int invline = 6;
    for (i = 0; i < 3; i++) {
        for (j = 0; j < 20; j++) {
            invstr[i * 32 + j] = invstr0[j];
        }
        putstr(win, winbuf,charset, ix, iy + i, 2, invstr + i * 32);
    }

(无戒365巅峰挑衅日更营~第62天)

则是绘制了美术上方的三排外星人。

继而大家要想艺术让图像动起来,这就需要使用到我们前边做过的定时器。首先,我们先添加代码,让飞机按照我们按键输入以便实现左右移动,因此大家添加日下代码:

void putstr(int win, char *winbuf,char * charset,
 int x, int y, int col, unsigned char *s) {
 ....
 }

 void waitting(int i, int timer, char* keyflag) {
    int j;
    if (i > 0) {
        api_settimer(timer, i);
        i = 128;
    } else {
        i = 0x1c;
    }

    for(;;) {
        j = api_getkey(1);
        if (i == j) {
            break;
        }

        if (j == 0x24) {
          //key j
            keyflag[0] = 1;
        }
        if (j == 0x25) {
            //key k
            keyflag[1] = 1;
        }
        if (j == 0x39) {
            //key space
            keyflag[2] = 1;
        }
    }

    return;
}

void main() {
    char keyflag[6];
    ....
    nextgroup:
    ....
    keyflag[0] = 0;
    keyflag[1] = 0;
    keyflag[2] = 0;

    for(;;) {
        waitting(4, timer, keyflag);
        if (keyflag[0] != 0 && fx > 1) {
           //fighter fly to left
           fx--;
           fly[0] = 'e', fly[1] = 'f', fly[2] = 'g', fly[3] = ' ';
           fly[4] = 0;
           putstr(win, winbuf, charset, fx, 11, 6, fly); 
           keyflag[0] = 0;
        }
        if (keyflag[1] != 0 && fx < 24) {
           fx++;
           fly[0] = ' ', fly[1] = 'e', fly[2] = 'f', fly[3] = 'g';
           fly[4] = 0;
           putstr(win, winbuf, charset, fx, 11, 6, fly);
           keyflag[1] = 0;
        }
    }
}

waitting
函数的效率是伺机定时器放回结果,同时监控键盘按键,如若用户在键盘上点击按键’j’,那么它通过调用api_getkey就能取得对应扫描码,也就是0x24,
如若用户点击按键’k’,那么相应的扫描码为0x25,在main函数里,大家添加一个for循环,在中间我们调用waitting函数获取用户按键音讯,当用户按下’j’后,我们把飞机向左移动一个单位,然后刷新窗口,假设用户点击的是k,那么我们把飞机向右移动一个单位,然后刷新窗口,下面代码完成后,编译加载到虚拟机,启动程序后,尝试按下按键’j’,大家可以发现飞机一直向左移动,知道抵达窗口左侧界结束,效果如下:

图片 4

此处写图片描述

飞机能移动后,大家需要外星人也飘飘起来,由此咱们抬高如下代码:

void main () {
....
    for(;;) {
        waitting(4, timer, keyflag);
        ....
                //move aliens
        if (movewait != 0) {
            movewait--;
        } else {
            movewait = movewait0;
            if (ix + idir > 12 || ix + idir < 1) {
                if (iy + invline == 13) {
                   //game over
                   break;
                }
                idir = -idir;
                putstr(win, winbuf, charset,ix + 1, iy, 0, "  ");
                iy++;
            } else {
                ix += idir;
            }
            for (i = 0; i < invline; i++) {
                putstr(win, winbuf,charset, ix, iy + i, 2, invstr + i * 32);
            }
        } 
    }   
}

地点代码先把外星人向右移动,直到边缘时,iy++,这就使得外星人向下活动,与此同时idir
= -idir;
使得外星人开首反向移动,向来到左边边缘,这种运动一贯在进展,直到外星人的y坐标做够大时,游戏截至。

图片 5

此处写图片描述

从下面咱们得以看到,飞机挪到左手时,外星人移动到右手。接着大家要实现用户按下空格键时,飞机能发出炮弹,并且判断一旦炮弹击中外星人后,外星人会被扑灭掉,这个效用的贯彻重大倚重上面代码:

for(;;) {
       if (laserwait != 0) {
            laserwait--;
            keyflag[2] = 0;
        }
        ....
       //draw laser bullet
        if (ly > 0) {
            if (ly < 8) {
                if (ix < lx && lx < ix + 25 && iy <= ly && ly < iy + invline) {
                    putstr(win, winbuf, charset, ix, ly, 2, invstr + (ly - iy) * 32);
                }//if (ix < lx ...)
                else {
                    putstr(win, winbuf, charset, lx, ly, 0, "  ");
               }
            }//if(ly < 13)
            ly--;
            if (ly > 0) {
                char c[2] = {'h', 0};
                putstr(win, winbuf,charset, lx, ly, 3, c);
            } // if(ly > 0)

            if (ix < lx && lx < ix + 25 && iy <= ly && ly < iy + invline) {
               p = invstr + (ly - iy) * 32 + (lx - ix);
               if (*p != ' ' ) {
                   //hit
                   for(p--; *p != ' '; p--) {}
                   for (i = 1; i < 5; i++) {
                       p[i] = ' ';
                   }
                  putstr(win, winbuf, charset, ix, ly, 2, invstr+(ly-iy)*32);
                  for(; invline > 0; invline--) {
                      for(p = invstr + (invline-1)*32; *p != 0; p++) {
                          if (*p != ' ') {
                              goto alive;
                          }
                      }
                  } 
            gameover:
                  //hit all 
                  movewait0 -= movewait0 / 3;
                  char gameover[12] = {'G','A','M','E',' ','O','V','E','R'};
                  putstr(win, winbuf, charset, 15, 6, 1, gameover);
                  waitting(10, timer, keyflag);

                  goto nextgroup;
            alive:
                  ly = 0;
               }
            } //if(ix < lx...)

        }//if (ly > 0)
    }
}

地方代码使得,当用户点击空格键时,程序就准备从飞机处发射出导弹。代码片段:

 if (ly > 0) {
                char c[2] = {'h', 0};
                putstr(win, winbuf,charset, lx, ly, 3, c);
            } // if(ly > 0)

它的功力是,如果用户点击了空格,同时外星人还未曾侵略到飞机所在的势力范围,那么我们就绘制一个从底层向上发射的炮弹,接下去的多级代码是,当炮弹发射后,判断炮弹是否击中外星人,尽管打到了,被打中的外星人要从屏幕上消失掉,这有的逻辑实现的气象如下:

图片 6

此地写图片描述

从上图可以看到,绿色的竖线块就是飞机发射出的导弹,中间这排外星人被击毙了,所以在屏幕上没有彰显出来。假若外星人下降到飞机所在低度,那么有娱乐为止,如若外星人全体被击毙,那么战机成功保卫地球,游戏也终结:

图片 7

此处写图片描述

游玩截至后,”GAME OVER”会打印到界面上,然后一切游戏重来四遍。

其一星球大战游戏是系统基本开发到此,大家做过的极致复杂的客户程序,它调用了几乎拥有api,例如窗口绘制,字符串输出,时钟,键盘监听等等,那一个纷繁程序的打响运行,在必然水平上标明,我们的体系基本具备一定水准的繁杂和鲁棒性,即使它不行与专业操作系统同日而语,但哪怕它再简单,再繁杂,它也对得起一个“操作系统”的称呼。

更详尽的任课和代码调试演示过程,请参见录像
Linux kernel Hacker,
从零构建自己的基石

更多技术信息,包括操作系统,编译器,面试算法,机器学习,人工智能,请看管我的公众号:

图片 8

此间写图片描述