2006年一月月 发布的文章

给“沙场点兵”一些掌声

最近央视一套正在热播一部反映当代军人题材的电视剧"沙场点兵",军事题材电视剧一直是我的最爱,从"突出重围"、到"DA师"再到这部"沙场点兵"我都尽量抽出时间看上几集。军事剧播出自然少不了军迷们对其的评论,我在很多军事网站上看到了对其负面的评论,当然主要是针对剧情逻辑的不合理以及对军事理论的理解差异。我并未从头看起,只是看了中间的几集,就这几集如果让我谈谈的话,我会给沙剧一些掌声。

众所周知,中国现在的军力无论在装备还是在思想上都逊色于欧美等国家的军队,所以在这种情况下拍摄的电视剧,自然不能跟得上军迷们的思维水平,所以那些负面评论就无可非议了。再强调一点的是这仅仅是部电视剧,它仅仅是一些稍熟悉军队的剧作家心目中的中国当代军人的形象和行为。所以我们大可不必过分在意剧中的那些可能有些不合军事逻辑的剧情和那些在我们军迷心中已经落后于时代的训练方法和军事理论。

那么我们到底该看重哪些方面呢?是战法、还是装备?我想都不是。在我看来,我们最应该关注的是通过该剧到底能暴露出多少我们的人民军队的问题和不足,这些不足不是指战法的低级,也不是指装备的落后,而是指那些真正影响人民军队在未来走向强大的那些问题,如官僚、演戏做秀、个人利益高于军队利益、生活在荣誉的光环下不思进取等等。而这部剧在反映这些问题上是值得我们给予其掌声的。

剧中给我印象最深的人物不是庞承功,也不是康凯,而是那个老头-军区副司令员楚淮海,我觉得他是一个真正合格的探索者,一个真正为人民军队的建设思索探路的军队高级将领,一个具有前瞻性的军队伯乐。有时候自己边看剧边在想:像这样的一个老头,在我们的真实的人民军队中到底有多少呢。我是希望越多越好。

剧中还有一个典型的角色让我印象深刻,那就是原师长魏嵩平。魏嵩平这个角色我想最能代表当代人民军队中的一些高级将领,而且还为数不少。如果非要从正反角度来谈,这也就算是个典型的反面角色,不过他却能给我们带来太多的思考:走过80多年坎坷岁月的人民军队当中为什么出现越来越多像魏嵩平这样的人呢?

剧中的两个唱对手戏的主角这里不能不提。庞承功,最开始我就意识到他是个深受西方军事文化影响的一名很有潜质的年轻指挥官,而且从各个方面来讲他都是蓝军部队指挥官的最佳人选,果不其然,剧情至中段,庞承功真的挂帅野狼团;庞承功这个角色映射的是我们当今人民军队中成长起来的新一代的骨干,他们思维敏锐,行为果断,敢于创新甚至铤而走险,自信甚至是自负,而恰恰在他们身上缺少一丝不苟稳重和老练。这样的人需要伯乐来引导,引导有方,才能使他们百炼成钢。康凯,则是自学成材的军中典范,他们没有什么背景,凭借着自己扎实的基本功和不懈的热情在军中扎稳脚跟。稳重是他们的特点。这两种角色的鲜明对比也是我想给沙剧掌声的原因。

最后用原猛虎团参谋长田青河的一句话来做结尾:"回头看的劲头儿多了,向前走的劲头儿小了"。我们很多人在现实生活中又何尝不是如此呢。

用GDB调试多进程程序

有一段时间没有写技术方面的东西了^_^。众所周知,GDB是Unix/Linux下调试程序的龙头老大,GDB功能强大,我们在平时多使用其一些最基本的功能,而且一般调试的都是单进程的程序。最近一个项目中的问题让我接触如何使用GDB调试多进程程序,更确切的是说调试调用fork的多进程程序。

使用GDB最好的文档就是其名为'Debugging with GDB'的参考手册。手册中有一小章节提到了如何调试多进程程序。一般情况下,如果被gdb调试的程序中调用fork派生出一个新的子进程,这时gdb调试的仍然还是父进程,其子进程的执行不被理会。如果之前你在子进程的执行routine上设置了断点,那么当子进程执行到那个断点时,子进程会因为收到一个SIGTRAP信号而自行终止,除非你在子进程中拦截了该信号。

那么使用GDB该如何调试多进程程序呢?在其参考手册中提供了一种通用方法,这里说说(GDB在某些平台上如HP-UX,还提供了更简便的方法,不过不具备通用性,这里不说):

[测试程序]
我们先看看我们的测试程序:
/* in eg1.c */

int wib(int no1, int no2)
{
        int result, diff;
        diff = no1 – no2;
        result = no1 / diff;
        return result;
}

int main()
{
        pid_t   pid;

        pid = fork();
        if (pid <0) {
                printf("fork err\n");
                exit(-1);
        } else if (pid == 0) {
                /* in child process */
                sleep(60); —————— (!)

                int     value   = 10;
                int     div     = 6;
                int     total   = 0;
                int     i       = 0;
                int     result  = 0;

                for (i = 0; i < 10; i++) {
                        result = wib(value, div);
                        total += result;
                        div++;
                        value–;
                }

                printf("%d wibed by %d equals %d\n", value, div, total);
                exit(0);
        } else {
                /* in parent process */
                sleep(4);
                wait(-1);
                exit(0);
        }
}
该测试程序中子进程运行过程中会在wib函数中出现一个'除0'异常。现在我们就要调试该子进程。

[调试原理]
不知道大家发现没有,在(!)处在我们的测试程序在父进程fork后,子进程调用sleep睡了60秒。这就是关键,这个sleep本来是不该存在于子进程代码中的,而是而了使用GDB调试后加入的,它是我们调试的一个关键点。为什么要让子进程刚刚运行就开始sleep呢?因为我们要在子进程睡眠期间,利用shell命令获取其process id,然后再利用gdb调试外部进程的方法attach到该process id上,调试该进程。

[调试过程]
我觉上面的调试原理的思路已经很清晰了,剩下的就是如何操作的问题了。我们来实践一次吧!
我所使用的环境是Solaris OS 9.0/GCC 3.2/GDB 6.1。

GDB调试程序的前提条件就是你编译程序时必须加入调试符号信息,即使用'-g'编译选项。首先编译我们的源程序'gcc -g -o eg1 eg1.c'。编译好之后,我们就有了我们的调试目标eg1。由于我们在调试过程中需要多个工具配合,所以你最好多打开几个终端窗口,另外一点需要注意的是最好在eg1的working directory下执行gdb程序,否则gdb回提示'No symbol table is loaded'。你还得手工load symbol table。好了,下面我们就'按部就班'的开始调试我们的eg1。

执行eg1:
eg1 &   — 让eg1后台运行吧。

查找进程id:
ps -fu YOUR_USER_NAME

运行gdb:
gdb
(gdb) attach xxxxx  — xxxxx为利用ps命令获得的子进程process id
(gdb) stop — 这点很重要,你需要先暂停那个子进程,然后设置一些断点和一些Watch
(gdb) break 37 — 在result = wib(value, div);这行设置一个断点,可以使用list命令察看源代码
Breakpoint 1 at 0×10808: file eg1.c, line 37.
(gdb) continue
Continuing.

Breakpoint 1, main () at eg1.c:37
37                              result = wib(value, div);
(gdb) step
wib (no1=10, no2=6) at eg1.c:13
13              diff = no1 – no2;
(gdb) continue
Continuing.

Breakpoint 1, main () at eg1.c:37
37                              result = wib(value, div);
(gdb) step
wib (no1=9, no2=7) at eg1.c:13
13              diff = no1 – no2;
(gdb) continue
Continuing.

Breakpoint 1, main () at eg1.c:37
37                              result = wib(value, div);
(gdb) step
wib (no1=8, no2=8) at eg1.c:13
13              diff = no1 – no2;
(gdb) next
14              result = no1 / diff;
(gdb) print diff
$6 = 0        ——- 除数为0,我们找到罪魁祸首了。
(gdb) next
Program received signal SIGFPE, Arithmetic exception.
0xff29d830 in .div () from /usr/lib/libc.so.1

至此,我们调试完毕。

上面仅仅是一个简单的多进程程序,在我们平时开发的多进程程序远远比这个复杂,但是调试基本原理是不变,有一些技巧则需要我们在实践中慢慢摸索。

如发现本站页面被黑,比如:挂载广告、挖矿等恶意代码,请朋友们及时联系我。十分感谢! Go语言第一课 Go语言精进之路1 Go语言精进之路2 商务合作请联系bigwhite.cn AT aliyun.com

欢迎使用邮件订阅我的博客

输入邮箱订阅本站,只要有新文章发布,就会第一时间发送邮件通知你哦!

这里是 Tony Bai的个人Blog,欢迎访问、订阅和留言! 订阅Feed请点击上面图片

如果您觉得这里的文章对您有帮助,请扫描上方二维码进行捐赠 ,加油后的Tony Bai将会为您呈现更多精彩的文章,谢谢!

如果您希望通过微信捐赠,请用微信客户端扫描下方赞赏码:

如果您希望通过比特币或以太币捐赠,可以扫描下方二维码:

比特币:

以太币:

如果您喜欢通过微信浏览本站内容,可以扫描下方二维码,订阅本站官方微信订阅号“iamtonybai”;点击二维码,可直达本人官方微博主页^_^:
本站Powered by Digital Ocean VPS。
选择Digital Ocean VPS主机,即可获得10美元现金充值,可 免费使用两个月哟! 著名主机提供商Linode 10$优惠码:linode10,在 这里注册即可免费获 得。阿里云推荐码: 1WFZ0V立享9折!


View Tony Bai's profile on LinkedIn
DigitalOcean Referral Badge

文章

评论

  • 正在加载...

分类

标签

归档



View My Stats