<?xml version="1.0" encoding="utf-8"?><?xml-stylesheet href='http://feeds.feedsky.com/styles/temp01.xsl' type='text/xsl' ?><!--这是一个由Feedsy提供技术支持的Feed，为了提高读者阅读的体验，以及满足用户美化自己Feed的需要，我们设计了多种精美的Feed模板，提供给大家选择，所有最终呈现出来的样式，皆由用户自愿选择使用，未经许可，任何团体和个人，请不要擅自修改样式或者盗用，这是对于用户选择权的尊重。--><rss xmlns:atom="http://www.w3.org/2005/Atom" xmlns:fs="http://www.feedsky.com/namespace/feed" xmlns:dc="http://purl.org/dc/elements/1.1/" version="2.0"><channel><atom:link href="http://feeds.feedsky.com/csdn.net/liang13664759" type="application/rss+xml" rel="self"></atom:link><fs:self_link href="http://feeds.feedsky.com/csdn.net/liang13664759" type="application/rss+xml"></fs:self_link><lastBuildDate>Fri, 05 Dec 2008 15:31:00 GMT</lastBuildDate><title>ENDLESS</title><description>ENDLESS.liang</description><link>http://blog.csdn.net/liang13664759</link><language>zh-cn</language><copyright>Copyright &amp;copy; liang13664759</copyright><pubDate>Wed, 23 May 2012 16:44:29 GMT</pubDate><image><url>http://static.blog.csdn.net/images/logo.gif</url><link>http://blog.csdn.net</link></image><item><title>[原]很久</title><link>http://blog.csdn.net/liang13664759/article/details/3456518</link><description>&lt;P&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 很久没有来写过东西了。因为没有感。毕业快半年了加上实习的时间工作快一年了。在科泰内核组工作了一年在专业上学到了很多，只是生活上让我不太适应，感觉总在奔波。有时会感到疲惫。&lt;/P&gt;
&lt;P&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 今天和小霸王一起下班的时候，讲起在学校的一些事，一股怀念涌上心头。刚才和我们那伙人聊了下，虽然有隔在千里之外，虽然没有声音只是文字，但是却感到那语气，那些我们的语言，真是他妈的亲切啊。有看了看鑫爷写的东西，让我觉得搞笑之余也有几分惆怅。想想，坐在寝室中往走廊里面大吼一声，隔壁的隔壁房间遥相呼应。哈哈哈哈哈哈哈，真是怀念，特想回去这样的日子。想想，大一时我们去踢振兴杯五人足球，癫狂小子被我们踢得癫狂了（我们7比1），想想我们横扫足球场，哈哈哈哈哈，真是意气风发啊，政法的那帮每次遇到我们就被蹂躏，好像在记忆足球场上中我们是很少输的。想想，我们几个寝室坐一起打游戏，CS，魔兽，玩得是不亦乐乎。想想，在KTV完达山的歌喉刺痛着我们每一个人的心，想想，这帮丫的麦霸。想想，冯姐的硬盘，只想说冯姐我们2011的男生感谢你无私的奉献。想想，南航到南大的那条路上有多少颗石子我都知道。想想，完达山的给我的那辆泡妞车。想想，鑫爷的那些事，绝对的传奇人物。想想，元哥的那些野屁，绝对的今天地泣鬼神。想想，小霸王这些年的变化，我和鑫爷总是感到内疚……&lt;/P&gt;
&lt;P&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 出来了，我们各奔东西，还好上海的兄弟还比较多，周末还一起吃个饭。可能我们都还在适应，可能我们都还时常感到疲惫。无论如何，我们都在奋斗，我们在积聚力量，ENDLESS正在厚积，为了薄发。我们都在诠释我们的生命，不断进取。在为每一个矛盾，找到它的平衡点。我们强大，因为我们无畏，因为我们平凡，因为我们心若止水。我们在为我们ENDLESS要做的事准备着。&lt;/P&gt;
&lt;P&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 很久，下一次不知什么时候了。&lt;/P&gt;
&lt;P&gt;&amp;nbsp;&lt;/P&gt;
&lt;P&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;/P&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2008-12-5 23:31:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/3456518&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：190 评论：1 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/3456518#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Fri, 05 Dec 2008 23:31:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/3456518</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/3456518</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057083/1064678</fs:itemid></item><item><title>[原]快走了</title><link>http://blog.csdn.net/liang13664759/article/details/2562750</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 今天早上我们去照了毕业照，我后天下午答辩。这次是真的要离开了。学生生涯就要结束但是我还没有来得急去感一下。这个和我回校之前的想法完全背离，回来一个月我只想好好休息，踢踢球，喝喝小酒，打打游戏，看看欧洲杯，看看NBA总决赛，看看电影，顺便把论文写了。然后毕业。 &amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;/p&gt;&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 但是今年的安利杯江西省的比赛和全国的总决赛都在我们学校举办，所以一回来就被抓去比赛。就我个人我真的不想去参加，只想很悠闲的过完在学校里的最后时间。结果最后的着一个月就像在打战，很累。比完赛离答辩还有五天的时间，又要开始写论文，感觉就像是一只停不下来骆驼。虽然在省里面的比赛拿了两个一等奖，但是一点也不兴奋，不激动。给我的感觉就是骗骗人的东西，没什么。现在能让我兴奋和激动的肯呢个就是阿宝，激情高昂的跑到我面前，乱七八糟的给我胡扯一通，然后一起去吃小吃，逛逛街，我还特想去游乐场玩玩，看看博物馆，看看电影。然后时不时的把她的嘴拿过来亲一个。哈哈哈哈哈哈！！！！！ &amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&lt;/p&gt;&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &amp;nbsp;四年真是弹指一挥间，在这四年间，我和我身边的人都发生着巨大的变化。大学四年让我改变了很多，四年里每一个人都在做着不同的事，我很高兴，我在这四年里不断的学习，建立了我们ENDLESS团队。而且我们的团队从稚嫩走向成熟。我一直有一种感觉我们的团队有一天一定会做一些和别人不一样的东西。厚积博发。 &amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;/p&gt;&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 生活不容易，但不管怎样抗着呗。只要是事它就躲不过，硬着头皮过。原来我家后面有一座山，叫惠山，我经常去爬山，山腰上经常有人在狂吼，但是不理解，但现在和他们有了些许共鸣。那应该是一种释放，对内心深处的一种释放，把它放出来透透气，不然活不出去了。大多数人都面临着自己的压力，不同的人对压力作出的反应不一样，从而压力也就造就了不同的人。有的人笑看风云，有的人把自己装在一个套子里面去钻牛角尖。其实我觉得压力对一个人的成长是非常有帮助的，在有压力的状态下人是活在逆境中，当然有很多人在逆境中可能永远都出不来，有的人一开始可能会不适应，但是很便学会了在逆境中思考，一人一旦能在逆境和压力下冷静沉着的思考，这个将变得强大，洞察力会越来越敏锐，能在漆黑的夜里看清道路。&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2008-6-18 21:53:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/2562750&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：159 评论：0 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/2562750#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Wed, 18 Jun 2008 21:53:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/2562750</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/2562750</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057082/1064678</fs:itemid></item><item><title>[原]半年</title><link>http://blog.csdn.net/liang13664759/article/details/2474359</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 有大概半年的时间没有来写过东西了。这半年是动荡的半年成长的半年。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 鑫哥去了盛大，去了快两个月了吧。刚去的时候很苦，他说他一个人单挑一个游戏（程序开发部分），他说他经常被骂，心里很难受，很想家，和女人一天至少大四个小时的电话。生活本来就不宽裕的我们，这是一笔不小的开销。最主要的是还和女人吵架，心里面的感受我完全理解，本来在外面混就不容易，上海这个地方什么东西都很贵，要租房，要吃饭，交通费也很贵。他娃也够能抗的，有一天他早上去上班，身上一分钱都没有打电话到处求救（没钱坐车去上班），去花园里找他前天晚上扔在那里面的一块钱硬币。这样的日子他妈的够惨了吧。这应该是鑫哥最惨的时候吧。不过现在好了，他努力的有让他的生活有了些起色他老大不再骂他了，还经常要请他吃饭，还主动要借钱给他。用自己的努力得到别人的认可和尊重。牛B！！！！！！雄得起。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 小伟哥，现在已经是毫无争议的成了中霸王。晚上我们一起走回宿舍的时候，他常常语出惊人，然我和鑫哥两眼相对语赛5秒，然后跪拜在他娃面前叫&amp;ldquo;大哥：我服！&amp;rdquo;有时候我们三个人一路上扯回去，有时候我们三个人一起给自己的女人打着电话回去。他最怕在我们面前打电话，我们会大叫&amp;ldquo;衣服&amp;rdquo;，把他吓得赶紧挂电话（因为他说过一些关于衣服的话让我和鑫哥感到震惊）。这娃别的方面到好像没有什么坎坷，就有几次吵架搞得上班的心思都没有。现在跟着大师兄做浏览器。挺不错的小伙。就是他妈的太急了。叫他明天一起回学校，他娃非要今天走，他说今晚走周末可以和女人玩两天，明天走只能玩一天，这几天上班心早已安在南昌，在公司的只是一堆蛋白质。他现在已经走了和鑫哥晚上九点的车，唉！又是一位积极参与造人运动的小青年。也罢，理解他。回去再扁他。妈的回去就要被别人叫叔，心里面还真是有些不爽！！！！&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 徐黎和丁涛她们两个住在一起，徐黎在一家公司实习，拿了毕业证可能会换个公司吧。丁涛在盛大，好像把工作辞了。去他们拿吃过几次饭。玩了几把黑桃5，小伟哥差点没把我笑死。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 表哥暂无音讯。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 元哥我很羡慕啊，你最无忧无虑的。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 关于我，怎么说呢，感觉就像在打战。很累！从4月30号到现在我还没有停过，在不同的城市间游走。在学校的时候我从没有感觉到钱这东西的重要性，在外面生活什么都要钱，租房，吃饭。圆圆来上海找工作，这段时间确实把我们都累惨了，还好她找工作比较顺利。我们两个三天一小吵，五天一大吵，有时候真觉得累，但是可能拿几天停下来不吵了到又不习惯了。哈哈哈哈哈！！！！！我们有一个小电饭锅！！！我们有吵不完的架，吵的时候想扁人，好的时候嘛又太好了想亲嘴！！！，不知道是不是所有的人都这样的，因为我听付，和鑫说他们也经常吵架。这也许就是生活吧，酸，甜，苦，麻，辣！！！！在专业方面，以前在学校做的都是应用，J2EE，现在出来做操作系统，从一只翅膀上有点羽毛的小鸟，一下变成了一只没有羽毛的雏鸟。半年的时间我学了很多很，但是还有很多很多不懂要去学习。现在我的腿渐渐有了些力量，翅膀上也渐渐长了些羽毛，我正迅速，茁壮的成长着，待我的羽毛长齐，翅膀肌肉有了力量，我将展翅去翱翔。ELastOS早晚有一天会火的，至少它的理念会火的。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 明天我将回南昌，一个月，休息一下！&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 2008年雪灾，藏独，撞火车，地震。我们的国家正饱受煎熬，我们团结在一起，我们会走出困境，我们会腾飞。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 2008我们毕业，我们会坚持我们的理想，坚持不懈地去奋斗，去砖研。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;strong&gt;Some day we&amp;nbsp;will taking on the world!!!&lt;/strong&gt;&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; we are fighting!!!&lt;/strong&gt;&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2008-5-23 17:19:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/2474359&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：177 评论：2 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/2474359#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Fri, 23 May 2008 17:19:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/2474359</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/2474359</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057081/1064678</fs:itemid></item><item><title>[原]适应环境</title><link>http://blog.csdn.net/liang13664759/article/details/2003588</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 不知不觉来上海已经三个星期了。来这里开始确实不适应，住的地方是我长这么大住过的地方中最烂的。&amp;nbsp;还好我的适应能力比较强。不然真的抗不下去。这三个星期没有周末，每一天都在公司。我和鑫、俊，我们每天早上很早到公司，晚上很晚才回家，在住处也就是睡个觉洗个澡。我们都在努力的适应周围的环境。我们以前在学校都是做WEB的，现在都改做嵌入式，而且做得不是应用是操作系统，中国仅此一家做操作系统的，我们都得花一些时间去适应。我很喜欢底层的东西，我觉得底层的东西更计算机。平时在学校做J2EE我觉得我不是很喜欢，并不是说J2EE不好，因为J2EE在WEB上还是很好的一个东西。兴趣所在吧。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 刚来公司的时候什么都不懂，怎么往目标设备上烧程序，怎样在目标设备上调试自己的程序都不懂，而且公司用的是自己的操作系统，不是windows,也不是linux遇到问题不能google，只能向老员工请教。公司有一点比较好的就是牛人的人品都比较好。又问题谁你都可以去问。现在开始慢慢上手了，但是还有太多的东西不懂了，因为做得毕竟是操作系统，我在内核组。每一天我都在努力的学习。每一天都有在进步，我觉这样感觉很踏实。元旦来了明天回家，休息两天。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 还有一点不适应的是我现在每天都看不到我的猪。我很想念她。希望她一个人在那边能好好的照顾自己，希望她的状态能慢慢的好起来，希望她平平安安，希望她健健康康，希望他能快快乐乐。希望我们能早些又在一起。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;fighting!&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-12-29 19:29:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/2003588&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：306 评论：3 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/2003588#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Sat, 29 Dec 2007 19:29:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/2003588</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/2003588</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057080/1064678</fs:itemid></item><item><title>[原]征战上海滩之上码头</title><link>http://blog.csdn.net/liang13664759/article/details/1930142</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 如果说前两次来上海面试算是码头初探的话，那这次应该算是上码头。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 国人自己的操作系统，自主知识产权。好像还真的没有，我实习的公司做的就是操作系统（中国人自己的）。这是让我很骄傲的一点。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 骄傲归骄傲，但是现实中的问题还是很多的。居住的环境，除了比露天多个床外别的也没有什么了。连个板凳桌子都没有，在寝室那叫一个寂寞啊，空荡荡的。不能洗澡，而且厕所之脏，实在让人汗颜。而且吃的东西很贵是学校里面的两倍到三倍。公司给的津贴少的可怜。在一个新的环境才鸟一个一切从零开始。毕竟今天是第一天。而且我觉得在第一天我的表现不错。写了程序但是没有被用上，可能以后会被用上吧。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 人非常的疲惫，前天扛着大包小包从南昌赶到上海，火车上睡了4个小时，然后第二天又跑了很远的路去买生活用品（主要是不认识路）。昨天晚上大概睡了七个小时，今天在公司下午就犯困，确实累。现在看书看不动了，所以就写写东西。征战上海滩确实不容易。但是不管怎么地也已经登陆了，一步一个脚印的往前走我想会好的。再过个一两天稳定下来，适应了这些新环境，可能加速度的方向就会变了。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; This is the city&amp;nbsp; we are&amp;nbsp; fighting for.Someday&amp;nbsp; We&amp;nbsp; Will&amp;nbsp;taking on the World.&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; fighting&amp;hellip;&amp;hellip;！！！&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-12-11 20:11:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1930142&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：286 评论：4 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1930142#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Tue, 11 Dec 2007 20:11:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/1930142</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/1930142</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057079/1064678</fs:itemid></item><item><title>[原]Something  I want to fuck</title><link>http://blog.csdn.net/liang13664759/article/details/1913211</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;今天早上很气愤，应该是非常的火大，刚才打电话把邮政骂了一通。我的EMS一个星期还没到。我去学校收发室，收发室说去我们计算机学院找，我去计算机学院收发室，他说每个星期一，三，五，会有学生去学校收发室取包裹。他妈的，最气人的就是江西邮政，学校收发室，计算机学院收发室，他们三个的当差配合，我靠，那叫一个绝啊！！！！！把钱交个邮政，邮政说我们的投递深度只到你们学院收发室，学院收发室说学生个人不能自己拿，要自己拿叫你们学院院长开证明，我操，他妈的这个他倒是还有内存保护机制。我拿着我的学生证和身份证权限还不够。还要院长签字。再去找计算机学院的收发室，那他妈的就更气人了。他说我不知道那个学生的电话（计算机学院的包裹只有这个学生能去学校的收发室拿），他每个一，三，五会去取，这里没有内存保护，但是他给你玩个订约者模式，而且我在计算机&amp;nbsp;收发室，竟然看到了六月份的EMS还放在那。我汉啊！！！&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 那个收发室的人就是个傻X，老子要是明天就毕业，今天老子肯定把他给DELETE了（把他送医院去），你问他，他一问三不知，妈的，这样的人就没他妈的一点责任感都没有，老子当时真想上去踹他两脚（要得处分，而且马上就要毕业了，所以忍了）。忍在心上插把刀， 我就觉得憋屈，然后就打电话去把EMS骂了一通。没办法钱是他们收的，应该的。EMS那他妈的还叫EMS吗？&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; EMS，南昌航空大学收发室，计算学院收发室，不就是投个包裹嘛，有必要建这么个类嘛，知不知道建类是很消耗资源的，而且他妈的程序做的那么没有水准，没有效率，滥用内存保护，滥用模式。官方机构真是让人汗啊。如果有一天邮政宣告破产，我一点都不会惊讶。圆通，申通，顺风，DHL，这样强有力的竞争者，难道还没有让EMS有一点点地危机感嘛！！！国产你拿什么让我珍惜你！！！！！&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-12-3 10:22:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1913211&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：415 评论：2 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1913211#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Mon, 03 Dec 2007 10:22:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/1913211</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/1913211</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057078/1064678</fs:itemid></item><item><title>[原]处理队列空</title><link>http://blog.csdn.net/liang13664759/article/details/1828055</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 忙了很久很久，有点累了。就像机器转了很久很久，PCB表中除了零进程还有几个快被处理完了的进程，但是现在把0进程-&amp;gt;next&amp;nbsp; = null;先压到硬盘上。过段时间在说吧。其实也没有什么了，广电还有个权限没有改完，蓝牙把写作图和时序图画完，也就这两个东西了。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;大四的课上了一半了，boss给推荐的工作现在还说不准，还在等待。但是，自己也该准备准备，上完课可能就要自己出去找工作了。读了十几年的书就要进社会了。应该没有问题吧！！！我们几个最近好像都应为工作和情感的问题有点浮。俊在找一段感情，都奔三的人了，还没有谈恋爱这事是有点让人着急，鑫最近可能是情况最糟糕的一个，一方面是工作的事悬着，感情方面还在大落大起，告别了，又在组建新的感情，颓废了不少，最近很少看他看书了，都在写小说了？？？？！！！！！不过他经常发生一些让人意想不到的事情，竟然会在半个小时不到时间连续捡到三次钱，共计205元！！！还是大个悠闲，什么都OK，什么都很稳定。关于我嘛，因为可能在有些事情的结果落差有点大，我也有点浮，而且我的那个肥妞状态不好，我很担心她。希望她能快点好起来。表哥也在和俊一样想找一段爱情和一份好工作，状态良好，应该是我们中状态比较好的了。徐徐最近好像状态还不错好像工作因该没有问题了吧，可能也有点想找男人了。完达山嘛，除了考研现在也把爱情这东西搅和进来了，现在和我是南大同盟，状态良好，毕竟他是阿兵哥，不用去找工作。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 其实我最担心的还是我的肥妞，看见她状态不好的时候我心里就焦，不知道怎样才能帮她，想了很多方法好像都没有什么用。我应该怎么办呢，让我担心的猪。我真想把我的脑浆挖出来和她的换换（如果可以的话）。我的猪好起来吧。平凡，从容，enjoy！！！太阳每天都是新的。是金子总会发光的。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 虽然现在我什么事都没有但是我什么都不想做，不想写代码，不想看书。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 洗个澡，我想看灌篮高手，我想看樱木！！&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 系统正在换页调整中！&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-10-16 19:54:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1828055&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：334 评论：2 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1828055#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Tue, 16 Oct 2007 19:54:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/1828055</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/1828055</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057077/1064678</fs:itemid></item><item><title>[转]Linux makefile 教程 非常详细，且易懂</title><link>http://blog.csdn.net/liang13664759/article/details/1771246</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 最近在学习Linux下的C编程，买了一本叫《Linux环境下的C编程指南》读到makefile就越看越迷糊，可能是我的理解能不行。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 于是google到了以下这篇文章。通俗易懂。然后把它贴出来，方便学习。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 后记，看完发现这篇文章和《Linux环境下的C编程指南》的makefile一章所讲述的惊人的相似，只是这篇文章从一个实例切入，在有些地方比较好理解。能让人看懂就是好文章。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;font face=&quot;Verdana&quot;&gt;跟我一起写 Makefile&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;陈皓 (CSDN)&lt;br /&gt;&lt;br /&gt;概述&lt;br /&gt;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;什么是makefile？或许很多Winodws的程序员都不知道这个东西，因为那些Windows的IDE都为你做了这个工作，但我觉得要作一个好的和professional的程序员，makefile还是要懂。这就好像现在有这么多的HTML的编辑器，但如果你想成为一个专业人士，你还是要了解HTML的标识的含义。特别在Unix下的软件编译，你就不能不自己写makefile了，会不会写makefile，从一个侧面说明了一个人是否具备完成大型工程的能力。&lt;br /&gt;&lt;br /&gt;因为，makefile关系到了整个工程的编译规则。一个工程中的源文件不计数，其按类型、功能、模块分别放在若干个目录中，makefile定义了一系列的规则来指定，哪些文件需要先编译，哪些文件需要后编译，哪些文件需要重新编译，甚至于进行更复杂的功能操作，因为makefile就像一个Shell脚本一样，其中也可以执行操作系统的命令。&lt;br /&gt;&lt;br /&gt;makefile带来的好处就是&amp;mdash;&amp;mdash;&amp;ldquo;自动化编译&amp;rdquo;，一旦写好，只需要一个make命令，整个工程完全自动编译，极大的提高了软件开发的效率。make是一个命令工具，是一个解释makefile中指令的命令工具，一般来说，大多数的IDE都有这个命令，比如：Delphi的make，Visual C++的nmake，Linux下GNU的make。可见，makefile都成为了一种在工程方面的编译方法。&lt;br /&gt;&lt;br /&gt;现在讲述如何写makefile的文章比较少，这是我想写这篇文章的原因。当然，不同产商的make各不相同，也有不同的语法，但其本质都是在&amp;ldquo;文件依赖性&amp;rdquo;上做文章，这里，我仅对GNU的make进行讲述，我的环境是RedHat Linux 8.0，make的版本是3.80。必竟，这个make是应用最为广泛的，也是用得最多的。而且其还是最遵循于IEEE 1003.2-1992 标准的（POSIX.2）。&lt;br /&gt;&lt;br /&gt;在这篇文档中，将以C/C++的源码作为我们基础，所以必然涉及一些关于C/C++的编译的知识，相关于这方面的内容，还请各位查看相关的编译器的文档。这里所默认的编译器是UNIX下的GCC和CC。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;关于程序的编译和链接&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;在此，我想多说关于程序编译的一些规范和方法，一般来说，无论是C、C++、还是pas，首先要把源文件编译成中间代码文件，在Windows下也就是 .obj 文件，UNIX下是 .o 文件，即 Object File，这个动作叫做编译（compile）。然后再把大量的Object File合成执行文件，这个动作叫作链接（link）。&lt;br /&gt;&lt;br /&gt;编译时，编译器需要的是语法的正确，函数与变量的声明的正确。对于后者，通常是你需要告诉编译器头文件的所在位置（头文件中应该只是声明，而定义应该放在C/C++文件中），只要所有的语法正确，编译器就可以编译出中间目标文件。一般来说，每个源文件都应该对应于一个中间目标文件（O文件或是OBJ文件）。&lt;br /&gt;&lt;br /&gt;链接时，主要是链接函数和全局变量，所以，我们可以使用这些中间目标文件（O文件或是OBJ文件）来链接我们的应用程序。链接器并不管函数所在的源文件，只管函数的中间目标文件（Object File），在大多数时候，由于源文件太多，编译生成的中间目标文件太多，而在链接时需要明显地指出中间目标文件名，这对于编译很不方便，所以，我们要给中间目标文件打个包，在Windows下这种包叫&amp;ldquo;库文件&amp;rdquo;（Library File)，也就是 .lib 文件，在UNIX下，是Archive File，也就是 .a 文件。&lt;br /&gt;&lt;br /&gt;总结一下，源文件首先会生成中间目标文件，再由中间目标文件生成执行文件。在编译时，编译器只检测程序语法，和函数、变量是否被声明。如果函数未被声明，编译器会给出一个警告，但可以生成Object File。而在链接程序时，链接器会在所有的Object File中找寻函数的实现，如果找不到，那到就会报链接错误码（Linker Error），在VC下，这种错误一般是：Link 2001错误，意思说是说，链接器未能找到函数的实现。你需要指定函数的Object File.&lt;br /&gt;&lt;br /&gt;好，言归正传，GNU的make有许多的内容，闲言少叙，还是让我们开始吧。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;Makefile 介绍&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;make命令执行时，需要一个 Makefile 文件，以告诉make命令需要怎么样的去编译和链接程序。&lt;br /&gt;&lt;br /&gt;首先，我们用一个示例来说明Makefile的书写规则。以便给大家一个感兴认识。这个示例来源于GNU的make使用手册，在这个示例中，我们的工程有8个C文件，和3个头文件，我们要写一个Makefile来告诉make命令如何编译和链接这几个文件。我们的规则是：&lt;br /&gt;1）如果这个工程没有编译过，那么我们的所有C文件都要编译并被链接。&lt;br /&gt;2）如果这个工程的某几个C文件被修改，那么我们只编译被修改的C文件，并链接目标程序。&lt;br /&gt;3）如果这个工程的头文件被改变了，那么我们需要编译引用了这几个头文件的C文件，并链接目标程序。&lt;br /&gt;&lt;br /&gt;只要我们的Makefile写得够好，所有的这一切，我们只用一个make命令就可以完成，make命令会自动智能地根据当前的文件修改的情况来确定哪些文件需要重编译，从而自己编译所需要的文件和链接目标程序。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;一、Makefile的规则&lt;br /&gt;&lt;br /&gt;在讲述这个Makefile之前，还是让我们先来粗略地看一看Makefile的规则。&lt;br /&gt;&lt;br /&gt;target ... : prerequisites ...&lt;br /&gt;command&lt;br /&gt;...&lt;br /&gt;...&lt;br /&gt;&lt;br /&gt;target也就是一个目标文件，可以是Object File，也可以是执行文件。还可以是一个标签（Label），对于标签这种特性，在后续的&amp;ldquo;伪目标&amp;rdquo;章节中会有叙述。&lt;br /&gt;&lt;br /&gt;prerequisites就是，要生成那个target所需要的文件或是目标。&lt;br /&gt;&lt;br /&gt;command也就是make需要执行的命令。（任意的Shell命令）&lt;br /&gt;&lt;br /&gt;这是一个文件的依赖关系，也就是说，target这一个或多个的目标文件依赖于prerequisites中的文件，其生成规则定义在command中。说白一点就是说，prerequisites中如果有一个以上的文件比target文件要新的话，command所定义的命令就会被执行。这就是Makefile的规则。也就是Makefile中最核心的内容。&lt;br /&gt;&lt;br /&gt;说到底，Makefile的东西就是这样一点，好像我的这篇文档也该结束了。呵呵。还不尽然，这是Makefile的主线和核心，但要写好一个Makefile还不够，我会以后面一点一点地结合我的工作经验给你慢慢到来。内容还多着呢。：）&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、一个示例&lt;br /&gt;&lt;br /&gt;正如前面所说的，如果一个工程有3个头文件，和8个C文件，我们为了完成前面所述的那三个规则，我们的Makefile应该是下面的这个样子的。&lt;br /&gt;&lt;br /&gt;edit : main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;cc -o edit main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;main.o : main.c defs.h&lt;br /&gt;cc -c main.c&lt;br /&gt;kbd.o : kbd.c defs.h command.h&lt;br /&gt;cc -c kbd.c&lt;br /&gt;command.o : command.c defs.h command.h&lt;br /&gt;cc -c command.c&lt;br /&gt;display.o : display.c defs.h buffer.h&lt;br /&gt;cc -c display.c&lt;br /&gt;insert.o : insert.c defs.h buffer.h&lt;br /&gt;cc -c insert.c&lt;br /&gt;search.o : search.c defs.h buffer.h&lt;br /&gt;cc -c search.c&lt;br /&gt;files.o : files.c defs.h buffer.h command.h&lt;br /&gt;cc -c files.c&lt;br /&gt;utils.o : utils.c defs.h&lt;br /&gt;cc -c utils.c&lt;br /&gt;clean :&lt;br /&gt;rm edit main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;反斜杠（/）是换行符的意思。这样比较便于Makefile的易读。我们可以把这个内容保存在文件为&amp;ldquo;Makefile&amp;rdquo;或&amp;ldquo;makefile&amp;rdquo;的文件中，然后在该目录下直接输入命令&amp;ldquo;make&amp;rdquo;就可以生成执行文件edit。如果要删除执行文件和所有的中间目标文件，那么，只要简单地执行一下&amp;ldquo;make clean&amp;rdquo;就可以了。&lt;br /&gt;&lt;br /&gt;在这个makefile中，目标文件（target）包含：执行文件edit和中间目标文件（*.o），依赖文件（prerequisites）就是冒号后面的那些 .c 文件和 .h文件。每一个 .o 文件都有一组依赖文件，而这些 .o 文件又是执行文件 edit 的依赖文件。依赖关系的实质上就是说明了目标文件是由哪些文件生成的，换言之，目标文件是哪些文件更新的。&lt;br /&gt;&lt;br /&gt;在定义好依赖关系后，后续的那一行定义了如何生成目标文件的操作系统命令，一定要以一个Tab键作为开头。记住，make并不管命令是怎么工作的，他只管执行所定义的命令。make会比较targets文件和prerequisites文件的修改日期，如果prerequisites文件的日期要比targets文件的日期要新，或者target不存在的话，那么，make就会执行后续定义的命令。&lt;br /&gt;&lt;br /&gt;这里要说明一点的是，clean不是一个文件，它只不过是一个动作名字，有点像C语言中的lable一样，其冒号后什么也没有，那么，make就不会自动去找文件的依赖性，也就不会自动执行其后所定义的命令。要执行其后的命令，就要在make命令后明显得指出这个lable的名字。这样的方法非常有用，我们可以在一个makefile中定义不用的编译或是和编译无关的命令，比如程序的打包，程序的备份，等等。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、make是如何工作的&lt;br /&gt;&lt;br /&gt;在默认的方式下，也就是我们只输入make命令。那么，&lt;br /&gt;&lt;br /&gt;1、make会在当前目录下找名字叫&amp;ldquo;Makefile&amp;rdquo;或&amp;ldquo;makefile&amp;rdquo;的文件。&lt;br /&gt;2、如果找到，它会找文件中的第一个目标文件（target），在上面的例子中，他会找到&amp;ldquo;edit&amp;rdquo;这个文件，并把这个文件作为最终的目标文件。&lt;br /&gt;3、如果edit文件不存在，或是edit所依赖的后面的 .o 文件的文件修改时间要比edit这个文件新，那么，他就会执行后面所定义的命令来生成edit这个文件。&lt;br /&gt;4、如果edit所依赖的.o文件也不存在，那么make会在当前文件中找目标为.o文件的依赖性，如果找到则再根据那一个规则生成.o文件。（这有点像一个堆栈的过程）&lt;br /&gt;5、当然，你的C文件和H文件是存在的啦，于是make会生成 .o 文件，然后再用 .o 文件生命make的终极任务，也就是执行文件edit了。&lt;br /&gt;&lt;br /&gt;这就是整个make的依赖性，make会一层又一层地去找文件的依赖关系，直到最终编译出第一个目标文件。在找寻的过程中，如果出现错误，比如最后被依赖的文件找不到，那么make就会直接退出，并报错，而对于所定义的命令的错误，或是编译不成功，make根本不理。make只管文件的依赖性，即，如果在我找了依赖关系之后，冒号后面的文件还是不在，那么对不起，我就不工作啦。&lt;br /&gt;&lt;br /&gt;通过上述分析，我们知道，像clean这种，没有被第一个目标文件直接或间接关联，那么它后面所定义的命令将不会被自动执行，不过，我们可以显示要make执行。即命令&amp;mdash;&amp;mdash;&amp;ldquo;make clean&amp;rdquo;，以此来清除所有的目标文件，以便重编译。&lt;br /&gt;&lt;br /&gt;于是在我们编程中，如果这个工程已被编译过了，当我们修改了其中一个源文件，比如file.c，那么根据我们的依赖性，我们的目标file.o会被重编译（也就是在这个依性关系后面所定义的命令），于是file.o的文件也是最新的啦，于是file.o的文件修改时间要比edit要新，所以edit也会被重新链接了（详见edit目标文件后定义的命令）。&lt;br /&gt;&lt;br /&gt;而如果我们改变了&amp;ldquo;command.h&amp;rdquo;，那么，kdb.o、command.o和files.o都会被重编译，并且，edit会被重链接。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、makefile中使用变量&lt;br /&gt;&lt;br /&gt;在上面的例子中，先让我们看看edit的规则：&lt;br /&gt;&lt;br /&gt;edit : main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;cc -o edit main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;我们可以看到[.o]文件的字符串被重复了两次，如果我们的工程需要加入一个新的[.o]文件，那么我们需要在两个地方加（应该是三个地方，还有一个地方在clean中）。当然，我们的makefile并不复杂，所以在两个地方加也不累，但如果makefile变得复杂，那么我们就有可能会忘掉一个需要加入的地方，而导致编译失败。所以，为了makefile的易维护，在makefile中我们可以使用变量。makefile的变量也就是一个字符串，理解成C语言中的宏可能会更好。&lt;br /&gt;&lt;br /&gt;比如，我们声明一个变量，叫objects, OBJECTS, objs, OBJS, obj, 或是 OBJ，反正不管什么啦，只要能够表示obj文件就行了。我们在makefile一开始就这样定义：&lt;br /&gt;&lt;br /&gt;objects = main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;于是，我们就可以很方便地在我们的makefile中以&amp;ldquo;$(objects)&amp;rdquo;的方式来使用这个变量了，于是我们的改良版makefile就变成下面这个样子：&lt;br /&gt;&lt;br /&gt;objects = main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;edit : $(objects)&lt;br /&gt;cc -o edit $(objects)&lt;br /&gt;main.o : main.c defs.h&lt;br /&gt;cc -c main.c&lt;br /&gt;kbd.o : kbd.c defs.h command.h&lt;br /&gt;cc -c kbd.c&lt;br /&gt;command.o : command.c defs.h command.h&lt;br /&gt;cc -c command.c&lt;br /&gt;display.o : display.c defs.h buffer.h&lt;br /&gt;cc -c display.c&lt;br /&gt;insert.o : insert.c defs.h buffer.h&lt;br /&gt;cc -c insert.c&lt;br /&gt;search.o : search.c defs.h buffer.h&lt;br /&gt;cc -c search.c&lt;br /&gt;files.o : files.c defs.h buffer.h command.h&lt;br /&gt;cc -c files.c&lt;br /&gt;utils.o : utils.c defs.h&lt;br /&gt;cc -c utils.c&lt;br /&gt;clean :&lt;br /&gt;rm edit $(objects)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;于是如果有新的 .o 文件加入，我们只需简单地修改一下 objects 变量就可以了。&lt;br /&gt;&lt;br /&gt;关于变量更多的话题，我会在后续给你一一道来。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、让make自动推导&lt;br /&gt;&lt;br /&gt;GNU的make很强大，它可以自动推导文件以及文件依赖关系后面的命令，于是我们就没必要去在每一个[.o]文件后都写上类似的命令，因为，我们的make会自动识别，并自己推导命令。&lt;br /&gt;&lt;br /&gt;只要make看到一个[.o]文件，它就会自动的把[.c]文件加在依赖关系中，如果make找到一个whatever.o，那么whatever.c，就会是whatever.o的依赖文件。并且 cc -c whatever.c 也会被推导出来，于是，我们的makefile再也不用写得这么复杂。我们的是新的makefile又出炉了。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;objects = main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;edit : $(objects)&lt;br /&gt;cc -o edit $(objects)&lt;br /&gt;&lt;br /&gt;main.o : defs.h&lt;br /&gt;kbd.o : defs.h command.h&lt;br /&gt;command.o : defs.h command.h&lt;br /&gt;display.o : defs.h buffer.h&lt;br /&gt;insert.o : defs.h buffer.h&lt;br /&gt;search.o : defs.h buffer.h&lt;br /&gt;files.o : defs.h buffer.h command.h&lt;br /&gt;utils.o : defs.h&lt;br /&gt;&lt;br /&gt;.PHONY : clean&lt;br /&gt;clean :&lt;br /&gt;rm edit $(objects)&lt;br /&gt;&lt;br /&gt;这种方法，也就是make的&amp;ldquo;隐晦规则&amp;rdquo;。上面文件内容中，&amp;ldquo;.PHONY&amp;rdquo;表示，clean是个伪目标文件。&lt;br /&gt;&lt;br /&gt;关于更为详细的&amp;ldquo;隐晦规则&amp;rdquo;和&amp;ldquo;伪目标文件&amp;rdquo;，我会在后续给你一一道来。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;六、另类风格的makefile&lt;br /&gt;&lt;br /&gt;即然我们的make可以自动推导命令，那么我看到那堆[.o]和[.h]的依赖就有点不爽，那么多的重复的[.h]，能不能把其收拢起来，好吧，没有问题，这个对于make来说很容易，谁叫它提供了自动推导命令和文件的功能呢？来看看最新风格的makefile吧。&lt;br /&gt;&lt;br /&gt;objects = main.o kbd.o command.o display.o /&lt;br /&gt;insert.o search.o files.o utils.o&lt;br /&gt;&lt;br /&gt;edit : $(objects)&lt;br /&gt;cc -o edit $(objects)&lt;br /&gt;&lt;br /&gt;$(objects) : defs.h&lt;br /&gt;kbd.o command.o files.o : command.h&lt;br /&gt;display.o insert.o search.o files.o : buffer.h&lt;br /&gt;&lt;br /&gt;.PHONY : clean&lt;br /&gt;clean :&lt;br /&gt;rm edit $(objects)&lt;br /&gt;&lt;br /&gt;这种风格，让我们的makefile变得很简单，但我们的文件依赖关系就显得有点凌乱了。鱼和熊掌不可兼得。还看你的喜好了。我是不喜欢这种风格的，一是文件的依赖关系看不清楚，二是如果文件一多，要加入几个新的.o文件，那就理不清楚了。&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&lt;font face=&quot;verdana, arial, helvetica&quot; size=&quot;2&quot;&gt;&lt;span class=&quot;javascript&quot; id=&quot;text1427648&quot; style=&quot;FONT-SIZE: 12px&quot;&gt;七、清空目标文件的规则&lt;br /&gt;&lt;br /&gt;每个Makefile中都应该写一个清空目标文件（.o和执行文件）的规则，这不仅便于重编译，也很利于保持文件的清洁。这是一个&amp;ldquo;修养&amp;rdquo;（呵呵，还记得我的《编程修养》吗）。一般的风格都是：&lt;br /&gt;&lt;br /&gt;clean:&lt;br /&gt;rm edit $(objects)&lt;br /&gt;&lt;br /&gt;更为稳健的做法是：&lt;br /&gt;&lt;br /&gt;.PHONY : clean&lt;br /&gt;clean :&lt;br /&gt;-rm edit $(objects)&lt;br /&gt;&lt;br /&gt;前面说过，.PHONY意思表示clean是一个&amp;ldquo;伪目标&amp;rdquo;，。而在rm命令前面加了一个小减号的意思就是，也许某些文件出现问题，但不要管，继续做后面的事。当然，clean的规则不要放在文件的开头，不然，这就会变成make的默认目标，相信谁也不愿意这样。不成文的规矩是&amp;mdash;&amp;mdash;&amp;ldquo;clean从来都是放在文件的最后&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;上面就是一个makefile的概貌，也是makefile的基础，下面还有很多makefile的相关细节，准备好了吗？准备好了就来。&lt;br /&gt;&lt;br /&gt;Makefile 总述&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;一、Makefile里有什么？&lt;br /&gt;&lt;br /&gt;Makefile里主要包含了五个东西：显式规则、隐晦规则、变量定义、文件指示和注释。&lt;br /&gt;&lt;br /&gt;1、显式规则。显式规则说明了，如何生成一个或多的的目标文件。这是由Makefile的书写者明显指出，要生成的文件，文件的依赖文件，生成的命令。&lt;br /&gt;&lt;br /&gt;2、隐晦规则。由于我们的make有自动推导的功能，所以隐晦的规则可以让我们比较粗糙地简略地书写Makefile，这是由make所支持的。&lt;br /&gt;&lt;br /&gt;3、变量的定义。在Makefile中我们要定义一系列的变量，变量一般都是字符串，这个有点你C语言中的宏，当Makefile被执行时，其中的变量都会被扩展到相应的引用位置上。&lt;br /&gt;&lt;br /&gt;4、文件指示。其包括了三个部分，一个是在一个Makefile中引用另一个Makefile，就像C语言中的include一样；另一个是指根据某些情况指定Makefile中的有效部分，就像C语言中的预编译#if一样；还有就是定义一个多行的命令。有关这一部分的内容，我会在后续的部分中讲述。&lt;br /&gt;&lt;br /&gt;5、注释。Makefile中只有行注释，和UNIX的Shell脚本一样，其注释是用&amp;ldquo;#&amp;rdquo;字符，这个就像C/C++中的&amp;ldquo;//&amp;rdquo;一样。如果你要在你的Makefile中使用&amp;ldquo;#&amp;rdquo;字符，可以用反斜框进行转义，如：&amp;ldquo;/#&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;最后，还值得一提的是，在Makefile中的命令，必须要以[Tab]键开始。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、Makefile的文件名&lt;br /&gt;&lt;br /&gt;默认的情况下，make命令会在当前目录下按顺序找寻文件名为&amp;ldquo;GNUmakefile&amp;rdquo;、&amp;ldquo;makefile&amp;rdquo;、&amp;ldquo;Makefile&amp;rdquo;的文件，找到了解释这个文件。在这三个文件名中，最好使用&amp;ldquo;Makefile&amp;rdquo;这个文件名，因为，这个文件名第一个字符为大写，这样有一种显目的感觉。最好不要用&amp;ldquo;GNUmakefile&amp;rdquo;，这个文件是GNU的make识别的。有另外一些make只对全小写的&amp;ldquo;makefile&amp;rdquo;文件名敏感，但是基本上来说，大多数的make都支持&amp;ldquo;makefile&amp;rdquo;和&amp;ldquo;Makefile&amp;rdquo;这两种默认文件名。&lt;br /&gt;&lt;br /&gt;当然，你可以使用别的文件名来书写Makefile，比如：&amp;ldquo;Make.Linux&amp;rdquo;，&amp;ldquo;Make.Solaris&amp;rdquo;，&amp;ldquo;Make.AIX&amp;rdquo;等，如果要指定特定的Makefile，你可以使用make的&amp;ldquo;-f&amp;rdquo;和&amp;ldquo;--file&amp;rdquo;参数，如：make -f Make.Linux或make --file Make.AIX。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、引用其它的Makefile&lt;br /&gt;&lt;br /&gt;在Makefile使用include关键字可以把别的Makefile包含进来，这很像C语言的#include，被包含的文件会原模原样的放在当前文件的包含位置。include的语法是：&lt;br /&gt;&lt;br /&gt;include &amp;lt;filename&amp;gt;&lt;br /&gt;&lt;br /&gt;filename可以是当前操作系统Shell的文件模式（可以保含路径和通配符）&lt;br /&gt;&lt;br /&gt;在include前面可以有一些空字符，但是绝不能是[Tab]键开始。include和&amp;lt;filename&amp;gt;可以用一个或多个空格隔开。举个例子，你有这样几个Makefile：a.mk、b.mk、c.mk，还有一个文件叫foo.make，以及一个变量$(bar)，其包含了e.mk和f.mk，那么，下面的语句：&lt;br /&gt;&lt;br /&gt;include foo.make *.mk $(bar)&lt;br /&gt;&lt;br /&gt;等价于：&lt;br /&gt;&lt;br /&gt;include foo.make a.mk b.mk c.mk e.mk f.mk&lt;br /&gt;&lt;br /&gt;make命令开始时，会把找寻include所指出的其它Makefile，并把其内容安置在当前的位置。就好像C/C++的#include指令一样。如果文件都没有指定绝对路径或是相对路径的话，make会在当前目录下首先寻找，如果当前目录下没有找到，那么，make还会在下面的几个目录下找：&lt;br /&gt;&lt;br /&gt;1、如果make执行时，有&amp;ldquo;-I&amp;rdquo;或&amp;ldquo;--include-dir&amp;rdquo;参数，那么make就会在这个参数所指定的目录下去寻找。&lt;br /&gt;2、如果目录&amp;lt;prefix&amp;gt;/include（一般是：/usr/local/bin或/usr/include）存在的话，make也会去找。&lt;br /&gt;&lt;br /&gt;如果有文件没有找到的话，make会生成一条警告信息，但不会马上出现致命错误。它会继续载入其它的文件，一旦完成makefile的读取，make会再重试这些没有找到，或是不能读取的文件，如果还是不行，make才会出现一条致命信息。如果你想让make不理那些无法读取的文件，而继续执行，你可以在include前加一个减号&amp;ldquo;-&amp;rdquo;。如：&lt;br /&gt;&lt;br /&gt;-include &amp;lt;filename&amp;gt;&lt;br /&gt;其表示，无论include过程中出现什么错误，都不要报错继续执行。和其它版本make兼容的相关命令是sinclude，其作用和这一个是一样的。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、环境变量 MAKEFILES &lt;br /&gt;&lt;br /&gt;如果你的当前环境中定义了环境变量MAKEFILES，那么，make会把这个变量中的值做一个类似于include的动作。这个变量中的值是其它的Makefile，用空格分隔。只是，它和include不同的是，从这个环境变中引入的Makefile的&amp;ldquo;目标&amp;rdquo;不会起作用，如果环境变量中定义的文件发现错误，make也会不理。&lt;br /&gt;&lt;br /&gt;但是在这里我还是建议不要使用这个环境变量，因为只要这个变量一被定义，那么当你使用make时，所有的Makefile都会受到它的影响，这绝不是你想看到的。在这里提这个事，只是为了告诉大家，也许有时候你的Makefile出现了怪事，那么你可以看看当前环境中有没有定义这个变量。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、make的工作方式&lt;br /&gt;&lt;br /&gt;GNU的make工作时的执行步骤入下：（想来其它的make也是类似）&lt;br /&gt;&lt;br /&gt;1、读入所有的Makefile。&lt;br /&gt;2、读入被include的其它Makefile。&lt;br /&gt;3、初始化文件中的变量。&lt;br /&gt;4、推导隐晦规则，并分析所有规则。&lt;br /&gt;5、为所有的目标文件创建依赖关系链。&lt;br /&gt;6、根据依赖关系，决定哪些目标要重新生成。&lt;br /&gt;7、执行生成命令。&lt;br /&gt;&lt;br /&gt;1-5步为第一个阶段，6-7为第二个阶段。第一个阶段中，如果定义的变量被使用了，那么，make会把其展开在使用的位置。但make并不会完全马上展开，make使用的是拖延战术，如果变量出现在依赖关系的规则中，那么仅当这条依赖被决定要使用了，变量才会在其内部展开。&lt;br /&gt;&lt;br /&gt;当然，这个工作方式你不一定要清楚，但是知道这个方式你也会对make更为熟悉。有了这个基础，后续部分也就容易看懂了。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;书写规则&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;规则包含两个部分，一个是依赖关系，一个是生成目标的方法。&lt;br /&gt;&lt;br /&gt;在Makefile中，规则的顺序是很重要的，因为，Makefile中只应该有一个最终目标，其它的目标都是被这个目标所连带出来的，所以一定要让make知道你的最终目标是什么。一般来说，定义在Makefile中的目标可能会有很多，但是第一条规则中的目标将被确立为最终的目标。如果第一条规则中的目标有很多个，那么，第一个目标会成为最终的目标。make所完成的也就是这个目标。&lt;br /&gt;&lt;br /&gt;好了，还是让我们来看一看如何书写规则。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;一、规则举例&lt;br /&gt;&lt;br /&gt;foo.o : foo.c defs.h # foo模块&lt;br /&gt;cc -c -g foo.c&lt;br /&gt;&lt;br /&gt;看到这个例子，各位应该不是很陌生了，前面也已说过，foo.o是我们的目标，foo.c和defs.h是目标所依赖的源文件，而只有一个命令&amp;ldquo;cc -c -g foo.c&amp;rdquo;（以Tab键开头）。这个规则告诉我们两件事：&lt;br /&gt;&lt;br /&gt;1、文件的依赖关系，foo.o依赖于foo.c和defs.h的文件，如果foo.c和defs.h的文件日期要比foo.o文件日期要新，或是foo.o不存在，那么依赖关系发生。&lt;br /&gt;2、如果生成（或更新）foo.o文件。也就是那个cc命令，其说明了，如何生成foo.o这个文件。（当然foo.c文件include了defs.h文件）&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、规则的语法&lt;br /&gt;&lt;br /&gt;targets : prerequisites&lt;br /&gt;command&lt;br /&gt;...&lt;br /&gt;&lt;br /&gt;或是这样： &lt;br /&gt;&lt;br /&gt;targets : prerequisites ; command&lt;br /&gt;command&lt;br /&gt;...&lt;br /&gt;&lt;br /&gt;targets是文件名，以空格分开，可以使用通配符。一般来说，我们的目标基本上是一个文件，但也有可能是多个文件。&lt;br /&gt;&lt;br /&gt;command是命令行，如果其不与&amp;ldquo;target&lt;img alt=&quot;吐舌&quot; border=&quot;0&quot; src=&quot;http://www.itpub.net/images/smilies/13.gif&quot; /&gt;rerequisites&amp;rdquo;在一行，那么，必须以[Tab键]开头，如果和prerequisites在一行，那么可以用分号做为分隔。（见上）&lt;br /&gt;&lt;br /&gt;prerequisites也就是目标所依赖的文件（或依赖目标）。如果其中的某个文件要比目标文件要新，那么，目标就被认为是&amp;ldquo;过时的&amp;rdquo;，被认为是需要重生成的。这个在前面已经讲过了。&lt;br /&gt;&lt;br /&gt;如果命令太长，你可以使用反斜框（&amp;lsquo;/&amp;rsquo;）作为换行符。make对一行上有多少个字符没有限制。规则告诉make两件事，文件的依赖关系和如何成成目标文件。&lt;br /&gt;&lt;br /&gt;一般来说，make会以UNIX的标准Shell，也就是/bin/sh来执行命令。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、在规则中使用通配符&lt;br /&gt;&lt;br /&gt;如果我们想定义一系列比较类似的文件，我们很自然地就想起使用通配符。make支持三各通配符：&amp;ldquo;*&amp;rdquo;，&amp;ldquo;?&amp;rdquo;和&amp;ldquo;[...]&amp;rdquo;。这是和Unix的B-Shell是相同的。&lt;br /&gt;&lt;br /&gt;波浪号（&amp;ldquo;~&amp;rdquo;）字符在文件名中也有比较特殊的用途。如果是&amp;ldquo;~/test&amp;rdquo;，这就表示当前用户的$HOME目录下的test目录。而&amp;ldquo;~hchen/test&amp;rdquo;则表示用户hchen的宿主目录下的test目录。（这些都是Unix下的小知识了，make也支持）而在Windows或是MS-DOS下，用户没有宿主目录，那么波浪号所指的目录则根据环境变量&amp;ldquo;HOME&amp;rdquo;而定。&lt;br /&gt;&lt;br /&gt;通配符代替了你一系列的文件，如&amp;ldquo;*.c&amp;rdquo;表示所以后缀为c的文件。一个需要我们注意的是，如果我们的文件名中有通配符，如：&amp;ldquo;*&amp;rdquo;，那么可以用转义字符&amp;ldquo;/&amp;rdquo;，如&amp;ldquo;/*&amp;rdquo;来表示真实的&amp;ldquo;*&amp;rdquo;字符，而不是任意长度的字符串。&lt;br /&gt;&lt;br /&gt;好吧，还是先来看几个例子吧：&lt;br /&gt;&lt;br /&gt;clean:&lt;br /&gt;rm -f *.o&lt;br /&gt;&lt;br /&gt;上面这个例子我不不多说了，这是操作系统Shell所支持的通配符。这是在命令中的通配符。&lt;br /&gt;&lt;br /&gt;print: *.c&lt;br /&gt;lpr -p $?&lt;br /&gt;touch print&lt;br /&gt;&lt;br /&gt;上面这个例子说明了通配符也可以在我们的规则中，目标print依赖于所有的[.c]文件。其中的&amp;ldquo;$?&amp;rdquo;是一个自动化变量，我会在后面给你讲述。&lt;br /&gt;&lt;br /&gt;objects = *.o&lt;br /&gt;&lt;br /&gt;上面这个例子，表示了，通符同样可以用在变量中。并不是说[*.o]会展开，不！objects的值就是&amp;ldquo;*.o&amp;rdquo;。Makefile中的变量其实就是C/C++中的宏。如果你要让通配符在变量中展开，也就是让objects的值是所有[.o]的文件名的集合，那么，你可以这样：&lt;br /&gt;&lt;br /&gt;objects := $(wildcard *.o)&lt;br /&gt;&lt;br /&gt;这种用法由关键字&amp;ldquo;wildcard&amp;rdquo;指出，关于Makefile的关键字，我们将在后面讨论。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、文件搜寻&lt;br /&gt;&lt;br /&gt;在一些大的工程中，有大量的源文件，我们通常的做法是把这许多的源文件分类，并存放在不同的目录中。所以，当make需要去找寻文件的依赖关系时，你可以在文件前加上路径，但最好的方法是把一个路径告诉make，让make在自动去找。&lt;br /&gt;&lt;br /&gt;Makefile文件中的特殊变量&amp;ldquo;VPATH&amp;rdquo;就是完成这个功能的，如果没有指明这个变量，make只会在当前的目录中去找寻依赖文件和目标文件。如果定义了这个变量，那么，make就会在当当前目录找不到的情况下，到所指定的目录中去找寻文件了。&lt;br /&gt;&lt;br /&gt;VPATH = src:../headers&lt;br /&gt;&lt;br /&gt;上面的的定义指定两个目录，&amp;ldquo;src&amp;rdquo;和&amp;ldquo;../headers&amp;rdquo;，make会按照这个顺序进行搜索。目录由&amp;ldquo;冒号&amp;rdquo;分隔。（当然，当前目录永远是最高优先搜索的地方）&lt;br /&gt;&lt;br /&gt;另一个设置文件搜索路径的方法是使用make的&amp;ldquo;vpath&amp;rdquo;关键字（注意，它是全小写的），这不是变量，这是一个make的关键字，这和上面提到的那个VPATH变量很类似，但是它更为灵活。它可以指定不同的文件在不同的搜索目录中。这是一个很灵活的功能。它的使用方法有三种：&lt;br /&gt;&lt;br /&gt;1、vpath &amp;lt;pattern&amp;gt; &amp;lt;directories&amp;gt;&lt;br /&gt;&lt;br /&gt;为符合模式&amp;lt;pattern&amp;gt;的文件指定搜索目录&amp;lt;directories&amp;gt;。&lt;br /&gt;&lt;br /&gt;2、vpath &amp;lt;pattern&amp;gt;&lt;br /&gt;&lt;br /&gt;清除符合模式&amp;lt;pattern&amp;gt;的文件的搜索目录。&lt;br /&gt;&lt;br /&gt;3、vpath&lt;br /&gt;&lt;br /&gt;清除所有已被设置好了的文件搜索目录。&lt;br /&gt;&lt;br /&gt;vapth使用方法中的&amp;lt;pattern&amp;gt;需要包含&amp;ldquo;%&amp;rdquo;字符。&amp;ldquo;%&amp;rdquo;的意思是匹配零或若干字符，例如，&amp;ldquo;%.h&amp;rdquo;表示所有以&amp;ldquo;.h&amp;rdquo;结尾的文件。&amp;lt;pattern&amp;gt;指定了要搜索的文件集，而&amp;lt;directories&amp;gt;则指定了&amp;lt;pattern&amp;gt;的文件集的搜索的目录。例如：&lt;br /&gt;&lt;br /&gt;vpath %.h ../headers&lt;br /&gt;&lt;br /&gt;该语句表示，要求make在&amp;ldquo;../headers&amp;rdquo;目录下搜索所有以&amp;ldquo;.h&amp;rdquo;结尾的文件。（如果某文件在当前目录没有找到的话）&lt;br /&gt;&lt;br /&gt;我们可以连续地使用vpath语句，以指定不同搜索策略。如果连续的vpath语句中出现了相同的&amp;lt;pattern&amp;gt;，或是被重复了的&amp;lt;pattern&amp;gt;，那么，make会按照vpath语句的先后顺序来执行搜索。如：&lt;br /&gt;&lt;br /&gt;vpath %.c foo&lt;br /&gt;vpath % blish&lt;br /&gt;vpath %.c bar&lt;br /&gt;&lt;br /&gt;其表示&amp;ldquo;.c&amp;rdquo;结尾的文件，先在&amp;ldquo;foo&amp;rdquo;目录，然后是&amp;ldquo;blish&amp;rdquo;，最后是&amp;ldquo;bar&amp;rdquo;目录。&lt;br /&gt;&lt;br /&gt;vpath %.c foo:bar&lt;br /&gt;vpath % blish&lt;br /&gt;&lt;br /&gt;而上面的语句则表示&amp;ldquo;.c&amp;rdquo;结尾的文件，先在&amp;ldquo;foo&amp;rdquo;目录，然后是&amp;ldquo;bar&amp;rdquo;目录，最后才是&amp;ldquo;blish&amp;rdquo;目录。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、伪目标&lt;br /&gt;&lt;br /&gt;最早先的一个例子中，我们提到过一个&amp;ldquo;clean&amp;rdquo;的目标，这是一个&amp;ldquo;伪目标&amp;rdquo;，&lt;br /&gt;&lt;br /&gt;clean:&lt;br /&gt;rm *.o temp&lt;br /&gt;&lt;br /&gt;正像我们前面例子中的&amp;ldquo;clean&amp;rdquo;一样，即然我们生成了许多文件编译文件，我们也应该提供一个清除它们的&amp;ldquo;目标&amp;rdquo;以备完整地重编译而用。 （以&amp;ldquo;make clean&amp;rdquo;来使用该目标）&lt;br /&gt;&lt;br /&gt;因为，我们并不生成&amp;ldquo;clean&amp;rdquo;这个文件。&amp;ldquo;伪目标&amp;rdquo;并不是一个文件，只是一个标签，由于&amp;ldquo;伪目标&amp;rdquo;不是文件，所以make无法生成它的依赖关系和决定它是否要执行。我们只有通过显示地指明这个&amp;ldquo;目标&amp;rdquo;才能让其生效。当然，&amp;ldquo;伪目标&amp;rdquo;的取名不能和文件名重名，不然其就失去了&amp;ldquo;伪目标&amp;rdquo;的意义了。&lt;br /&gt;&lt;br /&gt;当然，为了避免和文件重名的这种情况，我们可以使用一个特殊的标记&amp;ldquo;.PHONY&amp;rdquo;来显示地指明一个目标是&amp;ldquo;伪目标&amp;rdquo;，向make说明，不管是否有这个文件，这个目标就是&amp;ldquo;伪目标&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;.PHONY : clean&lt;br /&gt;&lt;br /&gt;只要有这个声明，不管是否有&amp;ldquo;clean&amp;rdquo;文件，要运行&amp;ldquo;clean&amp;rdquo;这个目标，只有&amp;ldquo;make clean&amp;rdquo;这样。于是整个过程可以这样写：&lt;br /&gt;&lt;br /&gt;.PHONY: clean&lt;br /&gt;clean:&lt;br /&gt;rm *.o temp&lt;br /&gt;&lt;br /&gt;伪目标一般没有依赖的文件。但是，我们也可以为伪目标指定所依赖的文件。伪目标同样可以作为&amp;ldquo;默认目标&amp;rdquo;，只要将其放在第一个。一个示例就是，如果你的Makefile需要一口气生成若干个可执行文件，但你只想简单地敲一个make完事，并且，所有的目标文件都写在一个Makefile中，那么你可以使用&amp;ldquo;伪目标&amp;rdquo;这个特性：&lt;br /&gt;&lt;br /&gt;all : prog1 prog2 prog3&lt;br /&gt;.PHONY : all&lt;br /&gt;&lt;br /&gt;prog1 : prog1.o utils.o&lt;br /&gt;cc -o prog1 prog1.o utils.o&lt;br /&gt;&lt;br /&gt;prog2 : prog2.o&lt;br /&gt;cc -o prog2 prog2.o&lt;br /&gt;&lt;br /&gt;prog3 : prog3.o sort.o utils.o&lt;br /&gt;cc -o prog3 prog3.o sort.o utils.o&lt;br /&gt;&lt;br /&gt;我们知道，Makefile中的第一个目标会被作为其默认目标。我们声明了一个&amp;ldquo;all&amp;rdquo;的伪目标，其依赖于其它三个目标。由于伪目标的特性是，总是被执行的，所以其依赖的那三个目标就总是不如&amp;ldquo;all&amp;rdquo;这个目标新。所以，其它三个目标的规则总是会被决议。也就达到了我们一口气生成多个目标的目的。&amp;ldquo;.PHONY : all&amp;rdquo;声明了&amp;ldquo;all&amp;rdquo;这个目标为&amp;ldquo;伪目标&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;随便提一句，从上面的例子我们可以看出，目标也可以成为依赖。所以，伪目标同样也可成为依赖。看下面的例子：&lt;br /&gt;&lt;br /&gt;.PHONY: cleanall cleanobj cleandiff&lt;br /&gt;&lt;br /&gt;cleanall : cleanobj cleandiff&lt;br /&gt;rm program&lt;br /&gt;&lt;br /&gt;cleanobj :&lt;br /&gt;rm *.o&lt;br /&gt;&lt;br /&gt;cleandiff :&lt;br /&gt;rm *.diff&lt;br /&gt;&lt;br /&gt;&amp;ldquo;make clean&amp;rdquo;将清除所有要被清除的文件。&amp;ldquo;cleanobj&amp;rdquo;和&amp;ldquo;cleandiff&amp;rdquo;这两个伪目标有点像&amp;ldquo;子程序&amp;rdquo;的意思。我们可以输入&amp;ldquo;make cleanall&amp;rdquo;和&amp;ldquo;make cleanobj&amp;rdquo;和&amp;ldquo;make cleandiff&amp;rdquo;命令来达到清除不同种类文件的目的。&lt;br /&gt;&lt;br /&gt;六、多目标&lt;br /&gt;&lt;br /&gt;Makefile的规则中的目标可以不止一个，其支持多目标，有可能我们的多个目标同时依赖于一个文件，并且其生成的命令大体类似。于是我们就能把其合并起来。当然，多个目标的生成规则的执行命令是同一个，这可能会可我们带来麻烦，不过好在我们的可以使用一个自动化变量&amp;ldquo;$@&amp;rdquo;（关于自动化变量，将在后面讲述），这个变量表示着目前规则中所有的目标的集合，这样说可能很抽象，还是看一个例子吧。&lt;br /&gt;&lt;br /&gt;bigoutput littleoutput : text.g&lt;br /&gt;generate text.g -$(subst output,,$@) &amp;gt; $@&lt;br /&gt;&lt;br /&gt;上述规则等价于：&lt;br /&gt;&lt;br /&gt;bigoutput : text.g&lt;br /&gt;generate text.g -big &amp;gt; bigoutput&lt;br /&gt;littleoutput : text.g&lt;br /&gt;generate text.g -little &amp;gt; littleoutput&lt;br /&gt;&lt;br /&gt;其中，-$(subst output,,$@)中的&amp;ldquo;$&amp;rdquo;表示执行一个Makefile的函数，函数名为subst，后面的为参数。关于函数，将在后面讲述。这里的这个函数是截取字符串的意思，&amp;ldquo;$@&amp;rdquo;表示目标的集合，就像一个数组，&amp;ldquo;$@&amp;rdquo;依次取出目标，并执于命令。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;七、静态模式&lt;br /&gt;&lt;br /&gt;静态模式可以更加容易地定义多目标的规则，可以让我们的规则变得更加的有弹性和灵活。我们还是先来看一下语法：&lt;br /&gt;&lt;br /&gt;&amp;lt;targets ...&amp;gt;: &amp;lt;target-pattern&amp;gt;: &amp;lt;prereq-patterns ...&amp;gt;&lt;br /&gt;&amp;lt;commands&amp;gt;&lt;br /&gt;...&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;targets定义了一系列的目标文件，可以有通配符。是目标的一个集合。&lt;br /&gt;&lt;br /&gt;target-parrtern是指明了targets的模式，也就是的目标集模式。&lt;br /&gt;&lt;br /&gt;prereq-parrterns是目标的依赖模式，它对target-parrtern形成的模式再进行一次依赖目标的定义。&lt;br /&gt;&lt;br /&gt;这样描述这三个东西，可能还是没有说清楚，还是举个例子来说明一下吧。如果我们的&amp;lt;target-parrtern&amp;gt;定义成&amp;ldquo;%.o&amp;rdquo;，意思是我们的&amp;lt;target&amp;gt;集合中都是以&amp;ldquo;.o&amp;rdquo;结尾的，而如果我们的&amp;lt;prereq-parrterns&amp;gt;定义成&amp;ldquo;%.c&amp;rdquo;，意思是对&amp;lt;target-parrtern&amp;gt;所形成的目标集进行二次定义，其计算方法是，取&amp;lt;target-parrtern&amp;gt;模式中的&amp;ldquo;%&amp;rdquo;（也就是去掉了[.o]这个结尾），并为其加上[.c]这个结尾，形成的新集合。&lt;br /&gt;&lt;br /&gt;所以，我们的&amp;ldquo;目标模式&amp;rdquo;或是&amp;ldquo;依赖模式&amp;rdquo;中都应该有&amp;ldquo;%&amp;rdquo;这个字符，如果你的文件名中有&amp;ldquo;%&amp;rdquo;那么你可以使用反斜杠&amp;ldquo;/&amp;rdquo;进行转义，来标明真实的&amp;ldquo;%&amp;rdquo;字符。&lt;br /&gt;&lt;br /&gt;看一个例子：&lt;br /&gt;&lt;br /&gt;objects = foo.o bar.o&lt;br /&gt;&lt;br /&gt;all: $(objects)&lt;br /&gt;&lt;br /&gt;$(objects): %.o: %.c&lt;br /&gt;$(CC) -c $(CFLAGS) $&amp;lt; -o $@&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;上面的例子中，指明了我们的目标从$object中获取，&amp;ldquo;%.o&amp;rdquo;表明要所有以&amp;ldquo;.o&amp;rdquo;结尾的目标，也就是&amp;ldquo;foo.o bar.o&amp;rdquo;，也就是变量$object集合的模式，而依赖模式&amp;ldquo;%.c&amp;rdquo;则取模式&amp;ldquo;%.o&amp;rdquo;的&amp;ldquo;%&amp;rdquo;，也就是&amp;ldquo;foo bar&amp;rdquo;，并为其加下&amp;ldquo;.c&amp;rdquo;的后缀，于是，我们的依赖目标就是&amp;ldquo;foo.c bar.c&amp;rdquo;。而命令中的&amp;ldquo;$&amp;lt;&amp;rdquo;和&amp;ldquo;$@&amp;rdquo;则是自动化变量，&amp;ldquo;$&amp;lt;&amp;rdquo;表示所有的依赖目标集（也就是&amp;ldquo;foo.c bar.c&amp;rdquo;），&amp;ldquo;$@&amp;rdquo;表示目标集（也就是&amp;ldquo;foo.o bar.o&amp;rdquo;）。于是，上面的规则展开后等价于下面的规则：&lt;br /&gt;&lt;br /&gt;foo.o : foo.c&lt;br /&gt;$(CC) -c $(CFLAGS) foo.c -o foo.o&lt;br /&gt;bar.o : bar.c&lt;br /&gt;$(CC) -c $(CFLAGS) bar.c -o bar.o&lt;br /&gt;&lt;br /&gt;试想，如果我们的&amp;ldquo;%.o&amp;rdquo;有几百个，那种我们只要用这种很简单的&amp;ldquo;静态模式规则&amp;rdquo;就可以写完一堆规则，实在是太有效率了。&amp;ldquo;静态模式规则&amp;rdquo;的用法很灵活，如果用得好，那会一个很强大的功能。再看一个例子：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;files = foo.elc bar.o lose.o&lt;br /&gt;&lt;br /&gt;$(filter %.o,$(files)): %.o: %.c&lt;br /&gt;$(CC) -c $(CFLAGS) $&amp;lt; -o $@&lt;br /&gt;$(filter %.elc,$(files)): %.elc: %.el&lt;br /&gt;emacs -f batch-byte-compile $&amp;lt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(filter %.o,$(files))表示调用Makefile的filter函数，过滤&amp;ldquo;$filter&amp;rdquo;集，只要其中模式为&amp;ldquo;%.o&amp;rdquo;的内容。其的它内容，我就不用多说了吧。这个例字展示了Makefile中更大的弹性。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;八、自动生成依赖性&lt;br /&gt;&lt;br /&gt;在Makefile中，我们的依赖关系可能会需要包含一系列的头文件，比如，如果我们的main.c中有一句&amp;ldquo;#include &amp;quot;defs.h&amp;quot;&amp;rdquo;，那么我们的依赖关系应该是：&lt;br /&gt;&lt;br /&gt;main.o : main.c defs.h&lt;br /&gt;&lt;br /&gt;但是，如果是一个比较大型的工程，你必需清楚哪些C文件包含了哪些头文件，并且，你在加入或删除头文件时，也需要小心地修改Makefile，这是一个很没有维护性的工作。为了避免这种繁重而又容易出错的事情，我们可以使用C/C++编译的一个功能。大多数的C/C++编译器都支持一个&amp;ldquo;-M&amp;rdquo;的选项，即自动找寻源文件中包含的头文件，并生成一个依赖关系。例如，如果我们执行下面的命令：&lt;br /&gt;&lt;br /&gt;cc -M main.c&lt;br /&gt;&lt;br /&gt;其输出是：&lt;br /&gt;&lt;br /&gt;main.o : main.c defs.h&lt;br /&gt;&lt;br /&gt;于是由编译器自动生成的依赖关系，这样一来，你就不必再手动书写若干文件的依赖关系，而由编译器自动生成了。需要提醒一句的是，如果你使用GNU的C/C++编译器，你得用&amp;ldquo;-MM&amp;rdquo;参数，不然，&amp;ldquo;-M&amp;rdquo;参数会把一些标准库的头文件也包含进来。&lt;br /&gt;&lt;br /&gt;gcc -M main.c的输出是：&lt;br /&gt;&lt;br /&gt;main.o: main.c defs.h /usr/include/stdio.h /usr/include/features.h /&lt;br /&gt;/usr/include/sys/cdefs.h /usr/include/gnu/stubs.h /&lt;br /&gt;/usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stddef.h /&lt;br /&gt;/usr/include/bits/types.h /usr/include/bits/pthreadtypes.h /&lt;br /&gt;/usr/include/bits/sched.h /usr/include/libio.h /&lt;br /&gt;/usr/include/_G_config.h /usr/include/wchar.h /&lt;br /&gt;/usr/include/bits/wchar.h /usr/include/gconv.h /&lt;br /&gt;/usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stdarg.h /&lt;br /&gt;/usr/include/bits/stdio_lim.h&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;gcc -MM main.c的输出则是：&lt;br /&gt;&lt;br /&gt;main.o: main.c defs.h&lt;br /&gt;&lt;br /&gt;那么，编译器的这个功能如何与我们的Makefile联系在一起呢。因为这样一来，我们的Makefile也要根据这些源文件重新生成，让Makefile自已依赖于源文件？这个功能并不现实，不过我们可以有其它手段来迂回地实现这一功能。GNU组织建议把编译器为每一个源文件的自动生成的依赖关系放到一个文件中，为每一个&amp;ldquo;name.c&amp;rdquo;的文件都生成一个&amp;ldquo;name.d&amp;rdquo;的Makefile文件，[.d]文件中就存放对应[.c]文件的依赖关系。&lt;br /&gt;&lt;br /&gt;于是，我们可以写出[.c]文件和[.d]文件的依赖关系，并让make自动更新或自成[.d]文件，并把其包含在我们的主Makefile中，这样，我们就可以自动化地生成每个文件的依赖关系了。&lt;br /&gt;&lt;br /&gt;这里，我们给出了一个模式规则来产生[.d]文件：&lt;br /&gt;&lt;br /&gt;%.d: %.c&lt;br /&gt;@set -e; rm -f $@; /&lt;br /&gt;$(CC) -M $(CPPFLAGS) $&amp;lt; &amp;gt; $@.$$$$; /&lt;br /&gt;sed 's,/($*/)/.o[ :]*,/1.o $@ : ,g' &amp;lt; $@.$$$$ &amp;gt; $@; /&lt;br /&gt;rm -f $@.$$$$&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;这个规则的意思是，所有的[.d]文件依赖于[.c]文件，&amp;ldquo;rm -f $@&amp;rdquo;的意思是删除所有的目标，也就是[.d]文件，第二行的意思是，为每个依赖文件&amp;ldquo;$&amp;lt;&amp;rdquo;，也就是[.c]文件生成依赖文件，&amp;ldquo;$@&amp;rdquo;表示模式&amp;ldquo;%.d&amp;rdquo;文件，如果有一个C文件是name.c，那么&amp;ldquo;%&amp;rdquo;就是&amp;ldquo;name&amp;rdquo;，&amp;ldquo;$$$$&amp;rdquo;意为一个随机编号，第二行生成的文件有可能是&amp;ldquo;name.d.12345&amp;rdquo;，第三行使用sed命令做了一个替换，关于sed命令的用法请参看相关的使用文档。第四行就是删除临时文件。&lt;br /&gt;&lt;br /&gt;总而言之，这个模式要做的事就是在编译器生成的依赖关系中加入[.d]文件的依赖，即把依赖关系：&lt;br /&gt;&lt;br /&gt;main.o : main.c defs.h&lt;br /&gt;&lt;br /&gt;转成：&lt;br /&gt;&lt;br /&gt;main.o main.d : main.c defs.h&lt;br /&gt;&lt;br /&gt;于是，我们的[.d]文件也会自动更新了，并会自动生成了，当然，你还可以在这个[.d]文件中加入的不只是依赖关系，包括生成的命令也可一并加入，让每个[.d]文件都包含一个完赖的规则。一旦我们完成这个工作，接下来，我们就要把这些自动生成的规则放进我们的主Makefile中。我们可以使用Makefile的&amp;ldquo;include&amp;rdquo;命令，来引入别的Makefile文件（前面讲过），例如：&lt;br /&gt;&lt;br /&gt;sources = foo.c bar.c&lt;br /&gt;&lt;br /&gt;include $(sources:.c=.d)&lt;br /&gt;&lt;br /&gt;上述语句中的&amp;ldquo;$(sources:.c=.d)&amp;rdquo;中的&amp;ldquo;.c=.d&amp;rdquo;的意思是做一个替换，把变量$(sources)所有[.c]的字串都替换成[.d]，关于这个&amp;ldquo;替换&amp;rdquo;的内容，在后面我会有更为详细的讲述。当然，你得注意次序，因为include是按次来载入文件，最先载入的[.d]文件中的目标会成为默认目标。&lt;/span&gt;&lt;/font&gt;&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&lt;/p&gt;
&lt;p&gt;&lt;font face=&quot;verdana, arial, helvetica&quot; size=&quot;2&quot;&gt;&lt;span class=&quot;javascript&quot; id=&quot;text1427649&quot; style=&quot;FONT-SIZE: 12px&quot;&gt;书写命令&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;每条规则中的命令和操作系统Shell的命令行是一致的。make会一按顺序一条一条的执行命令，每条命令的开头必须以[Tab]键开头，除非，命令是紧跟在依赖规则后面的分号后的。在命令行之间中的空格或是空行会被忽略，但是如果该空格或空行是以Tab键开头的，那么make会认为其是一个空命令。&lt;br /&gt;&lt;br /&gt;我们在UNIX下可能会使用不同的Shell，但是make的命令默认是被&amp;ldquo;/bin/sh&amp;rdquo;&amp;mdash;&amp;mdash;UNIX的标准Shell解释执行的。除非你特别指定一个其它的Shell。Makefile中，&amp;ldquo;#&amp;rdquo;是注释符，很像C/C++中的&amp;ldquo;//&amp;rdquo;，其后的本行字符都被注释。&lt;br /&gt;&lt;br /&gt;一、显示命令&lt;br /&gt;&lt;br /&gt;通常，make会把其要执行的命令行在命令执行前输出到屏幕上。当我们用&amp;ldquo;@&amp;rdquo;字符在命令行前，那么，这个命令将不被make显示出来，最具代表性的例子是，我们用这个功能来像屏幕显示一些信息。如：&lt;br /&gt;&lt;br /&gt;@echo 正在编译XXX模块......&lt;br /&gt;&lt;br /&gt;当make执行时，会输出&amp;ldquo;正在编译XXX模块......&amp;rdquo;字串，但不会输出命令，如果没有&amp;ldquo;@&amp;rdquo;，那么，make将输出：&lt;br /&gt;&lt;br /&gt;echo 正在编译XXX模块......&lt;br /&gt;正在编译XXX模块......&lt;br /&gt;&lt;br /&gt;如果make执行时，带入make参数&amp;ldquo;-n&amp;rdquo;或&amp;ldquo;--just-print&amp;rdquo;，那么其只是显示命令，但不会执行命令，这个功能很有利于我们调试我们的Makefile，看看我们书写的命令是执行起来是什么样子的或是什么顺序的。&lt;br /&gt;&lt;br /&gt;而make参数&amp;ldquo;-s&amp;rdquo;或&amp;ldquo;--slient&amp;rdquo;则是全面禁止命令的显示。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、命令执行&lt;br /&gt;&lt;br /&gt;当依赖目标新于目标时，也就是当规则的目标需要被更新时，make会一条一条的执行其后的命令。需要注意的是，如果你要让上一条命令的结果应用在下一条命令时，你应该使用分号分隔这两条命令。比如你的第一条命令是cd命令，你希望第二条命令得在cd之后的基础上运行，那么你就不能把这两条命令写在两行上，而应该把这两条命令写在一行上，用分号分隔。如：&lt;br /&gt;&lt;br /&gt;示例一：&lt;br /&gt;exec:&lt;br /&gt;cd /home/hchen&lt;br /&gt;pwd&lt;br /&gt;&lt;br /&gt;示例二：&lt;br /&gt;exec:&lt;br /&gt;cd /home/hchen; pwd&lt;br /&gt;&lt;br /&gt;当我们执行&amp;ldquo;make exec&amp;rdquo;时，第一个例子中的cd没有作用，pwd会打印出当前的Makefile目录，而第二个例子中，cd就起作用了，pwd会打印出&amp;ldquo;/home/hchen&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;make一般是使用环境变量SHELL中所定义的系统Shell来执行命令，默认情况下使用UNIX的标准Shell&amp;mdash;&amp;mdash;/bin/sh来执行命令。但在MS-DOS下有点特殊，因为MS-DOS下没有SHELL环境变量，当然你也可以指定。如果你指定了UNIX风格的目录形式，首先，make会在SHELL所指定的路径中找寻命令解释器，如果找不到，其会在当前盘符中的当前目录中寻找，如果再找不到，其会在PATH环境变量中所定义的所有路径中寻找。MS-DOS中，如果你定义的命令解释器没有找到，其会给你的命令解释器加上诸如&amp;ldquo;.exe&amp;rdquo;、&amp;ldquo;.com&amp;rdquo;、&amp;ldquo;.bat&amp;rdquo;、&amp;ldquo;.sh&amp;rdquo;等后缀。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、命令出错&lt;br /&gt;&lt;br /&gt;每当命令运行完后，make会检测每个命令的返回码，如果命令返回成功，那么make会执行下一条命令，当规则中所有的命令成功返回后，这个规则就算是成功完成了。如果一个规则中的某个命令出错了（命令退出码非零），那么make就会终止执行当前规则，这将有可能终止所有规则的执行。&lt;br /&gt;&lt;br /&gt;有些时候，命令的出错并不表示就是错误的。例如mkdir命令，我们一定需要建立一个目录，如果目录不存在，那么mkdir就成功执行，万事大吉，如果目录存在，那么就出错了。我们之所以使用mkdir的意思就是一定要有这样的一个目录，于是我们就不希望mkdir出错而终止规则的运行。&lt;br /&gt;&lt;br /&gt;为了做到这一点，忽略命令的出错，我们可以在Makefile的命令行前加一个减号&amp;ldquo;-&amp;rdquo;（在Tab键之后），标记为不管命令出不出错都认为是成功的。如：&lt;br /&gt;&lt;br /&gt;clean:&lt;br /&gt;-rm -f *.o&lt;br /&gt;&lt;br /&gt;还有一个全局的办法是，给make加上&amp;ldquo;-i&amp;rdquo;或是&amp;ldquo;--ignore-errors&amp;rdquo;参数，那么，Makefile中所有命令都会忽略错误。而如果一个规则是以&amp;ldquo;.IGNORE&amp;rdquo;作为目标的，那么这个规则中的所有命令将会忽略错误。这些是不同级别的防止命令出错的方法，你可以根据你的不同喜欢设置。&lt;br /&gt;&lt;br /&gt;还有一个要提一下的make的参数的是&amp;ldquo;-k&amp;rdquo;或是&amp;ldquo;--keep-going&amp;rdquo;，这个参数的意思是，如果某规则中的命令出错了，那么就终目该规则的执行，但继续执行其它规则。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、嵌套执行make&lt;br /&gt;&lt;br /&gt;在一些大的工程中，我们会把我们不同模块或是不同功能的源文件放在不同的目录中，我们可以在每个目录中都书写一个该目录的Makefile，这有利于让我们的Makefile变得更加地简洁，而不至于把所有的东西全部写在一个Makefile中，这样会很难维护我们的Makefile，这个技术对于我们模块编译和分段编译有着非常大的好处。&lt;br /&gt;&lt;br /&gt;例如，我们有一个子目录叫subdir，这个目录下有个Makefile文件，来指明了这个目录下文件的编译规则。那么我们总控的Makefile可以这样书写：&lt;br /&gt;&lt;br /&gt;subsystem:&lt;br /&gt;cd subdir &amp;amp;&amp;amp; $(MAKE)&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;subsystem:&lt;br /&gt;$(MAKE) -C subdir&lt;br /&gt;&lt;br /&gt;定义$(MAKE)宏变量的意思是，也许我们的make需要一些参数，所以定义成一个变量比较利于维护。这两个例子的意思都是先进入&amp;ldquo;subdir&amp;rdquo;目录，然后执行make命令。&lt;br /&gt;&lt;br /&gt;我们把这个Makefile叫做&amp;ldquo;总控Makefile&amp;rdquo;，总控Makefile的变量可以传递到下级的Makefile中（如果你显示的声明），但是不会覆盖下层的Makefile中所定义的变量，除非指定了&amp;ldquo;-e&amp;rdquo;参数。&lt;br /&gt;&lt;br /&gt;如果你要传递变量到下级Makefile中，那么你可以使用这样的声明：&lt;br /&gt;&lt;br /&gt;export &amp;lt;variable ...&amp;gt;&lt;br /&gt;&lt;br /&gt;如果你不想让某些变量传递到下级Makefile中，那么你可以这样声明： &lt;br /&gt;&lt;br /&gt;unexport &amp;lt;variable ...&amp;gt;&lt;br /&gt;&lt;br /&gt;如：&lt;br /&gt;&lt;br /&gt;示例一：&lt;br /&gt;&lt;br /&gt;export variable = value&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;variable = value&lt;br /&gt;export variable&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;export variable := value&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;variable := value&lt;br /&gt;export variable&lt;br /&gt;&lt;br /&gt;示例二：&lt;br /&gt;&lt;br /&gt;export variable += value&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;variable += value&lt;br /&gt;export variable&lt;br /&gt;&lt;br /&gt;如果你要传递所有的变量，那么，只要一个export就行了。后面什么也不用跟，表示传递所有的变量。&lt;br /&gt;&lt;br /&gt;需要注意的是，有两个变量，一个是SHELL，一个是MAKEFLAGS，这两个变量不管你是否export，其总是要传递到下层Makefile中，特别是MAKEFILES变量，其中包含了make的参数信息，如果我们执行&amp;ldquo;总控Makefile&amp;rdquo;时有make参数或是在上层Makefile中定义了这个变量，那么MAKEFILES变量将会是这些参数，并会传递到下层Makefile中，这是一个系统级的环境变量。&lt;br /&gt;&lt;br /&gt;但是make命令中的有几个参数并不往下传递，它们是&amp;ldquo;-C&amp;rdquo;,&amp;ldquo;-f&amp;rdquo;,&amp;ldquo;-h&amp;rdquo;&amp;ldquo;-o&amp;rdquo;和&amp;ldquo;-W&amp;rdquo;（有关Makefile参数的细节将在后面说明），如果你不想往下层传递参数，那么，你可以这样来：&lt;br /&gt;&lt;br /&gt;subsystem:&lt;br /&gt;cd subdir &amp;amp;&amp;amp; $(MAKE) MAKEFLAGS=&lt;br /&gt;&lt;br /&gt;如果你定义了环境变量MAKEFLAGS，那么你得确信其中的选项是大家都会用到的，如果其中有&amp;ldquo;-t&amp;rdquo;,&amp;ldquo;-n&amp;rdquo;,和&amp;ldquo;-q&amp;rdquo;参数，那么将会有让你意想不到的结果，或许会让你异常地恐慌。&lt;br /&gt;&lt;br /&gt;还有一个在&amp;ldquo;嵌套执行&amp;rdquo;中比较有用的参数，&amp;ldquo;-w&amp;rdquo;或是&amp;ldquo;--print-directory&amp;rdquo;会在make的过程中输出一些信息，让你看到目前的工作目录。比如，如果我们的下级make目录是&amp;ldquo;/home/hchen/gnu/make&amp;rdquo;，如果我们使用&amp;ldquo;make -w&amp;rdquo;来执行，那么当进入该目录时，我们会看到：&lt;br /&gt;&lt;br /&gt;make: Entering directory `/home/hchen/gnu/make'.&lt;br /&gt;&lt;br /&gt;而在完成下层make后离开目录时，我们会看到：&lt;br /&gt;&lt;br /&gt;make: Leaving directory `/home/hchen/gnu/make'&lt;br /&gt;&lt;br /&gt;当你使用&amp;ldquo;-C&amp;rdquo;参数来指定make下层Makefile时，&amp;ldquo;-w&amp;rdquo;会被自动打开的。如果参数中有&amp;ldquo;-s&amp;rdquo;（&amp;ldquo;--slient&amp;rdquo;）或是&amp;ldquo;--no-print-directory&amp;rdquo;，那么，&amp;ldquo;-w&amp;rdquo;总是失效的。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、定义命令包&lt;br /&gt;&lt;br /&gt;如果Makefile中出现一些相同命令序列，那么我们可以为这些相同的命令序列定义一个变量。定义这种命令序列的语法以&amp;ldquo;define&amp;rdquo;开始，以&amp;ldquo;endef&amp;rdquo;结束，如：&lt;br /&gt;&lt;br /&gt;define run-yacc&lt;br /&gt;yacc $(firstword $^)&lt;br /&gt;mv y.tab.c $@&lt;br /&gt;endef&lt;br /&gt;&lt;br /&gt;这里，&amp;ldquo;run-yacc&amp;rdquo;是这个命令包的名字，其不要和Makefile中的变量重名。在&amp;ldquo;define&amp;rdquo;和&amp;ldquo;endef&amp;rdquo;中的两行就是命令序列。这个命令包中的第一个命令是运行Yacc程序，因为Yacc程序总是生成&amp;ldquo;y.tab.c&amp;rdquo;的文件，所以第二行的命令就是把这个文件改改名字。还是把这个命令包放到一个示例中来看看吧。&lt;br /&gt;&lt;br /&gt;foo.c : foo.y&lt;br /&gt;$(run-yacc)&lt;br /&gt;&lt;br /&gt;我们可以看见，要使用这个命令包，我们就好像使用变量一样。在这个命令包的使用中，命令包&amp;ldquo;run-yacc&amp;rdquo;中的&amp;ldquo;$^&amp;rdquo;就是&amp;ldquo;foo.y&amp;rdquo;，&amp;ldquo;$@&amp;rdquo;就是&amp;ldquo;foo.c&amp;rdquo;（有关这种以&amp;ldquo;$&amp;rdquo;开头的特殊变量，我们会在后面介绍），make在执行命令包时，命令包中的每个命令会被依次独立执行。&lt;br /&gt;&lt;br /&gt;使用变量&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;在Makefile中的定义的变量，就像是C/C++语言中的宏一样，他代表了一个文本字串，在Makefile中执行的时候其会自动原模原样地展开在所使用的地方。其与C/C++所不同的是，你可以在Makefile中改变其值。在Makefile中，变量可以使用在&amp;ldquo;目标&amp;rdquo;，&amp;ldquo;依赖目标&amp;rdquo;，&amp;ldquo;命令&amp;rdquo;或是Makefile的其它部分中。&lt;br /&gt;&lt;br /&gt;变量的命名字可以包含字符、数字，下划线（可以是数字开头），但不应该含有&amp;ldquo;:&amp;rdquo;、&amp;ldquo;#&amp;rdquo;、&amp;ldquo;=&amp;rdquo;或是空字符（空格、回车等）。变量是大小写敏感的，&amp;ldquo;foo&amp;rdquo;、&amp;ldquo;Foo&amp;rdquo;和&amp;ldquo;FOO&amp;rdquo;是三个不同的变量名。传统的Makefile的变量名是全大写的命名方式，但我推荐使用大小写搭配的变量名，如：MakeFlags。这样可以避免和系统的变量冲突，而发生意外的事情。&lt;br /&gt;&lt;br /&gt;有一些变量是很奇怪字串，如&amp;ldquo;$&amp;lt;&amp;rdquo;、&amp;ldquo;$@&amp;rdquo;等，这些是自动化变量，我会在后面介绍。&lt;br /&gt;&lt;br /&gt;一、变量的基础&lt;br /&gt;&lt;br /&gt;变量在声明时需要给予初值，而在使用时，需要给在变量名前加上&amp;ldquo;$&amp;rdquo;符号，但最好用小括号&amp;ldquo;（）&amp;rdquo;或是大括号&amp;ldquo;{}&amp;rdquo;把变量给包括起来。如果你要使用真实的&amp;ldquo;$&amp;rdquo;字符，那么你需要用&amp;ldquo;$$&amp;rdquo;来表示。&lt;br /&gt;&lt;br /&gt;变量可以使用在许多地方，如规则中的&amp;ldquo;目标&amp;rdquo;、&amp;ldquo;依赖&amp;rdquo;、&amp;ldquo;命令&amp;rdquo;以及新的变量中。先看一个例子：&lt;br /&gt;&lt;br /&gt;objects = program.o foo.o utils.o&lt;br /&gt;program : $(objects)&lt;br /&gt;cc -o program $(objects)&lt;br /&gt;&lt;br /&gt;$(objects) : defs.h&lt;br /&gt;&lt;br /&gt;变量会在使用它的地方精确地展开，就像C/C++中的宏一样，例如：&lt;br /&gt;&lt;br /&gt;foo = c&lt;br /&gt;prog.o : prog.$(foo)&lt;br /&gt;$(foo)$(foo) -$(foo) prog.$(foo)&lt;br /&gt;&lt;br /&gt;展开后得到：&lt;br /&gt;&lt;br /&gt;prog.o : prog.c&lt;br /&gt;cc -c prog.c&lt;br /&gt;&lt;br /&gt;当然，千万不要在你的Makefile中这样干，这里只是举个例子来表明Makefile中的变量在使用处展开的真实样子。可见其就是一个&amp;ldquo;替代&amp;rdquo;的原理。&lt;br /&gt;&lt;br /&gt;另外，给变量加上括号完全是为了更加安全地使用这个变量，在上面的例子中，如果你不想给变量加上括号，那也可以，但我还是强烈建议你给变量加上括号。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、变量中的变量&lt;br /&gt;&lt;br /&gt;在定义变量的值时，我们可以使用其它变量来构造变量的值，在Makefile中有两种方式来在用变量定义变量的值。&lt;br /&gt;&lt;br /&gt;先看第一种方式，也就是简单的使用&amp;ldquo;=&amp;rdquo;号，在&amp;ldquo;=&amp;rdquo;左侧是变量，右侧是变量的值，右侧变量的值可以定义在文件的任何一处，也就是说，右侧中的变量不一定非要是已定义好的值，其也可以使用后面定义的值。如：&lt;br /&gt;&lt;br /&gt;foo = $(bar)&lt;br /&gt;bar = $(ugh)&lt;br /&gt;ugh = Huh?&lt;br /&gt;&lt;br /&gt;all:&lt;br /&gt;echo $(foo)&lt;br /&gt;&lt;br /&gt;我们执行&amp;ldquo;make all&amp;rdquo;将会打出变量$(foo)的值是&amp;ldquo;Huh?&amp;rdquo;（ $(foo)的值是$(bar)，$(bar)的值是$(ugh)，$(ugh)的值是&amp;ldquo;Huh?&amp;rdquo;）可见，变量是可以使用后面的变量来定义的。&lt;br /&gt;&lt;br /&gt;这个功能有好的地方，也有不好的地方，好的地方是，我们可以把变量的真实值推到后面来定义，如：&lt;br /&gt;&lt;br /&gt;CFLAGS = $(include_dirs) -O&lt;br /&gt;include_dirs = -Ifoo -Ibar&lt;br /&gt;&lt;br /&gt;当&amp;ldquo;CFLAGS&amp;rdquo;在命令中被展开时，会是&amp;ldquo;-Ifoo -Ibar -O&amp;rdquo;。但这种形式也有不好的地方，那就是递归定义，如：&lt;br /&gt;&lt;br /&gt;CFLAGS = $(CFLAGS) -O&lt;br /&gt;&lt;br /&gt;或：&lt;br /&gt;&lt;br /&gt;A = $(B)&lt;br /&gt;B = $(A)&lt;br /&gt;&lt;br /&gt;这会让make陷入无限的变量展开过程中去，当然，我们的make是有能力检测这样的定义，并会报错。还有就是如果在变量中使用函数，那么，这种方式会让我们的make运行时非常慢，更糟糕的是，他会使用得两个make的函数&amp;ldquo;wildcard&amp;rdquo;和&amp;ldquo;shell&amp;rdquo;发生不可预知的错误。因为你不会知道这两个函数会被调用多少次。&lt;br /&gt;&lt;br /&gt;为了避免上面的这种方法，我们可以使用make中的另一种用变量来定义变量的方法。这种方法使用的是&amp;ldquo;:=&amp;rdquo;操作符，如：&lt;br /&gt;&lt;br /&gt;x := foo&lt;br /&gt;y := $(x) bar&lt;br /&gt;x := later&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;y := foo bar&lt;br /&gt;x := later&lt;br /&gt;&lt;br /&gt;值得一提的是，这种方法，前面的变量不能使用后面的变量，只能使用前面已定义好了的变量。如果是这样：&lt;br /&gt;&lt;br /&gt;y := $(x) bar&lt;br /&gt;x := foo&lt;br /&gt;&lt;br /&gt;那么，y的值是&amp;ldquo;bar&amp;rdquo;，而不是&amp;ldquo;foo bar&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;上面都是一些比较简单的变量使用了，让我们来看一个复杂的例子，其中包括了make的函数、条件表达式和一个系统变量&amp;ldquo;MAKELEVEL&amp;rdquo;的使用：&lt;br /&gt;&lt;br /&gt;ifeq (0,${MAKELEVEL})&lt;br /&gt;cur-dir := $(shell pwd)&lt;br /&gt;whoami := $(shell whoami)&lt;br /&gt;host-type := $(shell arch)&lt;br /&gt;MAKE := ${MAKE} host-type=${host-type} whoami=${whoami}&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;关于条件表达式和函数，我们在后面再说，对于系统变量&amp;ldquo;MAKELEVEL&amp;rdquo;，其意思是，如果我们的make有一个嵌套执行的动作（参见前面的&amp;ldquo;嵌套使用make&amp;rdquo;），那么，这个变量会记录了我们的当前Makefile的调用层数。&lt;br /&gt;&lt;br /&gt;下面再介绍两个定义变量时我们需要知道的，请先看一个例子，如果我们要定义一个变量，其值是一个空格，那么我们可以这样来：&lt;br /&gt;&lt;br /&gt;nullstring :=&lt;br /&gt;space := $(nullstring) # end of the line&lt;br /&gt;&lt;br /&gt;nullstring是一个Empty变量，其中什么也没有，而我们的space的值是一个空格。因为在操作符的右边是很难描述一个空格的，这里采用的技术很管用，先用一个Empty变量来标明变量的值开始了，而后面采用&amp;ldquo;#&amp;rdquo;注释符来表示变量定义的终止，这样，我们可以定义出其值是一个空格的变量。请注意这里关于&amp;ldquo;#&amp;rdquo;的使用，注释符&amp;ldquo;#&amp;rdquo;的这种特性值得我们注意，如果我们这样定义一个变量：&lt;br /&gt;&lt;br /&gt;dir := /foo/bar # directory to put the frobs in&lt;br /&gt;&lt;br /&gt;dir这个变量的值是&amp;ldquo;/foo/bar&amp;rdquo;，后面还跟了4个空格，如果我们这样使用这样变量来指定别的目录&amp;mdash;&amp;mdash;&amp;ldquo;$(dir)/file&amp;rdquo;那么就完蛋了。&lt;br /&gt;&lt;br /&gt;还有一个比较有用的操作符是&amp;ldquo;?=&amp;rdquo;，先看示例：&lt;br /&gt;&lt;br /&gt;FOO ?= bar&lt;br /&gt;&lt;br /&gt;其含义是，如果FOO没有被定义过，那么变量FOO的值就是&amp;ldquo;bar&amp;rdquo;，如果FOO先前被定义过，那么这条语将什么也不做，其等价于：&lt;br /&gt;&lt;br /&gt;ifeq ($(origin FOO), undefined)&lt;br /&gt;FOO = bar&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、变量高级用法&lt;br /&gt;&lt;br /&gt;这里介绍两种变量的高级使用方法，第一种是变量值的替换。&lt;br /&gt;&lt;br /&gt;我们可以替换变量中的共有的部分，其格式是&amp;ldquo;$(var:a=b)&amp;rdquo;或是&amp;ldquo;${var:a=b}&amp;rdquo;，其意思是，把变量&amp;ldquo;var&amp;rdquo;中所有以&amp;ldquo;a&amp;rdquo;字串&amp;ldquo;结尾&amp;rdquo;的&amp;ldquo;a&amp;rdquo;替换成&amp;ldquo;b&amp;rdquo;字串。这里的&amp;ldquo;结尾&amp;rdquo;意思是&amp;ldquo;空格&amp;rdquo;或是&amp;ldquo;结束符&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;还是看一个示例吧：&lt;br /&gt;&lt;br /&gt;foo := a.o b.o c.o&lt;br /&gt;bar := $(foo:.o=.c)&lt;br /&gt;&lt;br /&gt;这个示例中，我们先定义了一个&amp;ldquo;$(foo)&amp;rdquo;变量，而第二行的意思是把&amp;ldquo;$(foo)&amp;rdquo;中所有以&amp;ldquo;.o&amp;rdquo;字串&amp;ldquo;结尾&amp;rdquo;全部替换成&amp;ldquo;.c&amp;rdquo;，所以我们的&amp;ldquo;$(bar)&amp;rdquo;的值就是&amp;ldquo;a.c b.c c.c&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;另外一种变量替换的技术是以&amp;ldquo;静态模式&amp;rdquo;（参见前面章节）定义的，如：&lt;br /&gt;&lt;br /&gt;foo := a.o b.o c.o&lt;br /&gt;bar := $(foo:%.o=%.c)&lt;br /&gt;&lt;br /&gt;这依赖于被替换字串中的有相同的模式，模式中必须包含一个&amp;ldquo;%&amp;rdquo;字符，这个例子同样让$(bar)变量的值为&amp;ldquo;a.c b.c c.c&amp;rdquo;。 &lt;br /&gt;&lt;br /&gt;第二种高级用法是&amp;mdash;&amp;mdash;&amp;ldquo;把变量的值再当成变量&amp;rdquo;。先看一个例子：&lt;br /&gt;&lt;br /&gt;x = y&lt;br /&gt;y = z&lt;br /&gt;a := $($(x))&lt;br /&gt;&lt;br /&gt;在这个例子中，$(x)的值是&amp;ldquo;y&amp;rdquo;，所以$($(x))就是$(y)，于是$(a)的值就是&amp;ldquo;z&amp;rdquo;。（注意，是&amp;ldquo;x=y&amp;rdquo;，而不是&amp;ldquo;x=$(y)&amp;rdquo;）&lt;br /&gt;&lt;br /&gt;我们还可以使用更多的层次：&lt;br /&gt;&lt;br /&gt;x = y&lt;br /&gt;y = z&lt;br /&gt;z = u&lt;br /&gt;a := $($($(x)))&lt;br /&gt;&lt;br /&gt;这里的$(a)的值是&amp;ldquo;u&amp;rdquo;，相关的推导留给读者自己去做吧。&lt;br /&gt;&lt;br /&gt;让我们再复杂一点，使用上&amp;ldquo;在变量定义中使用变量&amp;rdquo;的第一个方式，来看一个例子：&lt;br /&gt;&lt;br /&gt;x = $(y)&lt;br /&gt;y = z&lt;br /&gt;z = Hello&lt;br /&gt;a := $($(x))&lt;br /&gt;&lt;br /&gt;这里的$($(x))被替换成了$($(y))，因为$(y)值是&amp;ldquo;z&amp;rdquo;，所以，最终结果是：a:=$(z)，也就是&amp;ldquo;Hello&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;再复杂一点，我们再加上函数：&lt;br /&gt;&lt;br /&gt;x = variable1&lt;br /&gt;variable2 := Hello&lt;br /&gt;y = $(subst 1,2,$(x))&lt;br /&gt;z = y&lt;br /&gt;a := $($($(z)))&lt;br /&gt;&lt;br /&gt;这个例子中，&amp;ldquo;$($($(z)))&amp;rdquo;扩展为&amp;ldquo;$($(y))&amp;rdquo;，而其再次被扩展为&amp;ldquo;$($(subst 1,2,$(x)))&amp;rdquo;。$(x)的值是&amp;ldquo;variable1&amp;rdquo;，subst函数把&amp;ldquo;variable1&amp;rdquo;中的所有&amp;ldquo;1&amp;rdquo;字串替换成&amp;ldquo;2&amp;rdquo;字串，于是，&amp;ldquo;variable1&amp;rdquo;变成&amp;ldquo;variable2&amp;rdquo;，再取其值，所以，最终，$(a)的值就是$(variable2)的值&amp;mdash;&amp;mdash;&amp;ldquo;Hello&amp;rdquo;。（喔，好不容易）&lt;br /&gt;&lt;br /&gt;在这种方式中，或要可以使用多个变量来组成一个变量的名字，然后再取其值：&lt;br /&gt;&lt;br /&gt;first_second = Hello&lt;br /&gt;a = first&lt;br /&gt;b = second&lt;br /&gt;all = $($a_$b)&lt;br /&gt;&lt;br /&gt;这里的&amp;ldquo;$a_$b&amp;rdquo;组成了&amp;ldquo;first_second&amp;rdquo;，于是，$(all)的值就是&amp;ldquo;Hello&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;再来看看结合第一种技术的例子：&lt;br /&gt;&lt;br /&gt;a_objects := a.o b.o c.o&lt;br /&gt;1_objects := 1.o 2.o 3.o&lt;br /&gt;&lt;br /&gt;sources := $($(a1)_objects:.o=.c)&lt;br /&gt;&lt;br /&gt;这个例子中，如果$(a1)的值是&amp;ldquo;a&amp;rdquo;的话，那么，$(sources)的值就是&amp;ldquo;a.c b.c c.c&amp;rdquo;；如果$(a1)的值是&amp;ldquo;1&amp;rdquo;，那么$(sources)的值是&amp;ldquo;1.c 2.c 3.c&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;再来看一个这种技术和&amp;ldquo;函数&amp;rdquo;与&amp;ldquo;条件语句&amp;rdquo;一同使用的例子：&lt;br /&gt;&lt;br /&gt;ifdef do_sort&lt;br /&gt;func := sort&lt;br /&gt;else&lt;br /&gt;func := strip&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;bar := a d b g q c&lt;br /&gt;&lt;br /&gt;foo := $($(func) $(bar))&lt;br /&gt;&lt;br /&gt;这个示例中，如果定义了&amp;ldquo;do_sort&amp;rdquo;，那么：foo := $(sort a d b g q c)，于是$(foo)的值就是&amp;ldquo;a b c d g q&amp;rdquo;，而如果没有定义&amp;ldquo;do_sort&amp;rdquo;，那么：foo := $(sort a d b g q c)，调用的就是strip函数。&lt;br /&gt;&lt;br /&gt;当然，&amp;ldquo;把变量的值再当成变量&amp;rdquo;这种技术，同样可以用在操作符的左边：&lt;br /&gt;&lt;br /&gt;dir = foo&lt;br /&gt;$(dir)_sources := $(wildcard $(dir)/*.c)&lt;br /&gt;define $(dir)_print&lt;br /&gt;lpr $($(dir)_sources)&lt;br /&gt;endef&lt;br /&gt;&lt;br /&gt;这个例子中定义了三个变量：&amp;ldquo;dir&amp;rdquo;，&amp;ldquo;foo_sources&amp;rdquo;和&amp;ldquo;foo_print&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、追加变量值&lt;br /&gt;&lt;br /&gt;我们可以使用&amp;ldquo;+=&amp;rdquo;操作符给变量追加值，如：&lt;br /&gt;&lt;br /&gt;objects = main.o foo.o bar.o utils.o&lt;br /&gt;objects += another.o&lt;br /&gt;&lt;br /&gt;于是，我们的$(objects)值变成：&amp;ldquo;main.o foo.o bar.o utils.o another.o&amp;rdquo;（another.o被追加进去了）&lt;br /&gt;&lt;br /&gt;使用&amp;ldquo;+=&amp;rdquo;操作符，可以模拟为下面的这种例子：&lt;br /&gt;&lt;br /&gt;objects = main.o foo.o bar.o utils.o&lt;br /&gt;objects := $(objects) another.o&lt;br /&gt;&lt;br /&gt;所不同的是，用&amp;ldquo;+=&amp;rdquo;更为简洁。&lt;br /&gt;&lt;br /&gt;如果变量之前没有定义过，那么，&amp;ldquo;+=&amp;rdquo;会自动变成&amp;ldquo;=&amp;rdquo;，如果前面有变量定义，那么&amp;ldquo;+=&amp;rdquo;会继承于前次操作的赋值符。如果前一次的是&amp;ldquo;:=&amp;rdquo;，那么&amp;ldquo;+=&amp;rdquo;会以&amp;ldquo;:=&amp;rdquo;作为其赋值符，如：&lt;br /&gt;&lt;br /&gt;variable := value&lt;br /&gt;variable += more&lt;br /&gt;&lt;br /&gt;等价于：&lt;br /&gt;&lt;br /&gt;variable := value&lt;br /&gt;variable := $(variable) more&lt;br /&gt;&lt;br /&gt;但如果是这种情况： &lt;br /&gt;&lt;br /&gt;variable = value&lt;br /&gt;variable += more&lt;br /&gt;&lt;br /&gt;由于前次的赋值符是&amp;ldquo;=&amp;rdquo;，所以&amp;ldquo;+=&amp;rdquo;也会以&amp;ldquo;=&amp;rdquo;来做为赋值，那么岂不会发生变量的递补归定义，这是很不好的，所以make会自动为我们解决这个问题，我们不必担心这个问题。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、override 指示符&lt;br /&gt;&lt;br /&gt;如果有变量是通常make的命令行参数设置的，那么Makefile中对这个变量的赋值会被忽略。如果你想在Makefile中设置这类参数的值，那么，你可以使用&amp;ldquo;override&amp;rdquo;指示符。其语法是：&lt;br /&gt;&lt;br /&gt;override &amp;lt;variable&amp;gt; = &amp;lt;value&amp;gt;&lt;br /&gt;&lt;br /&gt;override &amp;lt;variable&amp;gt; := &amp;lt;value&amp;gt;&lt;br /&gt;&lt;br /&gt;当然，你还可以追加：&lt;br /&gt;&lt;br /&gt;override &amp;lt;variable&amp;gt; += &amp;lt;more text&amp;gt;&lt;br /&gt;&lt;br /&gt;对于多行的变量定义，我们用define指示符，在define指示符前，也同样可以使用ovveride指示符，如：&lt;br /&gt;&lt;br /&gt;override define foo&lt;br /&gt;bar&lt;br /&gt;endef&lt;br /&gt;&lt;br /&gt;六、多行变量&lt;br /&gt;&lt;br /&gt;还有一种设置变量值的方法是使用define关键字。使用define关键字设置变量的值可以有换行，这有利于定义一系列的命令（前面我们讲过&amp;ldquo;命令包&amp;rdquo;的技术就是利用这个关键字）。&lt;br /&gt;&lt;br /&gt;define指示符后面跟的是变量的名字，而重起一行定义变量的值，定义是以endef关键字结束。其工作方式和&amp;ldquo;=&amp;rdquo;操作符一样。变量的值可以包含函数、命令、文字，或是其它变量。因为命令需要以[Tab]键开头，所以如果你用define定义的命令变量中没有以[Tab]键开头，那么make就不会把其认为是命令。&lt;br /&gt;&lt;br /&gt;下面的这个示例展示了define的用法：&lt;br /&gt;&lt;br /&gt;define two-lines&lt;br /&gt;echo foo&lt;br /&gt;echo $(bar)&lt;br /&gt;endef&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;七、环境变量&lt;br /&gt;&lt;br /&gt;make运行时的系统环境变量可以在make开始运行时被载入到Makefile文件中，但是如果Makefile中已定义了这个变量，或是这个变量由make命令行带入，那么系统的环境变量的值将被覆盖。（如果make指定了&amp;ldquo;-e&amp;rdquo;参数，那么，系统环境变量将覆盖Makefile中定义的变量）&lt;br /&gt;&lt;br /&gt;因此，如果我们在环境变量中设置了&amp;ldquo;CFLAGS&amp;rdquo;环境变量，那么我们就可以在所有的Makefile中使用这个变量了。这对于我们使用统一的编译参数有比较大的好处。如果Makefile中定义了CFLAGS，那么则会使用Makefile中的这个变量，如果没有定义则使用系统环境变量的值，一个共性和个性的统一，很像&amp;ldquo;全局变量&amp;rdquo;和&amp;ldquo;局部变量&amp;rdquo;的特性。&lt;br /&gt;&lt;br /&gt;当make嵌套调用时（参见前面的&amp;ldquo;嵌套调用&amp;rdquo;章节），上层Makefile中定义的变量会以系统环境变量的方式传递到下层的Makefile中。当然，默认情况下，只有通过命令行设置的变量会被传递。而定义在文件中的变量，如果要向下层Makefile传递，则需要使用exprot关键字来声明。（参见前面章节）&lt;br /&gt;&lt;br /&gt;当然，我并不推荐把许多的变量都定义在系统环境中，这样，在我们执行不用的Makefile时，拥有的是同一套系统变量，这可能会带来更多的麻烦。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;八、目标变量&lt;br /&gt;&lt;br /&gt;前面我们所讲的在Makefile中定义的变量都是&amp;ldquo;全局变量&amp;rdquo;，在整个文件，我们都可以访问这些变量。当然，&amp;ldquo;自动化变量&amp;rdquo;除外，如&amp;ldquo;$&amp;lt;&amp;rdquo;等这种类量的自动化变量就属于&amp;ldquo;规则型变量&amp;rdquo;，这种变量的值依赖于规则的目标和依赖目标的定义。&lt;br /&gt;&lt;br /&gt;当然，我样同样可以为某个目标设置局部变量，这种变量被称为&amp;ldquo;Target-specific Variable&amp;rdquo;，它可以和&amp;ldquo;全局变量&amp;rdquo;同名，因为它的作用范围只在这条规则以及连带规则中，所以其值也只在作用范围内有效。而不会影响规则链以外的全局变量的值。&lt;br /&gt;&lt;br /&gt;其语法是：&lt;br /&gt;&lt;br /&gt;&amp;lt;target ...&amp;gt; : &amp;lt;variable-assignment&amp;gt;&lt;br /&gt;&lt;br /&gt;&amp;lt;target ...&amp;gt; : overide &amp;lt;variable-assignment&amp;gt;&lt;br /&gt;&lt;br /&gt;&amp;lt;variable-assignment&amp;gt;可以是前面讲过的各种赋值表达式，如&amp;ldquo;=&amp;rdquo;、&amp;ldquo;:=&amp;rdquo;、&amp;ldquo;+=&amp;rdquo;或是&amp;ldquo;？=&amp;rdquo;。第二个语法是针对于make命令行带入的变量，或是系统环境变量。&lt;br /&gt;&lt;br /&gt;这个特性非常的有用，当我们设置了这样一个变量，这个变量会作用到由这个目标所引发的所有的规则中去。如：&lt;br /&gt;&lt;br /&gt;prog : CFLAGS = -g&lt;br /&gt;prog : prog.o foo.o bar.o&lt;br /&gt;$(CC) $(CFLAGS) prog.o foo.o bar.o&lt;br /&gt;&lt;br /&gt;prog.o : prog.c&lt;br /&gt;$(CC) $(CFLAGS) prog.c&lt;br /&gt;&lt;br /&gt;foo.o : foo.c&lt;br /&gt;$(CC) $(CFLAGS) foo.c&lt;br /&gt;&lt;br /&gt;bar.o : bar.c&lt;br /&gt;$(CC) $(CFLAGS) bar.c&lt;br /&gt;&lt;br /&gt;在这个示例中，不管全局的$(CFLAGS)的值是什么，在prog目标，以及其所引发的所有规则中（prog.o foo.o bar.o的规则），$(CFLAGS)的值都是&amp;ldquo;-g&amp;rdquo;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;九、模式变量&lt;br /&gt;&lt;br /&gt;在GNU的make中，还支持模式变量（Pattern-specific Variable），通过上面的目标变量中，我们知道，变量可以定义在某个目标上。模式变量的好处就是，我们可以给定一种&amp;ldquo;模式&amp;rdquo;，可以把变量定义在符合这种模式的所有目标上。&lt;br /&gt;&lt;br /&gt;我们知道，make的&amp;ldquo;模式&amp;rdquo;一般是至少含有一个&amp;ldquo;%&amp;rdquo;的，所以，我们可以以如下方式给所有以[.o]结尾的目标定义目标变量：&lt;br /&gt;&lt;br /&gt;%.o : CFLAGS = -O&lt;br /&gt;&lt;br /&gt;同样，模式变量的语法和&amp;ldquo;目标变量&amp;rdquo;一样：&lt;br /&gt;&lt;br /&gt;&amp;lt;pattern ...&amp;gt; : &amp;lt;variable-assignment&amp;gt;&lt;br /&gt;&lt;br /&gt;&amp;lt;pattern ...&amp;gt; : override &amp;lt;variable-assignment&amp;gt;&lt;br /&gt;&lt;br /&gt;override同样是针对于系统环境传入的变量，或是make命令行指定的变量。&lt;/span&gt;&lt;/font&gt;&lt;/p&gt;
&lt;font face=&quot;verdana, arial, helvetica&quot; size=&quot;2&quot;&gt;&lt;span class=&quot;javascript&quot; style=&quot;FONT-SIZE: 12px&quot;&gt;
&lt;p&gt;&lt;font face=&quot;verdana, arial, helvetica&quot; size=&quot;2&quot;&gt;&lt;span class=&quot;javascript&quot; id=&quot;text1427650&quot; style=&quot;FONT-SIZE: 12px&quot;&gt;使用条件判断&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;使用条件判断，可以让make根据运行时的不同情况选择不同的执行分支。条件表达式可以是比较变量的值，或是比较变量和常量的值。&lt;br /&gt;&lt;br /&gt;一、示例&lt;br /&gt;&lt;br /&gt;下面的例子，判断$(CC)变量是否&amp;ldquo;gcc&amp;rdquo;，如果是的话，则使用GNU函数编译目标。&lt;br /&gt;&lt;br /&gt;libs_for_gcc = -lgnu&lt;br /&gt;normal_libs =&lt;br /&gt;&lt;br /&gt;foo: $(objects)&lt;br /&gt;ifeq ($(CC),gcc)&lt;br /&gt;$(CC) -o foo $(objects) $(libs_for_gcc)&lt;br /&gt;else&lt;br /&gt;$(CC) -o foo $(objects) $(normal_libs)&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;可见，在上面示例的这个规则中，目标&amp;ldquo;foo&amp;rdquo;可以根据变量&amp;ldquo;$(CC)&amp;rdquo;值来选取不同的函数库来编译程序。&lt;br /&gt;&lt;br /&gt;我们可以从上面的示例中看到三个关键字：ifeq、else和endif。ifeq的意思表示条件语句的开始，并指定一个条件表达式，表达式包含两个参数，以逗号分隔，表达式以圆括号括起。else表示条件表达式为假的情况。endif表示一个条件语句的结束，任何一个条件表达式都应该以endif结束。&lt;br /&gt;&lt;br /&gt;当我们的变量$(CC)值是&amp;ldquo;gcc&amp;rdquo;时，目标foo的规则是：&lt;br /&gt;&lt;br /&gt;foo: $(objects)&lt;br /&gt;$(CC) -o foo $(objects) $(libs_for_gcc)&lt;br /&gt;&lt;br /&gt;而当我们的变量$(CC)值不是&amp;ldquo;gcc&amp;rdquo;时（比如&amp;ldquo;cc&amp;rdquo;），目标foo的规则是：&lt;br /&gt;&lt;br /&gt;foo: $(objects)&lt;br /&gt;$(CC) -o foo $(objects) $(normal_libs)&lt;br /&gt;&lt;br /&gt;当然，我们还可以把上面的那个例子写得更简洁一些：&lt;br /&gt;&lt;br /&gt;libs_for_gcc = -lgnu&lt;br /&gt;normal_libs =&lt;br /&gt;&lt;br /&gt;ifeq ($(CC),gcc)&lt;br /&gt;libs=$(libs_for_gcc)&lt;br /&gt;else&lt;br /&gt;libs=$(normal_libs)&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;foo: $(objects)&lt;br /&gt;$(CC) -o foo $(objects) $(libs)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、语法&lt;br /&gt;&lt;br /&gt;条件表达式的语法为：&lt;br /&gt;&lt;br /&gt;&amp;lt;conditional-directive&amp;gt;&lt;br /&gt;&amp;lt;text-if-true&amp;gt;&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;以及：&lt;br /&gt;&lt;br /&gt;&amp;lt;conditional-directive&amp;gt;&lt;br /&gt;&amp;lt;text-if-true&amp;gt;&lt;br /&gt;else&lt;br /&gt;&amp;lt;text-if-false&amp;gt;&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;其中&amp;lt;conditional-directive&amp;gt;表示条件关键字，如&amp;ldquo;ifeq&amp;rdquo;。这个关键字有四个。&lt;br /&gt;&lt;br /&gt;第一个是我们前面所见过的&amp;ldquo;ifeq&amp;rdquo;&lt;br /&gt;&lt;br /&gt;ifeq (&amp;lt;arg1&amp;gt;, &amp;lt;arg2&amp;gt; ) &lt;br /&gt;ifeq '&amp;lt;arg1&amp;gt;' '&amp;lt;arg2&amp;gt;' &lt;br /&gt;ifeq &amp;quot;&amp;lt;arg1&amp;gt;&amp;quot; &amp;quot;&amp;lt;arg2&amp;gt;&amp;quot; &lt;br /&gt;ifeq &amp;quot;&amp;lt;arg1&amp;gt;&amp;quot; '&amp;lt;arg2&amp;gt;' &lt;br /&gt;ifeq '&amp;lt;arg1&amp;gt;' &amp;quot;&amp;lt;arg2&amp;gt;&amp;quot; &lt;br /&gt;&lt;br /&gt;比较参数&amp;ldquo;arg1&amp;rdquo;和&amp;ldquo;arg2&amp;rdquo;的值是否相同。当然，参数中我们还可以使用make的函数。如：&lt;br /&gt;&lt;br /&gt;ifeq ($(strip $(foo)),)&lt;br /&gt;&amp;lt;text-if-empty&amp;gt;&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;这个示例中使用了&amp;ldquo;strip&amp;rdquo;函数，如果这个函数的返回值是空（Empty），那么&amp;lt;text-if-empty&amp;gt;就生效。&lt;br /&gt;&lt;br /&gt;第二个条件关键字是&amp;ldquo;ifneq&amp;rdquo;。语法是：&lt;br /&gt;&lt;br /&gt;ifneq (&amp;lt;arg1&amp;gt;, &amp;lt;arg2&amp;gt; ) &lt;br /&gt;ifneq '&amp;lt;arg1&amp;gt;' '&amp;lt;arg2&amp;gt;' &lt;br /&gt;ifneq &amp;quot;&amp;lt;arg1&amp;gt;&amp;quot; &amp;quot;&amp;lt;arg2&amp;gt;&amp;quot; &lt;br /&gt;ifneq &amp;quot;&amp;lt;arg1&amp;gt;&amp;quot; '&amp;lt;arg2&amp;gt;' &lt;br /&gt;ifneq '&amp;lt;arg1&amp;gt;' &amp;quot;&amp;lt;arg2&amp;gt;&amp;quot; &lt;br /&gt;&lt;br /&gt;其比较参数&amp;ldquo;arg1&amp;rdquo;和&amp;ldquo;arg2&amp;rdquo;的值是否相同，如果不同，则为真。和&amp;ldquo;ifeq&amp;rdquo;类似。&lt;br /&gt;&lt;br /&gt;第三个条件关键字是&amp;ldquo;ifdef&amp;rdquo;。语法是：&lt;br /&gt;&lt;br /&gt;ifdef &amp;lt;variable-name&amp;gt; &lt;br /&gt;&lt;br /&gt;如果变量&amp;lt;variable-name&amp;gt;的值非空，那到表达式为真。否则，表达式为假。当然，&amp;lt;variable-name&amp;gt;同样可以是一个函数的返回值。注意，ifdef只是测试一个变量是否有值，其并不会把变量扩展到当前位置。还是来看两个例子：&lt;br /&gt;&lt;br /&gt;示例一：&lt;br /&gt;bar =&lt;br /&gt;foo = $(bar)&lt;br /&gt;ifdef foo&lt;br /&gt;frobozz = yes&lt;br /&gt;else&lt;br /&gt;frobozz = no&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;示例二：&lt;br /&gt;foo =&lt;br /&gt;ifdef foo&lt;br /&gt;frobozz = yes&lt;br /&gt;else&lt;br /&gt;frobozz = no&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;第一个例子中，&amp;ldquo;$(frobozz)&amp;rdquo;值是&amp;ldquo;yes&amp;rdquo;，第二个则是&amp;ldquo;no&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;第四个条件关键字是&amp;ldquo;ifndef&amp;rdquo;。其语法是：&lt;br /&gt;&lt;br /&gt;ifndef &amp;lt;variable-name&amp;gt;&lt;br /&gt;&lt;br /&gt;这个我就不多说了，和&amp;ldquo;ifdef&amp;rdquo;是相反的意思。&lt;br /&gt;&lt;br /&gt;在&amp;lt;conditional-directive&amp;gt;这一行上，多余的空格是被允许的，但是不能以[Tab]键做为开始（不然就被认为是命令）。而注释符&amp;ldquo;#&amp;rdquo;同样也是安全的。&amp;ldquo;else&amp;rdquo;和&amp;ldquo;endif&amp;rdquo;也一样，只要不是以[Tab]键开始就行了。&lt;br /&gt;&lt;br /&gt;特别注意的是，make是在读取Makefile时就计算条件表达式的值，并根据条件表达式的值来选择语句，所以，你最好不要把自动化变量（如&amp;ldquo;$@&amp;rdquo;等）放入条件表达式中，因为自动化变量是在运行时才有的。&lt;br /&gt;&lt;br /&gt;而且，为了避免混乱，make不允许把整个条件语句分成两部分放在不同的文件中。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;使用函数&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;在Makefile中可以使用函数来处理变量，从而让我们的命令或是规则更为的灵活和具有智能。make所支持的函数也不算很多，不过已经足够我们的操作了。函数调用后，函数的返回值可以当做变量来使用。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;一、函数的调用语法&lt;br /&gt;&lt;br /&gt;函数调用，很像变量的使用，也是以&amp;ldquo;$&amp;rdquo;来标识的，其语法如下：&lt;br /&gt;&lt;br /&gt;$(&amp;lt;function&amp;gt; &amp;lt;arguments&amp;gt; )&lt;br /&gt;&lt;br /&gt;或是&lt;br /&gt;&lt;br /&gt;${&amp;lt;function&amp;gt; &amp;lt;arguments&amp;gt;}&lt;br /&gt;&lt;br /&gt;这里，&amp;lt;function&amp;gt;就是函数名，make支持的函数不多。&amp;lt;arguments&amp;gt;是函数的参数，参数间以逗号&amp;ldquo;,&amp;rdquo;分隔，而函数名和参数之间以&amp;ldquo;空格&amp;rdquo;分隔。函数调用以&amp;ldquo;$&amp;rdquo;开头，以圆括号或花括号把函数名和参数括起。感觉很像一个变量，是不是？函数中的参数可以使用变量，为了风格的统一，函数和变量的括号最好一样，如使用&amp;ldquo;$(subst a,b,$(x))&amp;rdquo;这样的形式，而不是&amp;ldquo;$(subst a,b,${x})&amp;rdquo;的形式。因为统一会更清楚，也会减少一些不必要的麻烦。&lt;br /&gt;&lt;br /&gt;还是来看一个示例：&lt;br /&gt;&lt;br /&gt;comma:= ,&lt;br /&gt;empty:=&lt;br /&gt;space:= $(empty) $(empty)&lt;br /&gt;foo:= a b c&lt;br /&gt;bar:= $(subst $(space),$(comma),$(foo))&lt;br /&gt;&lt;br /&gt;在这个示例中，$(comma)的值是一个逗号。$(space)使用了$(empty)定义了一个空格，$(foo)的值是&amp;ldquo;a b c&amp;rdquo;，$(bar)的定义用，调用了函数&amp;ldquo;subst&amp;rdquo;，这是一个替换函数，这个函数有三个参数，第一个参数是被替换字串，第二个参数是替换字串，第三个参数是替换操作作用的字串。这个函数也就是把$(foo)中的空格替换成逗号，所以$(bar)的值是&amp;ldquo;a,b,c&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、字符串处理函数&lt;br /&gt;&lt;br /&gt;$(subst &amp;lt;from&amp;gt;,&amp;lt;to&amp;gt;,&amp;lt;text&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：字符串替换函数&amp;mdash;&amp;mdash;subst。&lt;br /&gt;功能：把字串&amp;lt;text&amp;gt;中的&amp;lt;from&amp;gt;字符串替换成&amp;lt;to&amp;gt;。&lt;br /&gt;返回：函数返回被替换过后的字符串。&lt;br /&gt;&lt;br /&gt;示例：&lt;br /&gt;&lt;br /&gt;$(subst ee,EE,feet on the street)，&lt;br /&gt;&lt;br /&gt;把&amp;ldquo;feet on the street&amp;rdquo;中的&amp;ldquo;ee&amp;rdquo;替换成&amp;ldquo;EE&amp;rdquo;，返回结果是&amp;ldquo;fEEt on the strEEt&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(patsubst &amp;lt;pattern&amp;gt;,&amp;lt;replacement&amp;gt;,&amp;lt;text&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：模式字符串替换函数&amp;mdash;&amp;mdash;patsubst。&lt;br /&gt;功能：查找&amp;lt;text&amp;gt;中的单词（单词以&amp;ldquo;空格&amp;rdquo;、&amp;ldquo;Tab&amp;rdquo;或&amp;ldquo;回车&amp;rdquo;&amp;ldquo;换行&amp;rdquo;分隔）是否符合模式&amp;lt;pattern&amp;gt;，如果匹配的话，则以&amp;lt;replacement&amp;gt;替换。这里，&amp;lt;pattern&amp;gt;可以包括通配符&amp;ldquo;%&amp;rdquo;，表示任意长度的字串。如果&amp;lt;replacement&amp;gt;中也包含&amp;ldquo;%&amp;rdquo;，那么，&amp;lt;replacement&amp;gt;中的这个&amp;ldquo;%&amp;rdquo;将是&amp;lt;pattern&amp;gt;中的那个&amp;ldquo;%&amp;rdquo;所代表的字串。（可以用&amp;ldquo;/&amp;rdquo;来转义，以&amp;ldquo;/%&amp;rdquo;来表示真实含义的&amp;ldquo;%&amp;rdquo;字符）&lt;br /&gt;返回：函数返回被替换过后的字符串。&lt;br /&gt;&lt;br /&gt;示例：&lt;br /&gt;&lt;br /&gt;$(patsubst %.c,%.o,x.c.c bar.c)&lt;br /&gt;&lt;br /&gt;把字串&amp;ldquo;x.c.c bar.c&amp;rdquo;符合模式[%.c]的单词替换成[%.o]，返回结果是&amp;ldquo;x.c.o bar.o&amp;rdquo;&lt;br /&gt;&lt;br /&gt;备注：&lt;br /&gt;&lt;br /&gt;这和我们前面&amp;ldquo;变量章节&amp;rdquo;说过的相关知识有点相似。如：&lt;br /&gt;&lt;br /&gt;&amp;ldquo;$(var:&amp;lt;pattern&amp;gt;=&amp;lt;replacement&amp;gt; )&amp;rdquo;&lt;br /&gt;相当于&lt;br /&gt;&amp;ldquo;$(patsubst &amp;lt;pattern&amp;gt;,&amp;lt;replacement&amp;gt;,$(var))&amp;rdquo;，&lt;br /&gt;&lt;br /&gt;而&amp;ldquo;$(var: &amp;lt;suffix&amp;gt;=&amp;lt;replacement&amp;gt; )&amp;rdquo;&lt;br /&gt;则相当于&lt;br /&gt;&amp;ldquo;$(patsubst %&amp;lt;suffix&amp;gt;,%&amp;lt;replacement&amp;gt;,$(var))&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;例如有：objects = foo.o bar.o baz.o，&lt;br /&gt;那么，&amp;ldquo;$(objects:.o=.c)&amp;rdquo;和&amp;ldquo;$(patsubst %.o,%.c,$(objects))&amp;rdquo;是一样的。&lt;br /&gt;&lt;br /&gt;$(strip &amp;lt;string&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：去空格函数&amp;mdash;&amp;mdash;strip。&lt;br /&gt;功能：去掉&amp;lt;string&amp;gt;字串中开头和结尾的空字符。&lt;br /&gt;返回：返回被去掉空格的字符串值。&lt;br /&gt;示例：&lt;br /&gt;&lt;br /&gt;$(strip a b c )&lt;br /&gt;&lt;br /&gt;把字串&amp;ldquo;a b c &amp;rdquo;去到开头和结尾的空格，结果是&amp;ldquo;a b c&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(findstring &amp;lt;find&amp;gt;,&amp;lt;in&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：查找字符串函数&amp;mdash;&amp;mdash;findstring。&lt;br /&gt;功能：在字串&amp;lt;in&amp;gt;中查找&amp;lt;find&amp;gt;字串。&lt;br /&gt;返回：如果找到，那么返回&amp;lt;find&amp;gt;，否则返回空字符串。&lt;br /&gt;示例：&lt;br /&gt;&lt;br /&gt;$(findstring a,a b c)&lt;br /&gt;$(findstring a,b c)&lt;br /&gt;&lt;br /&gt;第一个函数返回&amp;ldquo;a&amp;rdquo;字符串，第二个返回&amp;ldquo;&amp;rdquo;字符串（空字符串）&lt;br /&gt;&lt;br /&gt;$(filter &amp;lt;pattern...&amp;gt;,&amp;lt;text&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：过滤函数&amp;mdash;&amp;mdash;filter。&lt;br /&gt;功能：以&amp;lt;pattern&amp;gt;模式过滤&amp;lt;text&amp;gt;字符串中的单词，保留符合模式&amp;lt;pattern&amp;gt;的单词。可以有多个模式。&lt;br /&gt;返回：返回符合模式&amp;lt;pattern&amp;gt;的字串。&lt;br /&gt;示例：&lt;br /&gt;&lt;br /&gt;sources := foo.c bar.c baz.s ugh.h&lt;br /&gt;foo: $(sources)&lt;br /&gt;cc $(filter %.c %.s,$(sources)) -o foo&lt;br /&gt;&lt;br /&gt;$(filter %.c %.s,$(sources))返回的值是&amp;ldquo;foo.c bar.c baz.s&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(filter-out &amp;lt;pattern...&amp;gt;,&amp;lt;text&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：反过滤函数&amp;mdash;&amp;mdash;filter-out。&lt;br /&gt;功能：以&amp;lt;pattern&amp;gt;模式过滤&amp;lt;text&amp;gt;字符串中的单词，去除符合模式&amp;lt;pattern&amp;gt;的单词。可以有多个模式。&lt;br /&gt;返回：返回不符合模式&amp;lt;pattern&amp;gt;的字串。&lt;br /&gt;示例：&lt;br /&gt;&lt;br /&gt;objects=main1.o foo.o main2.o bar.o&lt;br /&gt;mains=main1.o main2.o&lt;br /&gt;&lt;br /&gt;$(filter-out $(mains),$(objects)) 返回值是&amp;ldquo;foo.o bar.o&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(sort &amp;lt;list&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：排序函数&amp;mdash;&amp;mdash;sort。&lt;br /&gt;功能：给字符串&amp;lt;list&amp;gt;中的单词排序（升序）。&lt;br /&gt;返回：返回排序后的字符串。&lt;br /&gt;示例：$(sort foo bar lose)返回&amp;ldquo;bar foo lose&amp;rdquo; 。&lt;br /&gt;备注：sort函数会去掉&amp;lt;list&amp;gt;中相同的单词。&lt;br /&gt;&lt;br /&gt;$(word &amp;lt;n&amp;gt;,&amp;lt;text&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：取单词函数&amp;mdash;&amp;mdash;word。&lt;br /&gt;功能：取字符串&amp;lt;text&amp;gt;中第&amp;lt;n&amp;gt;个单词。（从一开始）&lt;br /&gt;返回：返回字符串&amp;lt;text&amp;gt;中第&amp;lt;n&amp;gt;个单词。如果&amp;lt;n&amp;gt;比&amp;lt;text&amp;gt;中的单词数要大，那么返回空字符串。&lt;br /&gt;示例：$(word 2, foo bar baz)返回值是&amp;ldquo;bar&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(wordlist &amp;lt;s&amp;gt;,&amp;lt;e&amp;gt;,&amp;lt;text&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：取单词串函数&amp;mdash;&amp;mdash;wordlist。&lt;br /&gt;功能：从字符串&amp;lt;text&amp;gt;中取从&amp;lt;s&amp;gt;开始到&amp;lt;e&amp;gt;的单词串。&amp;lt;s&amp;gt;和&amp;lt;e&amp;gt;是一个数字。&lt;br /&gt;返回：返回字符串&amp;lt;text&amp;gt;中从&amp;lt;s&amp;gt;到&amp;lt;e&amp;gt;的单词字串。如果&amp;lt;s&amp;gt;比&amp;lt;text&amp;gt;中的单词数要大，那么返回空字符串。如果&amp;lt;e&amp;gt;大于&amp;lt;text&amp;gt;的单词数，那么返回从&amp;lt;s&amp;gt;开始，到&amp;lt;text&amp;gt;结束的单词串。&lt;br /&gt;示例： $(wordlist 2, 3, foo bar baz)返回值是&amp;ldquo;bar baz&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(words &amp;lt;text&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：单词个数统计函数&amp;mdash;&amp;mdash;words。&lt;br /&gt;功能：统计&amp;lt;text&amp;gt;中字符串中的单词个数。&lt;br /&gt;返回：返回&amp;lt;text&amp;gt;中的单词数。&lt;br /&gt;示例：$(words, foo bar baz)返回值是&amp;ldquo;3&amp;rdquo;。&lt;br /&gt;备注：如果我们要取&amp;lt;text&amp;gt;中最后的一个单词，我们可以这样：$(word $(words &amp;lt;text&amp;gt; ),&amp;lt;text&amp;gt; )。&lt;br /&gt;&lt;br /&gt;$(firstword &amp;lt;text&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：首单词函数&amp;mdash;&amp;mdash;firstword。&lt;br /&gt;功能：取字符串&amp;lt;text&amp;gt;中的第一个单词。&lt;br /&gt;返回：返回字符串&amp;lt;text&amp;gt;的第一个单词。&lt;br /&gt;示例：$(firstword foo bar)返回值是&amp;ldquo;foo&amp;rdquo;。&lt;br /&gt;备注：这个函数可以用word函数来实现：$(word 1,&amp;lt;text&amp;gt; )。&lt;br /&gt;&lt;br /&gt;以上，是所有的字符串操作函数，如果搭配混合使用，可以完成比较复杂的功能。这里，举一个现实中应用的例子。我们知道，make使用&amp;ldquo;VPATH&amp;rdquo;变量来指定&amp;ldquo;依赖文件&amp;rdquo;的搜索路径。于是，我们可以利用这个搜索路径来指定编译器对头文件的搜索路径参数CFLAGS，如：&lt;br /&gt;&lt;br /&gt;override CFLAGS += $(patsubst %,-I%,$(subst :, ,$(VPATH)))&lt;br /&gt;&lt;br /&gt;如果我们的&amp;ldquo;$(VPATH)&amp;rdquo;值是&amp;ldquo;src:../headers&amp;rdquo;，那么&amp;ldquo;$(patsubst %,-I%,$(subst :, ,$(VPATH)))&amp;rdquo;将返回&amp;ldquo;-Isrc -I../headers&amp;rdquo;，这正是cc或gcc搜索头文件路径的参数。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、文件名操作函数&lt;br /&gt;&lt;br /&gt;下面我们要介绍的函数主要是处理文件名的。每个函数的参数字符串都会被当做一个或是一系列的文件名来对待。&lt;br /&gt;&lt;br /&gt;$(dir &amp;lt;names...&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：取目录函数&amp;mdash;&amp;mdash;dir。&lt;br /&gt;功能：从文件名序列&amp;lt;names&amp;gt;中取出目录部分。目录部分是指最后一个反斜杠（&amp;ldquo;/&amp;rdquo;）之前的部分。如果没有反斜杠，那么返回&amp;ldquo;./&amp;rdquo;。&lt;br /&gt;返回：返回文件名序列&amp;lt;names&amp;gt;的目录部分。&lt;br /&gt;示例： $(dir src/foo.c hacks)返回值是&amp;ldquo;src/ ./&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(notdir &amp;lt;names...&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：取文件函数&amp;mdash;&amp;mdash;notdir。&lt;br /&gt;功能：从文件名序列&amp;lt;names&amp;gt;中取出非目录部分。非目录部分是指最后一个反斜杠（&amp;ldquo;/&amp;rdquo;）之后的部分。&lt;br /&gt;返回：返回文件名序列&amp;lt;names&amp;gt;的非目录部分。&lt;br /&gt;示例： $(notdir src/foo.c hacks)返回值是&amp;ldquo;foo.c hacks&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(suffix &amp;lt;names...&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：取后缀函数&amp;mdash;&amp;mdash;suffix。&lt;br /&gt;功能：从文件名序列&amp;lt;names&amp;gt;中取出各个文件名的后缀。&lt;br /&gt;返回：返回文件名序列&amp;lt;names&amp;gt;的后缀序列，如果文件没有后缀，则返回空字串。&lt;br /&gt;示例：$(suffix src/foo.c src-1.0/bar.c hacks)返回值是&amp;ldquo;.c .c&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(basename &amp;lt;names...&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：取前缀函数&amp;mdash;&amp;mdash;basename。&lt;br /&gt;功能：从文件名序列&amp;lt;names&amp;gt;中取出各个文件名的前缀部分。&lt;br /&gt;返回：返回文件名序列&amp;lt;names&amp;gt;的前缀序列，如果文件没有前缀，则返回空字串。&lt;br /&gt;示例：$(basename src/foo.c src-1.0/bar.c hacks)返回值是&amp;ldquo;src/foo src-1.0/bar hacks&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(addsuffix &amp;lt;suffix&amp;gt;,&amp;lt;names...&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：加后缀函数&amp;mdash;&amp;mdash;addsuffix。&lt;br /&gt;功能：把后缀&amp;lt;suffix&amp;gt;加到&amp;lt;names&amp;gt;中的每个单词后面。&lt;br /&gt;返回：返回加过后缀的文件名序列。&lt;br /&gt;示例：$(addsuffix .c,foo bar)返回值是&amp;ldquo;foo.c bar.c&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(addprefix &amp;lt;prefix&amp;gt;,&amp;lt;names...&amp;gt; ) &lt;br /&gt;&lt;br /&gt;名称：加前缀函数&amp;mdash;&amp;mdash;addprefix。&lt;br /&gt;功能：把前缀&amp;lt;prefix&amp;gt;加到&amp;lt;names&amp;gt;中的每个单词后面。&lt;br /&gt;返回：返回加过前缀的文件名序列。&lt;br /&gt;示例：$(addprefix src/,foo bar)返回值是&amp;ldquo;src/foo src/bar&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;$(join &amp;lt;list1&amp;gt;,&amp;lt;list2&amp;gt; )&lt;br /&gt;&lt;br /&gt;名称：连接函数&amp;mdash;&amp;mdash;join。&lt;br /&gt;功能：把&amp;lt;list2&amp;gt;中的单词对应地加到&amp;lt;list1&amp;gt;的单词后面。如果&amp;lt;list1&amp;gt;的单词个数要比&amp;lt;list2&amp;gt;的多，那么，&amp;lt;list1&amp;gt;中的多出来的单词将保持原样。如果&amp;lt;list2&amp;gt;的单词个数要比&amp;lt;list1&amp;gt;多，那么，&amp;lt;list2&amp;gt;多出来的单词将被复制到&amp;lt;list2&amp;gt;中。&lt;br /&gt;返回：返回连接过后的字符串。&lt;br /&gt;示例：$(join aaa bbb , 111 222 333)返回值是&amp;ldquo;aaa111 bbb222 333&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、foreach 函数&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;foreach函数和别的函数非常的不一样。因为这个函数是用来做循环用的，Makefile中的foreach函数几乎是仿照于Unix标准Shell（/bin/sh）中的for语句，或是C-Shell（/bin/csh）中的foreach语句而构建的。它的语法是：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(foreach &amp;lt;var&amp;gt;,&amp;lt;list&amp;gt;,&amp;lt;text&amp;gt; )&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;这个函数的意思是，把参数&amp;lt;list&amp;gt;中的单词逐一取出放到参数&amp;lt;var&amp;gt;所指定的变量中，然后再执行&amp;lt;text&amp;gt;所包含的表达式。每一次&amp;lt;text&amp;gt;会返回一个字符串，循环过程中，&amp;lt;text&amp;gt;的所返回的每个字符串会以空格分隔，最后当整个循环结束时，&amp;lt;text&amp;gt;所返回的每个字符串所组成的整个字符串（以空格分隔）将会是foreach函数的返回值。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;所以，&amp;lt;var&amp;gt;最好是一个变量名，&amp;lt;list&amp;gt;可以是一个表达式，而&amp;lt;text&amp;gt;中一般会使用&amp;lt;var&amp;gt;这个参数来依次枚举&amp;lt;list&amp;gt;中的单词。举个例子：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;names := a b c d&lt;br /&gt;&lt;br /&gt;files := $(foreach n,$(names),$(n).o)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;上面的例子中，$(name)中的单词会被挨个取出，并存到变量&amp;ldquo;n&amp;rdquo;中，&amp;ldquo;$(n).o&amp;rdquo;每次根据&amp;ldquo;$(n)&amp;rdquo;计算出一个值，这些值以空格分隔，最后作为foreach函数的返回，所以，$(files)的值是&amp;ldquo;a.o b.o c.o d.o&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;注意，foreach中的&amp;lt;var&amp;gt;参数是一个临时的局部变量，foreach函数执行完后，参数&amp;lt;var&amp;gt;的变量将不在作用，其作用域只在foreach函数当中。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、if 函数&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;if函数很像GNU的make所支持的条件语句&amp;mdash;&amp;mdash;ifeq（参见前面所述的章节），if函数的语法是：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(if &amp;lt;condition&amp;gt;,&amp;lt;then-part&amp;gt; ) &lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;或是&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(if &amp;lt;condition&amp;gt;,&amp;lt;then-part&amp;gt;,&amp;lt;else-part&amp;gt; )&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;可见，if函数可以包含&amp;ldquo;else&amp;rdquo;部分，或是不含。即if函数的参数可以是两个，也可以是三个。&amp;lt;condition&amp;gt;参数是if的表达式，如果其返回的为非空字符串，那么这个表达式就相当于返回真，于是，&amp;lt;then-part&amp;gt;会被计算，否则&amp;lt;else-part&amp;gt;会被计算。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;而if函数的返回值是，如果&amp;lt;condition&amp;gt;为真（非空字符串），那个&amp;lt;then-part&amp;gt;会是整个函数的返回值，如果&amp;lt;condition&amp;gt;为假（空字符串），那么&amp;lt;else-part&amp;gt;会是整个函数的返回值，此时如果&amp;lt;else-part&amp;gt;没有被定义，那么，整个函数返回空字串。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;所以，&amp;lt;then-part&amp;gt;和&amp;lt;else-part&amp;gt;只会有一个被计算。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;六、call函数&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;call函数是唯一一个可以用来创建新的参数化的函数。你可以写一个非常复杂的表达式，这个表达式中，你可以定义许多参数，然后你可以用call函数来向这个表达式传递参数。其语法是：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(call &amp;lt;expression&amp;gt;,&amp;lt;parm1&amp;gt;,&amp;lt;parm2&amp;gt;,&amp;lt;parm3&amp;gt;...)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;当make执行这个函数时，&amp;lt;expression&amp;gt;参数中的变量，如$(1)，$(2)，$(3)等，会被参数&amp;lt;parm1&amp;gt;，&amp;lt;parm2&amp;gt;，&amp;lt;parm3&amp;gt;依次取代。而&amp;lt;expression&amp;gt;的返回值就是call函数的返回值。例如：&lt;br /&gt;&lt;br /&gt;reverse = $(1) $(2)&lt;br /&gt;&lt;br /&gt;foo = $(call reverse,a,b)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;那么，foo的值就是&amp;ldquo;a b&amp;rdquo;。当然，参数的次序是可以自定义的，不一定是顺序的，如：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;reverse = $(2) $(1)&lt;br /&gt;&lt;br /&gt;foo = $(call reverse,a,b)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;此时的foo的值就是&amp;ldquo;b a&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;七、origin函数&lt;br /&gt;origin函数不像其它的函数，他并不操作变量的值，他只是告诉你你的这个变量是哪里来的？其语法是：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(origin &amp;lt;variable&amp;gt; )&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;注意，&amp;lt;variable&amp;gt;是变量的名字，不应该是引用。所以你最好不要在&amp;lt;variable&amp;gt;中使用&amp;ldquo;$&amp;rdquo;字符。Origin函数会以其返回值来告诉你这个变量的&amp;ldquo;出生情况&amp;rdquo;，下面，是origin函数的返回值:&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;undefined&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;从来没有定义过，origin函数返回这个值&amp;ldquo;undefined&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;default&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;是一个默认的定义，比如&amp;ldquo;CC&amp;rdquo;这个变量，这种变量我们将在后面讲述。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;environment&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;是一个环境变量，并且当Makefile被执行时，&amp;ldquo;-e&amp;rdquo;参数没有被打开。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;file&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;这个变量被定义在Makefile中。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;command line&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;这个变量是被命令行定义的。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;override&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;是被override指示符重新定义的。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&amp;ldquo;automatic&amp;rdquo;&lt;br /&gt;&lt;br /&gt;如果&amp;lt;variable&amp;gt;是一个命令运行中的自动化变量。关于自动化变量将在后面讲述。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;这些信息对于我们编写Makefile是非常有用的，例如，假设我们有一个Makefile其包了一个定义文件Make.def，在Make.def中定义了一个变量&amp;ldquo;bletch&amp;rdquo;，而我们的环境中也有一个环境变量&amp;ldquo;bletch&amp;rdquo;，此时，我们想判断一下，如果变量来源于环境，那么我们就把之重定义了，如果来源于Make.def或是命令行等非环境的，那么我们就不重新定义它。于是，在我们的Makefile中，我们可以这样写：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;ifdef bletch&lt;br /&gt;&lt;br /&gt;ifeq &amp;quot;$(origin bletch)&amp;quot; &amp;quot;environment&amp;quot;&lt;br /&gt;&lt;br /&gt;bletch = barf, gag, etc.&lt;br /&gt;&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;当然，你也许会说，使用override关键字不就可以重新定义环境中的变量了吗？为什么需要使用这样的步骤？是的，我们用override是可以达到这样的效果，可是override过于粗暴，它同时会把从命令行定义的变量也覆盖了，而我们只想重新定义环境传来的，而不想重新定义命令行传来的。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;八、shell函数&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;shell函数也不像其它的函数。顾名思义，它的参数应该就是操作系统Shell的命令。它和反引号&amp;ldquo;`&amp;rdquo;是相同的功能。这就是说，shell函数把执行操作系统命令后的输出作为函数返回。于是，我们可以用操作系统命令以及字符串处理命令awk，sed等等命令来生成一个变量，如：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;contents := $(shell cat foo)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;files := $(shell echo *.c)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;注意，这个函数会新生成一个Shell程序来执行命令，所以你要注意其运行性能，如果你的Makefile中有一些比较复杂的规则，并大量使用了这个函数，那么对于你的系统性能是有害的。特别是Makefile的隐晦的规则可能会让你的shell函数执行的次数比你想像的多得多。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;九、控制make的函数&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;make提供了一些函数来控制make的运行。通常，你需要检测一些运行Makefile时的运行时信息，并且根据这些信息来决定，你是让make继续执行，还是停止。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(error &amp;lt;text ...&amp;gt; )&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;产生一个致命的错误，&amp;lt;text ...&amp;gt;是错误信息。注意，error函数不会在一被使用就会产生错误信息，所以如果你把其定义在某个变量中，并在后续的脚本中使用这个变量，那么也是可以的。例如：&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;示例一：&lt;br /&gt;&lt;br /&gt;ifdef ERROR_001&lt;br /&gt;&lt;br /&gt;$(error error is $(ERROR_001))&lt;br /&gt;&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;示例二：&lt;br /&gt;&lt;br /&gt;ERR = $(error found an error!)&lt;br /&gt;&lt;br /&gt;.PHONY: err&lt;br /&gt;&lt;br /&gt;err: ; $(ERR)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;示例一会在变量ERROR_001定义了后执行时产生error调用，而示例二则在目录err被执行时才发生error调用。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;$(warning &amp;lt;text ...&amp;gt; )&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;这个函数很像error函数，只是它并不会让make退出，只是输出一段警告信息，而make继续执行。&lt;br /&gt;&lt;br /&gt;make 的运行&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;一般来说，最简单的就是直接在命令行下输入make命令，make命令会找当前目录的makefile来执行，一切都是自动的。但也有时你也许只想让make重编译某些文件，而不是整个工程，而又有的时候你有几套编译规则，你想在不同的时候使用不同的编译规则，等等。本章节就是讲述如何使用make命令的。&lt;br /&gt;&lt;br /&gt;一、make的退出码&lt;br /&gt;&lt;br /&gt;make命令执行后有三个退出码：&lt;br /&gt;&lt;br /&gt;0 &amp;mdash;&amp;mdash; 表示成功执行。&lt;br /&gt;1 &amp;mdash;&amp;mdash; 如果make运行时出现任何错误，其返回1。&lt;br /&gt;2 &amp;mdash;&amp;mdash; 如果你使用了make的&amp;ldquo;-q&amp;rdquo;选项，并且make使得一些目标不需要更新，那么返回2。&lt;br /&gt;&lt;br /&gt;Make的相关参数我们会在后续章节中讲述。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、指定Makefile&lt;br /&gt;&lt;br /&gt;前面我们说过，GNU make找寻默认的Makefile的规则是在当前目录下依次找三个文件&amp;mdash;&amp;mdash;&amp;ldquo;GNUmakefile&amp;rdquo;、&amp;ldquo;makefile&amp;rdquo;和&amp;ldquo;Makefile&amp;rdquo;。其按顺序找这三个文件，一旦找到，就开始读取这个文件并执行。&lt;br /&gt;&lt;br /&gt;当前，我们也可以给make命令指定一个特殊名字的Makefile。要达到这个功能，我们要使用make的&amp;ldquo;-f&amp;rdquo;或是&amp;ldquo;--file&amp;rdquo;参数（&amp;ldquo;--makefile&amp;rdquo;参数也行）。例如，我们有个makefile的名字是&amp;ldquo;hchen.mk&amp;rdquo;，那么，我们可以这样来让make来执行这个文件：&lt;br /&gt;&lt;br /&gt;make &amp;ndash;f hchen.mk&lt;br /&gt;&lt;br /&gt;如果在make的命令行是，你不只一次地使用了&amp;ldquo;-f&amp;rdquo;参数，那么，所有指定的makefile将会被连在一起传递给make执行。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、指定目标&lt;br /&gt;&lt;br /&gt;一般来说，make的最终目标是makefile中的第一个目标，而其它目标一般是由这个目标连带出来的。这是make的默认行为。当然，一般来说，你的makefile中的第一个目标是由许多个目标组成，你可以指示make，让其完成你所指定的目标。要达到这一目的很简单，需在make命令后直接跟目标的名字就可以完成（如前面提到的&amp;ldquo;make clean&amp;rdquo;形式）&lt;br /&gt;&lt;br /&gt;任何在makefile中的目标都可以被指定成终极目标，但是除了以&amp;ldquo;-&amp;rdquo;打头，或是包含了&amp;ldquo;=&amp;rdquo;的目标，因为有这些字符的目标，会被解析成命令行参数或是变量。甚至没有被我们明确写出来的目标也可以成为make的终极目标，也就是说，只要make可以找到其隐含规则推导规则，那么这个隐含目标同样可以被指定成终极目标。&lt;br /&gt;&lt;br /&gt;有一个make的环境变量叫&amp;ldquo;MAKECMDGOALS&amp;rdquo;，这个变量中会存放你所指定的终极目标的列表，如果在命令行上，你没有指定目标，那么，这个变量是空值。这个变量可以让你使用在一些比较特殊的情形下。比如下面的例子：&lt;br /&gt;&lt;br /&gt;sources = foo.c bar.c&lt;br /&gt;ifneq ( $(MAKECMDGOALS),clean)&lt;br /&gt;include $(sources:.c=.d)&lt;br /&gt;endif&lt;br /&gt;&lt;br /&gt;基于上面的这个例子，只要我们输入的命令不是&amp;ldquo;make clean&amp;rdquo;，那么makefile会自动包含&amp;ldquo;foo.d&amp;rdquo;和&amp;ldquo;bar.d&amp;rdquo;这两个makefile。&lt;br /&gt;&lt;br /&gt;使用指定终极目标的方法可以很方便地让我们编译我们的程序，例如下面这个例子：&lt;br /&gt;&lt;br /&gt;.PHONY: all&lt;br /&gt;all: prog1 prog2 prog3 prog4&lt;br /&gt;&lt;br /&gt;从这个例子中，我们可以看到，这个makefile中有四个需要编译的程序&amp;mdash;&amp;mdash;&amp;ldquo;prog1&amp;rdquo;， &amp;ldquo;prog2&amp;rdquo;， &amp;ldquo;prog3&amp;rdquo;和 &amp;ldquo;prog4&amp;rdquo;，我们可以使用&amp;ldquo;make all&amp;rdquo;命令来编译所有的目标（如果把all置成第一个目标，那么只需执行&amp;ldquo;make&amp;rdquo;），我们也可以使用&amp;ldquo;make prog2&amp;rdquo;来单独编译目标&amp;ldquo;prog2&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;即然make可以指定所有makefile中的目标，那么也包括&amp;ldquo;伪目标&amp;rdquo;，于是我们可以根据这种性质来让我们的makefile根据指定的不同的目标来完成不同的事。在Unix世界中，软件发布时，特别是GNU这种开源软件的发布时，其makefile都包含了编译、安装、打包等功能。我们可以参照这种规则来书写我们的makefile中的目标。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;all&amp;rdquo;&lt;br /&gt;这个伪目标是所有目标的目标，其功能一般是编译所有的目标。&lt;br /&gt;&amp;ldquo;clean&amp;rdquo;&lt;br /&gt;这个伪目标功能是删除所有被make创建的文件。&lt;br /&gt;&amp;ldquo;install&amp;rdquo;&lt;br /&gt;这个伪目标功能是安装已编译好的程序，其实就是把目标执行文件拷贝到指定的目标中去。&lt;br /&gt;&amp;ldquo;print&amp;rdquo;&lt;br /&gt;这个伪目标的功能是例出改变过的源文件。&lt;br /&gt;&amp;ldquo;tar&amp;rdquo;&lt;br /&gt;这个伪目标功能是把源程序打包备份。也就是一个tar文件。&lt;br /&gt;&amp;ldquo;dist&amp;rdquo;&lt;br /&gt;这个伪目标功能是创建一个压缩文件，一般是把tar文件压成Z文件。或是gz文件。&lt;br /&gt;&amp;ldquo;TAGS&amp;rdquo;&lt;br /&gt;这个伪目标功能是更新所有的目标，以备完整地重编译使用。&lt;br /&gt;&amp;ldquo;check&amp;rdquo;和&amp;ldquo;test&amp;rdquo;&lt;br /&gt;这两个伪目标一般用来测试makefile的流程。&lt;br /&gt;&lt;br /&gt;当然一个项目的makefile中也不一定要书写这样的目标，这些东西都是GNU的东西，但是我想，GNU搞出这些东西一定有其可取之处（等你的UNIX下的程序文件一多时你就会发现这些功能很有用了），这里只不过是说明了，如果你要书写这种功能，最好使用这种名字命名你的目标，这样规范一些，规范的好处就是&amp;mdash;&amp;mdash;不用解释，大家都明白。而且如果你的makefile中有这些功能，一是很实用，二是可以显得你的makefile很专业（不是那种初学者的作品）。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、检查规则&lt;br /&gt;&lt;br /&gt;有时候，我们不想让我们的makefile中的规则执行起来，我们只想检查一下我们的命令，或是执行的序列。于是我们可以使用make命令的下述参数：&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-n&amp;rdquo;&lt;br /&gt;&amp;ldquo;--just-print&amp;rdquo;&lt;br /&gt;&amp;ldquo;--dry-run&amp;rdquo;&lt;br /&gt;&amp;ldquo;--recon&amp;rdquo;&lt;br /&gt;不执行参数，这些参数只是打印命令，不管目标是否更新，把规则和连带规则下的命令打印出来，但不执行，这些参数对于我们调试makefile很有用处。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-t&amp;rdquo;&lt;br /&gt;&amp;ldquo;--touch&amp;rdquo;&lt;br /&gt;这个参数的意思就是把目标文件的时间更新，但不更改目标文件。也就是说，make假装编译目标，但不是真正的编译目标，只是把目标变成已编译过的状态。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-q&amp;rdquo;&lt;br /&gt;&amp;ldquo;--question&amp;rdquo;&lt;br /&gt;这个参数的行为是找目标的意思，也就是说，如果目标存在，那么其什么也不会输出，当然也不会执行编译，如果目标不存在，其会打印出一条出错信息。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-W &amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--what-if=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--assume-new=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--new-file=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;这个参数需要指定一个文件。一般是是源文件（或依赖文件），Make会根据规则推导来运行依赖于这个文件的命令，一般来说，可以和&amp;ldquo;-n&amp;rdquo;参数一同使用，来查看这个依赖文件所发生的规则命令。&lt;br /&gt;&lt;br /&gt;另外一个很有意思的用法是结合&amp;ldquo;-p&amp;rdquo;和&amp;ldquo;-v&amp;rdquo;来输出makefile被执行时的信息（这个将在后面讲述）。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、make的参数&lt;br /&gt;&lt;br /&gt;下面列举了所有GNU make 3.80版的参数定义。其它版本和产商的make大同小异，不过其它产商的make的具体参数还是请参考各自的产品文档。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-b&amp;rdquo;&lt;br /&gt;&amp;ldquo;-m&amp;rdquo;&lt;br /&gt;这两个参数的作用是忽略和其它版本make的兼容性。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-B&amp;rdquo;&lt;br /&gt;&amp;ldquo;--always-make&amp;rdquo;&lt;br /&gt;认为所有的目标都需要更新（重编译）。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-C &amp;lt;dir&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--directory=&amp;lt;dir&amp;gt;&amp;rdquo;&lt;br /&gt;指定读取makefile的目录。如果有多个&amp;ldquo;-C&amp;rdquo;参数，make的解释是后面的路径以前面的作为相对路径，并以最后的目录作为被指定目录。如：&amp;ldquo;make &amp;ndash;C ~hchen/test &amp;ndash;C prog&amp;rdquo;等价于&amp;ldquo;make &amp;ndash;C ~hchen/test/prog&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;&amp;mdash;debug[=&amp;lt;options&amp;gt;]&amp;rdquo;&lt;br /&gt;输出make的调试信息。它有几种不同的级别可供选择，如果没有参数，那就是输出最简单的调试信息。下面是&amp;lt;options&amp;gt;的取值：&lt;br /&gt;a &amp;mdash;&amp;mdash; 也就是all，输出所有的调试信息。（会非常的多）&lt;br /&gt;b &amp;mdash;&amp;mdash; 也就是basic，只输出简单的调试信息。即输出不需要重编译的目标。&lt;br /&gt;v &amp;mdash;&amp;mdash; 也就是verbose，在b选项的级别之上。输出的信息包括哪个makefile被解析，不需要被重编译的依赖文件（或是依赖目标）等。&lt;br /&gt;i &amp;mdash;&amp;mdash; 也就是implicit，输出所以的隐含规则。&lt;br /&gt;j &amp;mdash;&amp;mdash; 也就是jobs，输出执行规则中命令的详细信息，如命令的PID、返回码等。&lt;br /&gt;m &amp;mdash;&amp;mdash; 也就是makefile，输出make读取makefile，更新makefile，执行makefile的信息。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-d&amp;rdquo;&lt;br /&gt;相当于&amp;ldquo;--debug=a&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-e&amp;rdquo;&lt;br /&gt;&amp;ldquo;--environment-overrides&amp;rdquo;&lt;br /&gt;指明环境变量的值覆盖makefile中定义的变量的值。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-f=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--file=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--makefile=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;指定需要执行的makefile。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-h&amp;rdquo;&lt;br /&gt;&amp;ldquo;--help&amp;rdquo;&lt;br /&gt;显示帮助信息。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-i&amp;rdquo;&lt;br /&gt;&amp;ldquo;--ignore-errors&amp;rdquo;&lt;br /&gt;在执行时忽略所有的错误。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-I &amp;lt;dir&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--include-dir=&amp;lt;dir&amp;gt;&amp;rdquo;&lt;br /&gt;指定一个被包含makefile的搜索目标。可以使用多个&amp;ldquo;-I&amp;rdquo;参数来指定多个目录。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-j [&amp;lt;jobsnum&amp;gt;]&amp;rdquo;&lt;br /&gt;&amp;ldquo;--jobs[=&amp;lt;jobsnum&amp;gt;]&amp;rdquo;&lt;br /&gt;指同时运行命令的个数。如果没有这个参数，make运行命令时能运行多少就运行多少。如果有一个以上的&amp;ldquo;-j&amp;rdquo;参数，那么仅最后一个&amp;ldquo;-j&amp;rdquo;才是有效的。（注意这个参数在MS-DOS中是无用的）&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-k&amp;rdquo;&lt;br /&gt;&amp;ldquo;--keep-going&amp;rdquo;&lt;br /&gt;出错也不停止运行。如果生成一个目标失败了，那么依赖于其上的目标就不会被执行了。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-l &amp;lt;load&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--load-average[=&amp;lt;load]&amp;rdquo;&lt;br /&gt;&amp;ldquo;&amp;mdash;max-load[=&amp;lt;load&amp;gt;]&amp;rdquo;&lt;br /&gt;指定make运行命令的负载。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-n&amp;rdquo;&lt;br /&gt;&amp;ldquo;--just-print&amp;rdquo;&lt;br /&gt;&amp;ldquo;--dry-run&amp;rdquo;&lt;br /&gt;&amp;ldquo;--recon&amp;rdquo;&lt;br /&gt;仅输出执行过程中的命令序列，但并不执行。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-o &amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--old-file=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--assume-old=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;不重新生成的指定的&amp;lt;file&amp;gt;，即使这个目标的依赖文件新于它。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-p&amp;rdquo;&lt;br /&gt;&amp;ldquo;--print-data-base&amp;rdquo;&lt;br /&gt;输出makefile中的所有数据，包括所有的规则和变量。这个参数会让一个简单的makefile都会输出一堆信息。如果你只是想输出信息而不想执行makefile，你可以使用&amp;ldquo;make -qp&amp;rdquo;命令。如果你想查看执行makefile前的预设变量和规则，你可以使用&amp;ldquo;make &amp;ndash;p &amp;ndash;f /dev/null&amp;rdquo;。这个参数输出的信息会包含着你的makefile文件的文件名和行号，所以，用这个参数来调试你的makefile会是很有用的，特别是当你的环境变量很复杂的时候。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-q&amp;rdquo;&lt;br /&gt;&amp;ldquo;--question&amp;rdquo;&lt;br /&gt;不运行命令，也不输出。仅仅是检查所指定的目标是否需要更新。如果是0则说明要更新，如果是2则说明有错误发生。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-r&amp;rdquo;&lt;br /&gt;&amp;ldquo;--no-builtin-rules&amp;rdquo;&lt;br /&gt;禁止make使用任何隐含规则。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-R&amp;rdquo;&lt;br /&gt;&amp;ldquo;--no-builtin-variabes&amp;rdquo;&lt;br /&gt;禁止make使用任何作用于变量上的隐含规则。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-s&amp;rdquo;&lt;br /&gt;&amp;ldquo;--silent&amp;rdquo;&lt;br /&gt;&amp;ldquo;--quiet&amp;rdquo;&lt;br /&gt;在命令运行时不输出命令的输出。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-S&amp;rdquo;&lt;br /&gt;&amp;ldquo;--no-keep-going&amp;rdquo;&lt;br /&gt;&amp;ldquo;--stop&amp;rdquo;&lt;br /&gt;取消&amp;ldquo;-k&amp;rdquo;选项的作用。因为有些时候，make的选项是从环境变量&amp;ldquo;MAKEFLAGS&amp;rdquo;中继承下来的。所以你可以在命令行中使用这个参数来让环境变量中的&amp;ldquo;-k&amp;rdquo;选项失效。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-t&amp;rdquo;&lt;br /&gt;&amp;ldquo;--touch&amp;rdquo;&lt;br /&gt;相当于UNIX的touch命令，只是把目标的修改日期变成最新的，也就是阻止生成目标的命令运行。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-v&amp;rdquo;&lt;br /&gt;&amp;ldquo;--version&amp;rdquo;&lt;br /&gt;输出make程序的版本、版权等关于make的信息。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-w&amp;rdquo;&lt;br /&gt;&amp;ldquo;--print-directory&amp;rdquo;&lt;br /&gt;输出运行makefile之前和之后的信息。这个参数对于跟踪嵌套式调用make时很有用。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;--no-print-directory&amp;rdquo;&lt;br /&gt;禁止&amp;ldquo;-w&amp;rdquo;选项。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;-W &amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--what-if=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--new-file=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;&amp;ldquo;--assume-file=&amp;lt;file&amp;gt;&amp;rdquo;&lt;br /&gt;假定目标&amp;lt;file&amp;gt;需要更新，如果和&amp;ldquo;-n&amp;rdquo;选项使用，那么这个参数会输出该目标更新时的运行动作。如果没有&amp;ldquo;-n&amp;rdquo;那么就像运行UNIX的&amp;ldquo;touch&amp;rdquo;命令一样，使得&amp;lt;file&amp;gt;的修改时间为当前时间。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;--warn-undefined-variables&amp;rdquo;&lt;br /&gt;只要make发现有未定义的变量，那么就输出警告信息。&lt;/span&gt;&lt;/font&gt;&lt;/p&gt;
&lt;font face=&quot;verdana, arial, helvetica&quot; size=&quot;2&quot;&gt;&lt;span class=&quot;javascript&quot; style=&quot;FONT-SIZE: 12px&quot;&gt;
&lt;p&gt;&lt;font face=&quot;verdana, arial, helvetica&quot; size=&quot;2&quot;&gt;&lt;span class=&quot;javascript&quot; id=&quot;text1427651&quot; style=&quot;FONT-SIZE: 12px&quot;&gt;隐含规则&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;在我们使用Makefile时，有一些我们会经常使用，而且使用频率非常高的东西，比如，我们编译C/C++的源程序为中间目标文件（Unix下是[.o]文件，Windows下是[.obj]文件）。本章讲述的就是一些在Makefile中的&amp;ldquo;隐含的&amp;rdquo;，早先约定了的，不需要我们再写出来的规则。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;隐含规则&amp;rdquo;也就是一种惯例，make会按照这种&amp;ldquo;惯例&amp;rdquo;心照不喧地来运行，那怕我们的Makefile中没有书写这样的规则。例如，把[.c]文件编译成[.o]文件这一规则，你根本就不用写出来，make会自动推导出这种规则，并生成我们需要的[.o]文件。&lt;br /&gt;&lt;br /&gt;&amp;ldquo;隐含规则&amp;rdquo;会使用一些我们系统变量，我们可以改变这些系统变量的值来定制隐含规则的运行时的参数。如系统变量&amp;ldquo;CFLAGS&amp;rdquo;可以控制编译时的编译器参数。&lt;br /&gt;&lt;br /&gt;我们还可以通过&amp;ldquo;模式规则&amp;rdquo;的方式写下自己的隐含规则。用&amp;ldquo;后缀规则&amp;rdquo;来定义隐含规则会有许多的限制。使用&amp;ldquo;模式规则&amp;rdquo;会更回得智能和清楚，但&amp;ldquo;后缀规则&amp;rdquo;可以用来保证我们Makefile的兼容性。&lt;br /&gt;我们了解了&amp;ldquo;隐含规则&amp;rdquo;，可以让其为我们更好的服务，也会让我们知道一些&amp;ldquo;约定俗成&amp;rdquo;了的东西，而不至于使得我们在运行Makefile时出现一些我们觉得莫名其妙的东西。当然，任何事物都是矛盾的，水能载舟，亦可覆舟，所以，有时候&amp;ldquo;隐含规则&amp;rdquo;也会给我们造成不小的麻烦。只有了解了它，我们才能更好地使用它。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;一、使用隐含规则&lt;br /&gt;&lt;br /&gt;如果要使用隐含规则生成你需要的目标，你所需要做的就是不要写出这个目标的规则。那么，make会试图去自动推导产生这个目标的规则和命令，如果make可以自动推导生成这个目标的规则和命令，那么这个行为就是隐含规则的自动推导。当然，隐含规则是make事先约定好的一些东西。例如，我们有下面的一个Makefile：&lt;br /&gt;&lt;br /&gt;foo : foo.o bar.o&lt;br /&gt;cc &amp;ndash;o foo foo.o bar.o $(CFLAGS) $(LDFLAGS)&lt;br /&gt;&lt;br /&gt;我们可以注意到，这个Makefile中并没有写下如何生成foo.o和bar.o这两目标的规则和命令。因为make的&amp;ldquo;隐含规则&amp;rdquo;功能会自动为我们自动去推导这两个目标的依赖目标和生成命令。&lt;br /&gt;&lt;br /&gt;make会在自己的&amp;ldquo;隐含规则&amp;rdquo;库中寻找可以用的规则，如果找到，那么就会使用。如果找不到，那么就会报错。在上面的那个例子中，make调用的隐含规则是，把[.o]的目标的依赖文件置成[.c]，并使用C的编译命令&amp;ldquo;cc &amp;ndash;c $(CFLAGS) [.c]&amp;rdquo;来生成[.o]的目标。也就是说，我们完全没有必要写下下面的两条规则：&lt;br /&gt;&lt;br /&gt;foo.o : foo.c&lt;br /&gt;cc &amp;ndash;c foo.c $(CFLAGS)&lt;br /&gt;bar.o : bar.c&lt;br /&gt;cc &amp;ndash;c bar.c $(CFLAGS)&lt;br /&gt;&lt;br /&gt;因为，这已经是&amp;ldquo;约定&amp;rdquo;好了的事了，make和我们约定好了用C编译器&amp;ldquo;cc&amp;rdquo;生成[.o]文件的规则，这就是隐含规则。&lt;br /&gt;&lt;br /&gt;当然，如果我们为[.o]文件书写了自己的规则，那么make就不会自动推导并调用隐含规则，它会按照我们写好的规则忠实地执行。&lt;br /&gt;&lt;br /&gt;还有，在make的&amp;ldquo;隐含规则库&amp;rdquo;中，每一条隐含规则都在库中有其顺序，越靠前的则是越被经常使用的，所以，这会导致我们有些时候即使我们显示地指定了目标依赖，make也不会管。如下面这条规则（没有命令）：&lt;br /&gt;&lt;br /&gt;foo.o : foo.p&lt;br /&gt;&lt;br /&gt;依赖文件&amp;ldquo;foo.p&amp;rdquo;（Pascal程序的源文件）有可能变得没有意义。如果目录下存在了&amp;ldquo;foo.c&amp;rdquo;文件，那么我们的隐含规则一样会生效，并会通过&amp;ldquo;foo.c&amp;rdquo;调用C的编译器生成foo.o文件。因为，在隐含规则中，Pascal的规则出现在C的规则之后，所以，make找到可以生成foo.o的C的规则就不再寻找下一条规则了。如果你确实不希望任何隐含规则推导，那么，你就不要只写出&amp;ldquo;依赖规则&amp;rdquo;，而不写命令。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、隐含规则一览&lt;br /&gt;&lt;br /&gt;这里我们将讲述所有预先设置（也就是make内建）的隐含规则，如果我们不明确地写下规则，那么，make就会在这些规则中寻找所需要规则和命令。当然，我们也可以使用make的参数&amp;ldquo;-r&amp;rdquo;或&amp;ldquo;--no-builtin-rules&amp;rdquo;选项来取消所有的预设置的隐含规则。&lt;br /&gt;&lt;br /&gt;当然，即使是我们指定了&amp;ldquo;-r&amp;rdquo;参数，某些隐含规则还是会生效，因为有许多的隐含规则都是使用了&amp;ldquo;后缀规则&amp;rdquo;来定义的，所以，只要隐含规则中有&amp;ldquo;后缀列表&amp;rdquo;（也就一系统定义在目标.SUFFIXES的依赖目标），那么隐含规则就会生效。默认的后缀列表是：.out, .a, .ln, .o, .c, .cc, .C, .p, .f, .F, .r, .y, .l, .s, .S, .mod, .sym, .def, .h, .info, .dvi, .tex, .texinfo, .texi, .txinfo, .w, .ch .web, .sh, .elc, .el。具体的细节，我们会在后面讲述。&lt;br /&gt;&lt;br /&gt;还是先来看一看常用的隐含规则吧。&lt;br /&gt;&lt;br /&gt;1、编译C程序的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.o&amp;rdquo;的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.c&amp;rdquo;，并且其生成命令是&amp;ldquo;$(CC) &amp;ndash;c $(CPPFLAGS) $(CFLAGS)&amp;rdquo;&lt;br /&gt;&lt;br /&gt;2、编译C++程序的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.o&amp;rdquo;的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.cc&amp;rdquo;或是&amp;ldquo;&amp;lt;n&amp;gt;.C&amp;rdquo;，并且其生成命令是&amp;ldquo;$(CXX) &amp;ndash;c $(CPPFLAGS) $(CFLAGS)&amp;rdquo;。（建议使用&amp;ldquo;.cc&amp;rdquo;作为C++源文件的后缀，而不是&amp;ldquo;.C&amp;rdquo;）&lt;br /&gt;&lt;br /&gt;3、编译Pascal程序的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.o&amp;rdquo;的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.p&amp;rdquo;，并且其生成命令是&amp;ldquo;$(PC) &amp;ndash;c $(PFLAGS)&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;4、编译Fortran/Ratfor程序的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.o&amp;rdquo;的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.r&amp;rdquo;或&amp;ldquo;&amp;lt;n&amp;gt;.F&amp;rdquo;或&amp;ldquo;&amp;lt;n&amp;gt;.f&amp;rdquo;，并且其生成命令是:&lt;br /&gt;&amp;ldquo;.f&amp;rdquo; &amp;ldquo;$(FC) &amp;ndash;c $(FFLAGS)&amp;rdquo;&lt;br /&gt;&amp;ldquo;.F&amp;rdquo; &amp;ldquo;$(FC) &amp;ndash;c $(FFLAGS) $(CPPFLAGS)&amp;rdquo;&lt;br /&gt;&amp;ldquo;.f&amp;rdquo; &amp;ldquo;$(FC) &amp;ndash;c $(FFLAGS) $(RFLAGS)&amp;rdquo;&lt;br /&gt;&lt;br /&gt;5、预处理Fortran/Ratfor程序的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.f&amp;rdquo;的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.r&amp;rdquo;或&amp;ldquo;&amp;lt;n&amp;gt;.F&amp;rdquo;。这个规则只是转换Ratfor或有预处理的Fortran程序到一个标准的Fortran程序。其使用的命令是：&lt;br /&gt;&amp;ldquo;.F&amp;rdquo; &amp;ldquo;$(FC) &amp;ndash;F $(CPPFLAGS) $(FFLAGS)&amp;rdquo;&lt;br /&gt;&amp;ldquo;.r&amp;rdquo; &amp;ldquo;$(FC) &amp;ndash;F $(FFLAGS) $(RFLAGS)&amp;rdquo;&lt;br /&gt;&lt;br /&gt;6、编译Modula-2程序的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.sym&amp;rdquo;的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.def&amp;rdquo;，并且其生成命令是：&amp;ldquo;$(M2C) $(M2FLAGS) $(DEFFLAGS)&amp;rdquo;。&amp;ldquo;&amp;lt;n.o&amp;gt;&amp;rdquo; 的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.mod&amp;rdquo;，并且其生成命令是：&amp;ldquo;$(M2C) $(M2FLAGS) $(MODFLAGS)&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;7、汇编和汇编预处理的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.o&amp;rdquo; 的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.s&amp;rdquo;，默认使用编译品&amp;ldquo;as&amp;rdquo;，并且其生成命令是：&amp;ldquo;$(AS) $(ASFLAGS)&amp;rdquo;。&amp;ldquo;&amp;lt;n&amp;gt;.s&amp;rdquo; 的目标的依赖目标会自动推导为&amp;ldquo;&amp;lt;n&amp;gt;.S&amp;rdquo;，默认使用C预编译器&amp;ldquo;cpp&amp;rdquo;，并且其生成命令是：&amp;ldquo;$(AS) $(ASFLAGS)&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;8、链接Object文件的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;&amp;rdquo;目标依赖于&amp;ldquo;&amp;lt;n&amp;gt;.o&amp;rdquo;，通过运行C的编译器来运行链接程序生成（一般是&amp;ldquo;ld&amp;rdquo;），其生成命令是：&amp;ldquo;$(CC) $(LDFLAGS) &amp;lt;n&amp;gt;.o $(LOADLIBES) $(LDLIBS)&amp;rdquo;。这个规则对于只有一个源文件的工程有效，同时也对多个Object文件（由不同的源文件生成）的也有效。例如如下规则：&lt;br /&gt;&lt;br /&gt;x : y.o z.o&lt;br /&gt;&lt;br /&gt;并且&amp;ldquo;x.c&amp;rdquo;、&amp;ldquo;y.c&amp;rdquo;和&amp;ldquo;z.c&amp;rdquo;都存在时，隐含规则将执行如下命令：&lt;br /&gt;&lt;br /&gt;cc -c x.c -o x.o&lt;br /&gt;cc -c y.c -o y.o&lt;br /&gt;cc -c z.c -o z.o&lt;br /&gt;cc x.o y.o z.o -o x&lt;br /&gt;rm -f x.o&lt;br /&gt;rm -f y.o&lt;br /&gt;rm -f z.o&lt;br /&gt;&lt;br /&gt;如果没有一个源文件（如上例中的x.c）和你的目标名字（如上例中的x）相关联，那么，你最好写出自己的生成规则，不然，隐含规则会报错的。&lt;br /&gt;&lt;br /&gt;9、Yacc C程序时的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.c&amp;rdquo;的依赖文件被自动推导为&amp;ldquo;n.y&amp;rdquo;（Yacc生成的文件），其生成命令是：&amp;ldquo;$(YACC) $(YFALGS)&amp;rdquo;。（&amp;ldquo;Yacc&amp;rdquo;是一个语法分析器，关于其细节请查看相关资料）&lt;br /&gt;&lt;br /&gt;10、Lex C程序时的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.c&amp;rdquo;的依赖文件被自动推导为&amp;ldquo;n.l&amp;rdquo;（Lex生成的文件），其生成命令是：&amp;ldquo;$(LEX) $(LFALGS)&amp;rdquo;。（关于&amp;ldquo;Lex&amp;rdquo;的细节请查看相关资料）&lt;br /&gt;&lt;br /&gt;11、Lex Ratfor程序时的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.r&amp;rdquo;的依赖文件被自动推导为&amp;ldquo;n.l&amp;rdquo;（Lex生成的文件），其生成命令是：&amp;ldquo;$(LEX) $(LFALGS)&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;12、从C程序、Yacc文件或Lex文件创建Lint库的隐含规则。&lt;br /&gt;&amp;ldquo;&amp;lt;n&amp;gt;.ln&amp;rdquo; （lint生成的文件）的依赖文件被自动推导为&amp;ldquo;n.c&amp;rdquo;，其生成命令是：&amp;ldquo;$(LINT) $(LINTFALGS) $(CPPFLAGS) -i&amp;rdquo;。对于&amp;ldquo;&amp;lt;n&amp;gt;.y&amp;rdquo;和&amp;ldquo;&amp;lt;n&amp;gt;.l&amp;rdquo;也是同样的规则。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、隐含规则使用的变量&lt;br /&gt;&lt;br /&gt;在隐含规则中的命令中，基本上都是使用了一些预先设置的变量。你可以在你的makefile中改变这些变量的值，或是在make的命令行中传入这些值，或是在你的环境变量中设置这些值，无论怎么样，只要设置了这些特定的变量，那么其就会对隐含规则起作用。当然，你也可以利用make的&amp;ldquo;-R&amp;rdquo;或&amp;ldquo;--no&amp;ndash;builtin-variables&amp;rdquo;参数来取消你所定义的变量对隐含规则的作用。&lt;br /&gt;&lt;br /&gt;例如，第一条隐含规则&amp;mdash;&amp;mdash;编译C程序的隐含规则的命令是&amp;ldquo;$(CC) &amp;ndash;c $(CFLAGS) $(CPPFLAGS)&amp;rdquo;。Make默认的编译命令是&amp;ldquo;cc&amp;rdquo;，如果你把变量&amp;ldquo;$(CC)&amp;rdquo;重定义成&amp;ldquo;gcc&amp;rdquo;，把变量&amp;ldquo;$(CFLAGS)&amp;rdquo;重定义成&amp;ldquo;-g&amp;rdquo;，那么，隐含规则中的命令全部会以&amp;ldquo;gcc &amp;ndash;c -g $(CPPFLAGS)&amp;rdquo;的样子来执行了。&lt;br /&gt;&lt;br /&gt;我们可以把隐含规则中使用的变量分成两种：一种是命令相关的，如&amp;ldquo;CC&amp;rdquo;；一种是参数相的关，如&amp;ldquo;CFLAGS&amp;rdquo;。下面是所有隐含规则中会用到的变量：&lt;br /&gt;&lt;br /&gt;1、关于命令的变量。&lt;br /&gt;&lt;br /&gt;AR &lt;br /&gt;函数库打包程序。默认命令是&amp;ldquo;ar&amp;rdquo;。 &lt;br /&gt;AS &lt;br /&gt;汇编语言编译程序。默认命令是&amp;ldquo;as&amp;rdquo;。&lt;br /&gt;CC &lt;br /&gt;C语言编译程序。默认命令是&amp;ldquo;cc&amp;rdquo;。&lt;br /&gt;CXX &lt;br /&gt;C++语言编译程序。默认命令是&amp;ldquo;g++&amp;rdquo;。&lt;br /&gt;CO &lt;br /&gt;从 RCS文件中扩展文件程序。默认命令是&amp;ldquo;co&amp;rdquo;。&lt;br /&gt;CPP &lt;br /&gt;C程序的预处理器（输出是标准输出设备）。默认命令是&amp;ldquo;$(CC) &amp;ndash;E&amp;rdquo;。&lt;br /&gt;FC &lt;br /&gt;Fortran 和 Ratfor 的编译器和预处理程序。默认命令是&amp;ldquo;f77&amp;rdquo;。&lt;br /&gt;GET &lt;br /&gt;从SCCS文件中扩展文件的程序。默认命令是&amp;ldquo;get&amp;rdquo;。 &lt;br /&gt;LEX &lt;br /&gt;Lex方法分析器程序（针对于C或Ratfor）。默认命令是&amp;ldquo;lex&amp;rdquo;。&lt;br /&gt;PC &lt;br /&gt;Pascal语言编译程序。默认命令是&amp;ldquo;pc&amp;rdquo;。&lt;br /&gt;YACC &lt;br /&gt;Yacc文法分析器（针对于C程序）。默认命令是&amp;ldquo;yacc&amp;rdquo;。&lt;br /&gt;YACCR &lt;br /&gt;Yacc文法分析器（针对于Ratfor程序）。默认命令是&amp;ldquo;yacc &amp;ndash;r&amp;rdquo;。&lt;br /&gt;MAKEINFO &lt;br /&gt;转换Texinfo源文件（.texi）到Info文件程序。默认命令是&amp;ldquo;makeinfo&amp;rdquo;。&lt;br /&gt;TEX &lt;br /&gt;从TeX源文件创建TeX DVI文件的程序。默认命令是&amp;ldquo;tex&amp;rdquo;。&lt;br /&gt;TEXI2DVI &lt;br /&gt;从Texinfo源文件创建军TeX DVI 文件的程序。默认命令是&amp;ldquo;texi2dvi&amp;rdquo;。&lt;br /&gt;WEAVE &lt;br /&gt;转换Web到TeX的程序。默认命令是&amp;ldquo;weave&amp;rdquo;。&lt;br /&gt;CWEAVE &lt;br /&gt;转换C Web 到 TeX的程序。默认命令是&amp;ldquo;cweave&amp;rdquo;。&lt;br /&gt;TANGLE &lt;br /&gt;转换Web到Pascal语言的程序。默认命令是&amp;ldquo;tangle&amp;rdquo;。&lt;br /&gt;CTANGLE &lt;br /&gt;转换C Web 到 C。默认命令是&amp;ldquo;ctangle&amp;rdquo;。&lt;br /&gt;RM &lt;br /&gt;删除文件命令。默认命令是&amp;ldquo;rm &amp;ndash;f&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;2、关于命令参数的变量&lt;br /&gt;&lt;br /&gt;下面的这些变量都是相关上面的命令的参数。如果没有指明其默认值，那么其默认值都是空。&lt;br /&gt;&lt;br /&gt;ARFLAGS &lt;br /&gt;函数库打包程序AR命令的参数。默认值是&amp;ldquo;rv&amp;rdquo;。&lt;br /&gt;ASFLAGS &lt;br /&gt;汇编语言编译器参数。（当明显地调用&amp;ldquo;.s&amp;rdquo;或&amp;ldquo;.S&amp;rdquo;文件时）。 &lt;br /&gt;CFLAGS &lt;br /&gt;C语言编译器参数。&lt;br /&gt;CXXFLAGS &lt;br /&gt;C++语言编译器参数。&lt;br /&gt;COFLAGS &lt;br /&gt;RCS命令参数。 &lt;br /&gt;CPPFLAGS &lt;br /&gt;C预处理器参数。（ C 和 Fortran 编译器也会用到）。&lt;br /&gt;FFLAGS &lt;br /&gt;Fortran语言编译器参数。&lt;br /&gt;GFLAGS &lt;br /&gt;SCCS &amp;ldquo;get&amp;rdquo;程序参数。&lt;br /&gt;LDFLAGS &lt;br /&gt;链接器参数。（如：&amp;ldquo;ld&amp;rdquo;）&lt;br /&gt;LFLAGS &lt;br /&gt;Lex文法分析器参数。&lt;br /&gt;PFLAGS &lt;br /&gt;Pascal语言编译器参数。&lt;br /&gt;RFLAGS &lt;br /&gt;Ratfor 程序的Fortran 编译器参数。&lt;br /&gt;YFLAGS &lt;br /&gt;Yacc文法分析器参数。 &lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、隐含规则链&lt;br /&gt;&lt;br /&gt;有些时候，一个目标可能被一系列的隐含规则所作用。例如，一个[.o]的文件生成，可能会是先被Yacc的[.y]文件先成[.c]，然后再被C的编译器生成。我们把这一系列的隐含规则叫做&amp;ldquo;隐含规则链&amp;rdquo;。&lt;br /&gt;&lt;br /&gt;在上面的例子中，如果文件[.c]存在，那么就直接调用C的编译器的隐含规则，如果没有[.c]文件，但有一个[.y]文件，那么Yacc的隐含规则会被调用，生成[.c]文件，然后，再调用C编译的隐含规则最终由[.c]生成[.o]文件，达到目标。&lt;br /&gt;&lt;br /&gt;我们把这种[.c]的文件（或是目标），叫做中间目标。不管怎么样，make会努力自动推导生成目标的一切方法，不管中间目标有多少，其都会执着地把所有的隐含规则和你书写的规则全部合起来分析，努力达到目标，所以，有些时候，可能会让你觉得奇怪，怎么我的目标会这样生成？怎么我的makefile发疯了？&lt;br /&gt;&lt;br /&gt;在默认情况下，对于中间目标，它和一般的目标有两个地方所不同：第一个不同是除非中间的目标不存在，才会引发中间规则。第二个不同的是，只要目标成功产生，那么，产生最终目标过程中，所产生的中间目标文件会被以&amp;ldquo;rm -f&amp;rdquo;删除。&lt;br /&gt;&lt;br /&gt;通常，一个被makefile指定成目标或是依赖目标的文件不能被当作中介。然而，你可以明显地说明一个文件或是目标是中介目标，你可以使用伪目标&amp;ldquo;.INTERMEDIATE&amp;rdquo;来强制声明。（如：.INTERMEDIATE ： mid ）&lt;br /&gt;&lt;br /&gt;你也可以阻止make自动删除中间目标，要做到这一点，你可以使用伪目标&amp;ldquo;.SECONDARY&amp;rdquo;来强制声明（如：.SECONDARY : sec）。你还可以把你的目标，以模式的方式来指定（如：%.o）成伪目标&amp;ldquo;.PRECIOUS&amp;rdquo;的依赖目标，以保存被隐含规则所生成的中间文件。&lt;br /&gt;&lt;br /&gt;在&amp;ldquo;隐含规则链&amp;rdquo;中，禁止同一个目标出现两次或两次以上，这样一来，就可防止在make自动推导时出现无限递归的情况。&lt;br /&gt;&lt;br /&gt;Make会优化一些特殊的隐含规则，而不生成中间文件。如，从文件&amp;ldquo;foo.c&amp;rdquo;生成目标程序&amp;ldquo;foo&amp;rdquo;，按道理，make会编译生成中间文件&amp;ldquo;foo.o&amp;rdquo;，然后链接成&amp;ldquo;foo&amp;rdquo;，但在实际情况下，这一动作可以被一条&amp;ldquo;cc&amp;rdquo;的命令完成（cc &amp;ndash;o foo foo.c），于是优化过的规则就不会生成中间文件。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;五、定义模式规则&lt;br /&gt;&lt;br /&gt;你可以使用模式规则来定义一个隐含规则。一个模式规则就好像一个一般的规则，只是在规则中，目标的定义需要有&amp;quot;%&amp;quot;字符。&amp;quot;%&amp;quot;的意思是表示一个或多个任意字符。在依赖目标中同样可以使用&amp;quot;%&amp;quot;，只是依赖目标中的&amp;quot;%&amp;quot;的取值，取决于其目标。&lt;br /&gt;&lt;br /&gt;有一点需要注意的是，&amp;quot;%&amp;quot;的展开发生在变量和函数的展开之后，变量和函数的展开发生在make载入Makefile时，而模式规则中的&amp;quot;%&amp;quot;则发生在运行时。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;1、模式规则介绍&lt;br /&gt;&lt;br /&gt;模式规则中，至少在规则的目标定义中要包含&amp;quot;%&amp;quot;，否则，就是一般的规则。目标中的&amp;quot;%&amp;quot;定义表示对文件名的匹配，&amp;quot;%&amp;quot;表示长度任意的非空字符串。例如：&amp;quot;%.c&amp;quot;表示以&amp;quot;.c&amp;quot;结尾的文件名（文件名的长度至少为3），而&amp;quot;s.%.c&amp;quot;则表示以&amp;quot;s.&amp;quot;开头，&amp;quot;.c&amp;quot;结尾的文件名（文件名的长度至少为5）。&lt;br /&gt;&lt;br /&gt;如果&amp;quot;%&amp;quot;定义在目标中，那么，目标中的&amp;quot;%&amp;quot;的值决定了依赖目标中的&amp;quot;%&amp;quot;的值，也就是说，目标中的模式的&amp;quot;%&amp;quot;决定了依赖目标中&amp;quot;%&amp;quot;的样子。例如有一个模式规则如下：&lt;br /&gt;&lt;br /&gt;%.o : %.c ; &amp;lt;command ......&amp;gt;&lt;br /&gt;&lt;br /&gt;其含义是，指出了怎么从所有的[.c]文件生成相应的[.o]文件的规则。如果要生成的目标是&amp;quot;a.o b.o&amp;quot;，那么&amp;quot;%c&amp;quot;就是&amp;quot;a.c b.c&amp;quot;。&lt;br /&gt;&lt;br /&gt;一旦依赖目标中的&amp;quot;%&amp;quot;模式被确定，那么，make会被要求去匹配当前目录下所有的文件名，一旦找到，make就会规则下的命令，所以，在模式规则中，目标可能会是多个的，如果有模式匹配出多个目标，make就会产生所有的模式目标，此时，make关心的是依赖的文件名和生成目标的命令这两件事。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;2、模式规则示例&lt;br /&gt;&lt;br /&gt;下面这个例子表示了,把所有的[.c]文件都编译成[.o]文件.&lt;br /&gt;&lt;br /&gt;%.o : %.c&lt;br /&gt;$(CC) -c $(CFLAGS) $(CPPFLAGS) $&amp;lt; -o $@&lt;br /&gt;&lt;br /&gt;其中，&amp;quot;$@&amp;quot;表示所有的目标的挨个值，&amp;quot;$&amp;lt;&amp;quot;表示了所有依赖目标的挨个值。这些奇怪的变量我们叫&amp;quot;自动化变量&amp;quot;，后面会详细讲述。&lt;br /&gt;&lt;br /&gt;下面的这个例子中有两个目标是模式的：&lt;br /&gt;&lt;br /&gt;%.tab.c %.tab.h: %.y&lt;br /&gt;bison -d $&amp;lt;&lt;br /&gt;&lt;br /&gt;这条规则告诉make把所有的[.y]文件都以&amp;quot;bison -d &amp;lt;n&amp;gt;.y&amp;quot;执行，然后生成&amp;quot;&amp;lt;n&amp;gt;.tab.c&amp;quot;和&amp;quot;&amp;lt;n&amp;gt;.tab.h&amp;quot;文件。（其中，&amp;quot;&amp;lt;n&amp;gt;&amp;quot;表示一个任意字符串）。如果我们的执行程序&amp;quot;foo&amp;quot;依赖于文件&amp;quot;parse.tab.o&amp;quot;和&amp;quot;scan.o&amp;quot;，并且文件&amp;quot;scan.o&amp;quot;依赖于文件&amp;quot;parse.tab.h&amp;quot;，如果&amp;quot;parse.y&amp;quot;文件被更新了，那么根据上述的规则，&amp;quot;bison -d parse.y&amp;quot;就会被执行一次，于是，&amp;quot;parse.tab.o&amp;quot;和&amp;quot;scan.o&amp;quot;的依赖文件就齐了。（假设，&amp;quot;parse.tab.o&amp;quot;由&amp;quot;parse.tab.c&amp;quot;生成，和&amp;quot;scan.o&amp;quot;由&amp;quot;scan.c&amp;quot;生成，而&amp;quot;foo&amp;quot;由&amp;quot;parse.tab.o&amp;quot;和&amp;quot;scan.o&amp;quot;链接生成，而且foo和其[.o]文件的依赖关系也写好，那么，所有的目标都会得到满足）&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;3、自动化变量&lt;br /&gt;&lt;br /&gt;在上述的模式规则中，目标和依赖文件都是一系例的文件，那么我们如何书写一个命令来完成从不同的依赖文件生成相应的目标？因为在每一次的对模式规则的解析时，都会是不同的目标和依赖文件。&lt;br /&gt;&lt;br /&gt;自动化变量就是完成这个功能的。在前面，我们已经对自动化变量有所提涉，相信你看到这里已对它有一个感性认识了。所谓自动化变量，就是这种变量会把模式中所定义的一系列的文件自动地挨个取出，直至所有的符合模式的文件都取完了。这种自动化变量只应出现在规则的命令中。&lt;br /&gt;&lt;br /&gt;下面是所有的自动化变量及其说明：&lt;br /&gt;&lt;br /&gt;$@&lt;br /&gt;表示规则中的目标文件集。在模式规则中，如果有多个目标，那么，&amp;quot;$@&amp;quot;就是匹配于目标中模式定义的集合。&lt;br /&gt;&lt;br /&gt;$%&lt;br /&gt;仅当目标是函数库文件中，表示规则中的目标成员名。例如，如果一个目标是&amp;quot;foo.a(bar.o)&amp;quot;，那么，&amp;quot;$%&amp;quot;就是&amp;quot;bar.o&amp;quot;，&amp;quot;$@&amp;quot;就是&amp;quot;foo.a&amp;quot;。如果目标不是函数库文件（Unix下是[.a]，Windows下是[.lib]），那么，其值为空。&lt;br /&gt;&lt;br /&gt;$&amp;lt;&lt;br /&gt;依赖目标中的第一个目标名字。如果依赖目标是以模式（即&amp;quot;%&amp;quot;）定义的，那么&amp;quot;$&amp;lt;&amp;quot;将是符合模式的一系列的文件集。注意，其是一个一个取出来的。&lt;br /&gt;&lt;br /&gt;$?&lt;br /&gt;所有比目标新的依赖目标的集合。以空格分隔。&lt;br /&gt;&lt;br /&gt;$^&lt;br /&gt;所有的依赖目标的集合。以空格分隔。如果在依赖目标中有多个重复的，那个这个变量会去除重复的依赖目标，只保留一份。&lt;br /&gt;&lt;br /&gt;$+&lt;br /&gt;这个变量很像&amp;quot;$^&amp;quot;，也是所有依赖目标的集合。只是它不去除重复的依赖目标。&lt;br /&gt;&lt;br /&gt;$* &lt;br /&gt;这个变量表示目标模式中&amp;quot;%&amp;quot;及其之前的部分。如果目标是&amp;quot;dir/a.foo.b&amp;quot;，并且目标的模式是&amp;quot;a.%.b&amp;quot;，那么，&amp;quot;$*&amp;quot;的值就是&amp;quot;dir/a.foo&amp;quot;。这个变量对于构造有关联的文件名是比较有较。如果目标中没有模式的定义，那么&amp;quot;$*&amp;quot;也就不能被推导出，但是，如果目标文件的后缀是make所识别的，那么&amp;quot;$*&amp;quot;就是除了后缀的那一部分。例如：如果目标是&amp;quot;foo.c&amp;quot;，因为&amp;quot;.c&amp;quot;是make所能识别的后缀名，所以，&amp;quot;$*&amp;quot;的值就是&amp;quot;foo&amp;quot;。这个特性是GNU make的，很有可能不兼容于其它版本的make，所以，你应该尽量避免使用&amp;quot;$*&amp;quot;，除非是在隐含规则或是静态模式中。如果目标中的后缀是make所不能识别的，那么&amp;quot;$*&amp;quot;就是空值。&lt;br /&gt;&lt;br /&gt;当你希望只对更新过的依赖文件进行操作时，&amp;quot;$?&amp;quot;在显式规则中很有用，例如，假设有一个函数库文件叫&amp;quot;lib&amp;quot;，其由其它几个object文件更新。那么把object文件打包的比较有效率的Makefile规则是：&lt;br /&gt;&lt;br /&gt;lib : foo.o bar.o lose.o win.o&lt;br /&gt;ar r lib $?&lt;br /&gt;&lt;br /&gt;在上述所列出来的自动量变量中。四个变量（$@、$&amp;lt;、$%、$*）在扩展时只会有一个文件，而另三个的值是一个文件列表。这七个自动化变量还可以取得文件的目录名或是在当前目录下的符合模式的文件名，只需要搭配上&amp;quot;D&amp;quot;或&amp;quot;F&amp;quot;字样。这是GNU make中老版本的特性，在新版本中，我们使用函数&amp;quot;dir&amp;quot;或&amp;quot;notdir&amp;quot;就可以做到了。&amp;quot;D&amp;quot;的含义就是Directory，就是目录，&amp;quot;F&amp;quot;的含义就是File，就是文件。&lt;br /&gt;&lt;br /&gt;下面是对于上面的七个变量分别加上&amp;quot;D&amp;quot;或是&amp;quot;F&amp;quot;的含义：&lt;br /&gt;&lt;br /&gt;$(@D)&lt;br /&gt;表示&amp;quot;$@&amp;quot;的目录部分（不以斜杠作为结尾），如果&amp;quot;$@&amp;quot;值是&amp;quot;dir/foo.o&amp;quot;，那么&amp;quot;$(@D)&amp;quot;就是&amp;quot;dir&amp;quot;，而如果&amp;quot;$@&amp;quot;中没有包含斜杠的话，其值就是&amp;quot;.&amp;quot;（当前目录）。&lt;br /&gt;&lt;br /&gt;$(@F)&lt;br /&gt;表示&amp;quot;$@&amp;quot;的文件部分，如果&amp;quot;$@&amp;quot;值是&amp;quot;dir/foo.o&amp;quot;，那么&amp;quot;$(@F)&amp;quot;就是&amp;quot;foo.o&amp;quot;，&amp;quot;$(@F)&amp;quot;相当于函数&amp;quot;$(notdir $@)&amp;quot;。&lt;br /&gt;&lt;br /&gt;&amp;quot;$(*D)&amp;quot;&lt;br /&gt;&amp;quot;$(*F)&amp;quot;&lt;br /&gt;和上面所述的同理，也是取文件的目录部分和文件部分。对于上面的那个例子，&amp;quot;$(*D)&amp;quot;返回&amp;quot;dir&amp;quot;，而&amp;quot;$(*F)&amp;quot;返回&amp;quot;foo&amp;quot;&lt;br /&gt;&lt;br /&gt;&amp;quot;$(%D)&amp;quot;&lt;br /&gt;&amp;quot;$(%F)&amp;quot;&lt;br /&gt;分别表示了函数包文件成员的目录部分和文件部分。这对于形同&amp;quot;archive(member)&amp;quot;形式的目标中的&amp;quot;member&amp;quot;中包含了不同的目录很有用。&lt;br /&gt;&lt;br /&gt;&amp;quot;$(&amp;lt;D)&amp;quot;&lt;br /&gt;&amp;quot;$(&amp;lt;F)&amp;quot;&lt;br /&gt;分别表示依赖文件的目录部分和文件部分。&lt;br /&gt;&lt;br /&gt;&amp;quot;$(^D)&amp;quot;&lt;br /&gt;&amp;quot;$(^F)&amp;quot;&lt;br /&gt;分别表示所有依赖文件的目录部分和文件部分。（无相同的）&lt;br /&gt;&lt;br /&gt;&amp;quot;$(+D)&amp;quot;&lt;br /&gt;&amp;quot;$(+F)&amp;quot;&lt;br /&gt;分别表示所有依赖文件的目录部分和文件部分。（可以有相同的）&lt;br /&gt;&lt;br /&gt;&amp;quot;$(?D)&amp;quot;&lt;br /&gt;&amp;quot;$(?F)&amp;quot;&lt;br /&gt;分别表示被更新的依赖文件的目录部分和文件部分。&lt;br /&gt;&lt;br /&gt;最后想提醒一下的是，对于&amp;quot;$&amp;lt;&amp;quot;，为了避免产生不必要的麻烦，我们最好给$后面的那个特定字符都加上圆括号，比如，&amp;quot;$(&amp;lt; )&amp;quot;就要比&amp;quot;$&amp;lt;&amp;quot;要好一些。&lt;br /&gt;&lt;br /&gt;还得要注意的是，这些变量只使用在规则的命令中，而且一般都是&amp;quot;显式规则&amp;quot;和&amp;quot;静态模式规则&amp;quot;（参见前面&amp;quot;书写规则&amp;quot;一章）。其在隐含规则中并没有意义。&lt;br /&gt;&lt;br /&gt;4、模式的匹配&lt;br /&gt;&lt;br /&gt;一般来说，一个目标的模式有一个有前缀或是后缀的&amp;quot;%&amp;quot;，或是没有前后缀，直接就是一个&amp;quot;%&amp;quot;。因为&amp;quot;%&amp;quot;代表一个或多个字符，所以在定义好了的模式中，我们把&amp;quot;%&amp;quot;所匹配的内容叫做&amp;quot;茎&amp;quot;，例如&amp;quot;%.c&amp;quot;所匹配的文件&amp;quot;test.c&amp;quot;中&amp;quot;test&amp;quot;就是&amp;quot;茎&amp;quot;。因为在目标和依赖目标中同时有&amp;quot;%&amp;quot;时，依赖目标的&amp;quot;茎&amp;quot;会传给目标，当做目标中的&amp;quot;茎&amp;quot;。&lt;br /&gt;&lt;br /&gt;当一个模式匹配包含有斜杠（实际也不经常包含）的文件时，那么在进行模式匹配时，目录部分会首先被移开，然后进行匹配，成功后，再把目录加回去。在进行&amp;quot;茎&amp;quot;的传递时，我们需要知道这个步骤。例如有一个模式&amp;quot;e%t&amp;quot;，文件&amp;quot;src/eat&amp;quot;匹配于该模式，于是&amp;quot;src/a&amp;quot;就是其&amp;quot;茎&amp;quot;，如果这个模式定义在依赖目标中，而被依赖于这个模式的目标中又有个模式&amp;quot;c%r&amp;quot;，那么，目标就是&amp;quot;src/car&amp;quot;。（&amp;quot;茎&amp;quot;被传递）&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;5、重载内建隐含规则&lt;br /&gt;&lt;br /&gt;你可以重载内建的隐含规则（或是定义一个全新的），例如你可以重新构造和内建隐含规则不同的命令，如：&lt;br /&gt;&lt;br /&gt;%.o : %.c&lt;br /&gt;$(CC) -c $(CPPFLAGS) $(CFLAGS) -D$(date)&lt;br /&gt;&lt;br /&gt;你可以取消内建的隐含规则，只要不在后面写命令就行。如：&lt;br /&gt;&lt;br /&gt;%.o : %.s&lt;br /&gt;&lt;br /&gt;同样，你也可以重新定义一个全新的隐含规则，其在隐含规则中的位置取决于你在哪里写下这个规则。朝前的位置就靠前。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;六、老式风格的&amp;quot;后缀规则&amp;quot;&lt;br /&gt;&lt;br /&gt;后缀规则是一个比较老式的定义隐含规则的方法。后缀规则会被模式规则逐步地取代。因为模式规则更强更清晰。为了和老版本的Makefile兼容，GNU make同样兼容于这些东西。后缀规则有两种方式：&amp;quot;双后缀&amp;quot;和&amp;quot;单后缀&amp;quot;。&lt;br /&gt;&lt;br /&gt;双后缀规则定义了一对后缀：目标文件的后缀和依赖目标（源文件）的后缀。如&amp;quot;.c.o&amp;quot;相当于&amp;quot;%o : %c&amp;quot;。单后缀规则只定义一个后缀，也就是源文件的后缀。如&amp;quot;.c&amp;quot;相当于&amp;quot;% : %.c&amp;quot;。&lt;br /&gt;&lt;br /&gt;后缀规则中所定义的后缀应该是make所认识的，如果一个后缀是make所认识的，那么这个规则就是单后缀规则，而如果两个连在一起的后缀都被make所认识，那就是双后缀规则。例如：&amp;quot;.c&amp;quot;和&amp;quot;.o&amp;quot;都是make所知道。因而，如果你定义了一个规则是&amp;quot;.c.o&amp;quot;那么其就是双后缀规则，意义就是&amp;quot;.c&amp;quot;是源文件的后缀，&amp;quot;.o&amp;quot;是目标文件的后缀。如下示例：&lt;br /&gt;&lt;br /&gt;.c.o:&lt;br /&gt;$(CC) -c $(CFLAGS) $(CPPFLAGS) -o $@ $&amp;lt;&lt;br /&gt;&lt;br /&gt;后缀规则不允许任何的依赖文件，如果有依赖文件的话，那就不是后缀规则，那些后缀统统被认为是文件名，如：&lt;br /&gt;&lt;br /&gt;.c.o: foo.h&lt;br /&gt;$(CC) -c $(CFLAGS) $(CPPFLAGS) -o $@ $&amp;lt;&lt;br /&gt;&lt;br /&gt;这个例子，就是说，文件&amp;quot;.c.o&amp;quot;依赖于文件&amp;quot;foo.h&amp;quot;，而不是我们想要的这样：&lt;br /&gt;&lt;br /&gt;%.o: %.c foo.h&lt;br /&gt;$(CC) -c $(CFLAGS) $(CPPFLAGS) -o $@ $&amp;lt;&lt;br /&gt;&lt;br /&gt;后缀规则中，如果没有命令，那是毫无意义的。因为他也不会移去内建的隐含规则。&lt;br /&gt;&lt;br /&gt;而要让make知道一些特定的后缀，我们可以使用伪目标&amp;quot;.SUFFIXES&amp;quot;来定义或是删除，如：&lt;br /&gt;&lt;br /&gt;.SUFFIXES: .hack .win&lt;br /&gt;&lt;br /&gt;把后缀.hack和.win加入后缀列表中的末尾。&lt;br /&gt;&lt;br /&gt;.SUFFIXES: # 删除默认的后缀&lt;br /&gt;.SUFFIXES: .c .o .h # 定义自己的后缀&lt;br /&gt;&lt;br /&gt;先清楚默认后缀，后定义自己的后缀列表。&lt;br /&gt;&lt;br /&gt;make的参数&amp;quot;-r&amp;quot;或&amp;quot;-no-builtin-rules&amp;quot;也会使用得默认的后缀列表为空。而变量&amp;quot;SUFFIXE&amp;quot;被用来定义默认的后缀列表，你可以用&amp;quot;.SUFFIXES&amp;quot;来改变后缀列表，但请不要改变变量&amp;quot;SUFFIXE&amp;quot;的值。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;七、隐含规则搜索算法&lt;br /&gt;&lt;br /&gt;比如我们有一个目标叫 T。下面是搜索目标T的规则的算法。请注意，在下面，我们没有提到后缀规则，原因是，所有的后缀规则在Makefile被载入内存时，会被转换成模式规则。如果目标是&amp;quot;archive(member)&amp;quot;的函数库文件模式，那么这个算法会被运行两次，第一次是找目标T，如果没有找到的话，那么进入第二次，第二次会把&amp;quot;member&amp;quot;当作T来搜索。&lt;br /&gt;&lt;br /&gt;1、把T的目录部分分离出来。叫D，而剩余部分叫N。（如：如果T是&amp;quot;src/foo.o&amp;quot;，那么，D就是&amp;quot;src/&amp;quot;，N就是&amp;quot;foo.o&amp;quot;）&lt;br /&gt;&lt;br /&gt;2、创建所有匹配于T或是N的模式规则列表。&lt;br /&gt;&lt;br /&gt;3、如果在模式规则列表中有匹配所有文件的模式，如&amp;quot;%&amp;quot;，那么从列表中移除其它的模式。&lt;br /&gt;&lt;br /&gt;4、移除列表中没有命令的规则。&lt;br /&gt;&lt;br /&gt;5、对于第一个在列表中的模式规则：&lt;br /&gt;1）推导其&amp;quot;茎&amp;quot;S，S应该是T或是N匹配于模式中&amp;quot;%&amp;quot;非空的部分。&lt;br /&gt;2）计算依赖文件。把依赖文件中的&amp;quot;%&amp;quot;都替换成&amp;quot;茎&amp;quot;S。如果目标模式中没有包含斜框字符，而把D加在第一个依赖文件的开头。&lt;br /&gt;3）测试是否所有的依赖文件都存在或是理当存在。（如果有一个文件被定义成另外一个规则的目标文件，或者是一个显式规则的依赖文件，那么这个文件就叫&amp;quot;理当存在&amp;quot;）&lt;br /&gt;4）如果所有的依赖文件存在或是理当存在，或是就没有依赖文件。那么这条规则将被采用，退出该算法。&lt;br /&gt;&lt;br /&gt;6、如果经过第5步，没有模式规则被找到，那么就做更进一步的搜索。对于存在于列表中的第一个模式规则：&lt;br /&gt;1）如果规则是终止规则，那就忽略它，继续下一条模式规则。&lt;br /&gt;2）计算依赖文件。（同第5步）&lt;br /&gt;3）测试所有的依赖文件是否存在或是理当存在。&lt;br /&gt;4）对于不存在的依赖文件，递归调用这个算法查找他是否可以被隐含规则找到。&lt;br /&gt;5）如果所有的依赖文件存在或是理当存在，或是就根本没有依赖文件。那么这条规则被采用，退出该算法。&lt;br /&gt;&lt;br /&gt;7、如果没有隐含规则可以使用，查看&amp;quot;.DEFAULT&amp;quot;规则，如果有，采用，把&amp;quot;.DEFAULT&amp;quot;的命令给T使用。&lt;br /&gt;&lt;br /&gt;一旦规则被找到，就会执行其相当的命令，而此时，我们的自动化变量的值才会生成。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;使用make更新函数库文件&lt;br /&gt;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;函数库文件也就是对Object文件（程序编译的中间文件）的打包文件。在Unix下，一般是由命令&amp;quot;ar&amp;quot;来完成打包工作。&lt;br /&gt;&lt;br /&gt;一、函数库文件的成员&lt;br /&gt;&lt;br /&gt;一个函数库文件由多个文件组成。你可以以如下格式指定函数库文件及其组成：&lt;br /&gt;&lt;br /&gt;archive(member)&lt;br /&gt;&lt;br /&gt;这个不是一个命令，而一个目标和依赖的定义。一般来说，这种用法基本上就是为了&amp;quot;ar&amp;quot;命令来服务的。如：&lt;br /&gt;&lt;br /&gt;foolib(hack.o) : hack.o&lt;br /&gt;ar cr foolib hack.o&lt;br /&gt;&lt;br /&gt;如果要指定多个member，那就以空格分开，如：&lt;br /&gt;&lt;br /&gt;foolib(hack.o kludge.o)&lt;br /&gt;&lt;br /&gt;其等价于：&lt;br /&gt;&lt;br /&gt;foolib(hack.o) foolib(kludge.o)&lt;br /&gt;&lt;br /&gt;你还可以使用Shell的文件通配符来定义，如：&lt;br /&gt;&lt;br /&gt;foolib(*.o)&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;二、函数库成员的隐含规则&lt;br /&gt;&lt;br /&gt;当make搜索一个目标的隐含规则时，一个特殊的特性是，如果这个目标是&amp;quot;a(m)&amp;quot;形式的，其会把目标变成&amp;quot;(m)&amp;quot;。于是，如果我们的成员是&amp;quot;%.o&amp;quot;的模式定义，并且如果我们使用&amp;quot;make foo.a(bar.o)&amp;quot;的形式调用Makefile时，隐含规则会去找&amp;quot;bar.o&amp;quot;的规则，如果没有定义bar.o的规则，那么内建隐含规则生效，make会去找bar.c文件来生成bar.o，如果找得到的话，make执行的命令大致如下：&lt;br /&gt;&lt;br /&gt;cc -c bar.c -o bar.o&lt;br /&gt;ar r foo.a bar.o&lt;br /&gt;rm -f bar.o&lt;br /&gt;&lt;br /&gt;还有一个变量要注意的是&amp;quot;$%&amp;quot;，这是专属函数库文件的自动化变量，有关其说明请参见&amp;quot;自动化变量&amp;quot;一节。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;三、函数库文件的后缀规则&lt;br /&gt;&lt;br /&gt;你可以使用&amp;quot;后缀规则&amp;quot;和&amp;quot;隐含规则&amp;quot;来生成函数库打包文件，如：&lt;br /&gt;&lt;br /&gt;.c.a:&lt;br /&gt;$(CC) $(CFLAGS) $(CPPFLAGS) -c $&amp;lt; -o $*.o&lt;br /&gt;$(AR) r $@ $*.o&lt;br /&gt;$(RM) $*.o&lt;br /&gt;&lt;br /&gt;其等效于：&lt;br /&gt;&lt;br /&gt;(%.o) : %.c&lt;br /&gt;$(CC) $(CFLAGS) $(CPPFLAGS) -c $&amp;lt; -o $*.o&lt;br /&gt;$(AR) r $@ $*.o&lt;br /&gt;$(RM) $*.o&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;四、注意事项&lt;br /&gt;&lt;br /&gt;在进行函数库打包文件生成时，请小心使用make的并行机制（&amp;quot;-j&amp;quot;参数）。如果多个ar命令在同一时间运行在同一个函数库打包文件上，就很有可以损坏这个函数库文件。所以，在make未来的版本中，应该提供一种机制来避免并行操作发生在函数打包文件上。&lt;br /&gt;&lt;br /&gt;但就目前而言，你还是应该不要尽量不要使用&amp;quot;-j&amp;quot;参数。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;后序&lt;br /&gt;&amp;mdash;&amp;mdash;&lt;br /&gt;&lt;br /&gt;终于到写结束语的时候了，以上基本上就是GNU make的Makefile的所有细节了。其它的产商的make基本上也就是这样的，无论什么样的make，都是以文件的依赖性为基础的，其基本是都是遵循一个标准的。这篇文档中80%的技术细节都适用于任何的make，我猜测&amp;quot;函数&amp;quot;那一章的内容可能不是其它make所支持的，而隐含规则方面，我想不同的make会有不同的实现，我没有精力来查看GNU的make和VC的nmake、BCB的make，或是别的UNIX下的make有些什么样的差别，一是时间精力不够，二是因为我基本上都是在Unix下使用make，以前在SCO Unix和IBM的AIX，现在在Linux、Solaris、HP-UX、AIX和Alpha下使用，Linux和Solaris下更多一点。不过，我可以肯定的是，在Unix下的make，无论是哪种平台，几乎都使用了Richard Stallman开发的make和cc/gcc的编译器，而且，基本上都是GNU的make（公司里所有的UNIX机器上都被装上了GNU的东西，所以，使用GNU的程序也就多了一些）。GNU的东西还是很不错的，特别是使用得深了以后，越来越觉得GNU的软件的强大，也越来越觉得GNU的在操作系统中（主要是Unix，甚至Windows）&amp;quot;杀伤力&amp;quot;。&lt;br /&gt;&lt;br /&gt;对于上述所有的make的细节，我们不但可以利用make这个工具来编译我们的程序，还可以利用make来完成其它的工作，因为规则中的命令可以是任何Shell之下的命令，所以，在Unix下，你不一定只是使用程序语言的编译器，你还可以在Makefile中书写其它的命令，如：tar、awk、mail、sed、cvs、compress、ls、rm、yacc、rpm、ftp&amp;hellip;&amp;hellip;等等，等等，来完成诸如&amp;quot;程序打包&amp;quot;、&amp;quot;程序备份&amp;quot;、&amp;quot;制作程序安装包&amp;quot;、&amp;quot;提交代码&amp;quot;、&amp;quot;使用程序模板&amp;quot;、&amp;quot;合并文件&amp;quot;等等五花八门的功能，文件操作，文件管理，编程开发设计，或是其它一些异想天开的东西。比如，以前在书写银行交易程序时，由于银行的交易程序基本一样，就见到有人书写了一些交易的通用程序模板，在该模板中把一些网络通讯、数据库操作的、业务操作共性的东西写在一个文件中，在这些文件中用些诸如&amp;quot;@@@N、###N&amp;quot;奇怪字串标注一些位置，然后书写交易时，只需按照一种特定的规则书写特定的处理，最后在make时，使用awk和sed，把模板中的&amp;quot;@@@N、###N&amp;quot;等字串替代成特定的程序，形成C文件，然后再编译。这个动作很像数据库的&amp;quot;扩展C&amp;quot;语言（即在C语言中用&amp;quot;EXEC　SQL&amp;quot;的样子执行SQL语句，在用cc/gcc编译之前，需要使用&amp;quot;扩展C&amp;quot;的翻译程序，如cpre，把其翻译成标准C）。如果你在使用make时有一些更为绝妙的方法，请记得告诉我啊。&lt;br /&gt;&lt;br /&gt;回头看看整篇文档，不觉记起几年前刚刚开始在Unix下做开发的时候，有人问我会不会写Makefile时，我两眼发直，根本不知道在说什么。一开始看到别人在vi中写完程序后输入&amp;quot;!make&amp;quot;时，还以为是vi的功能，后来才知道有一个Makefile在作怪，于是上网查啊查，那时又不愿意看英文，发现就根本没有中文的文档介绍Makefile，只得看别人写的Makefile，自己瞎碰瞎搞才积累了一点知识，但在很多地方完全是知其然不知所以然。后来开始从事UNIX下产品软件的开发，看到一个400人年，近200万行代码的大工程，发现要编译这样一个庞然大物，如果没有Makefile，那会是多么恐怖的一样事啊。于是横下心来，狠命地读了一堆英文文档，才觉得对其掌握了。但发现目前网上对Makefile介绍的文章还是少得那么的可怜，所以想写这样一篇文章，共享给大家，希望能对各位有所帮助。&lt;br /&gt;&lt;br /&gt;现在我终于写完了，看了看文件的创建时间，这篇技术文档也写了两个多月了。发现，自己知道是一回事，要写下来，跟别人讲述又是另外一回事，而且，现在越来越没有时间专研技术细节，所以在写作时，发现在阐述一些细节问题时很难做到严谨和精练，而且对先讲什么后讲什么不是很清楚，所以，还是参考了一些国外站点上的资料和题纲，以及一些技术书籍的语言风格，才得以完成。整篇文档的提纲是基于GNU的Makefile技术手册的提纲来书写的，并结合了自己的工作经验，以及自己的学习历程。因为从来没有写过这么长，这么细的文档，所以一定会有很多地方存在表达问题，语言歧义或是错误。因些，我迫切地得等待各位给我指证和建议，以及任何的反馈。&lt;br /&gt;&lt;br /&gt;最后，还是利用这个后序，介绍一下自己。我目前从事于所有Unix平台下的软件研发，主要是做分布式计算/网格计算方面的系统产品软件，并且我对于下一代的计算机革命&amp;mdash;&amp;mdash;网格计算非常地感兴趣，对于分布式计算、P2P、Web Service、J2EE技术方向也很感兴趣，同时，对于项目实施、团队管理、项目管理也小有心得，希望同样和我战斗在&amp;ldquo;技术和管理并重&amp;rdquo;的阵线上的年轻一代，能够和我多多地交流。我的MSN是：haoel@hotmail.com（常用），QQ是：753640（不常用）。（注：请勿给我MSN的邮箱发信，由于hotmail的垃圾邮件导致我拒收这个邮箱的所有来信）&lt;br /&gt;&lt;br /&gt;我欢迎任何形式的交流，无论是讨论技术还是管理，或是其它海阔天空的东西。除了政治和娱乐新闻我不关心，其它只要积极向上的东西我都欢迎！&lt;br /&gt;&lt;br /&gt;最最后，我还想介绍一下make程序的设计开发者。&lt;br /&gt;&lt;br /&gt;首当其冲的是： Richard Stallman &lt;br /&gt;&lt;br /&gt;开源软件的领袖和先驱，从来没有领过一天工资，从来没有使用过Windows操作系统。对于他的事迹和他的软件以及他的思想，我无需说过多的话，相信大家对这个人并不比我陌生，这是他的主页：&lt;a target=&quot;_blank&quot; href=&quot;http://www.stallman.org/&quot;&gt;http://www.stallman.org/&lt;/a&gt; 。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;第二位是：Roland McGrath &lt;br /&gt;&lt;br /&gt;个人主页是：&lt;a target=&quot;_blank&quot; href=&quot;http://www.frob.com/~roland/&quot;&gt;http://www.frob.com/~roland/&lt;/a&gt; ，下面是他的一些事迹：&lt;br /&gt;&lt;br /&gt;1） 合作编写了并维护GNU make。&lt;br /&gt;&lt;br /&gt;2） 和Thomas Bushnell一同编写了GNU Hurd。&lt;br /&gt;&lt;br /&gt;3） 编写并维护着GNU C library。 &lt;br /&gt;&lt;br /&gt;4） 合作编写并维护着部分的GNU Emacs。 &lt;br /&gt;&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;在此，向这两位开源项目的斗士致以最真切的敬意。&lt;br /&gt;&lt;br /&gt;&lt;br /&gt;（全文完）&lt;/span&gt;&lt;/font&gt;&lt;/p&gt;
&lt;/span&gt;&lt;/font&gt;&lt;/span&gt;&lt;/font&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-9-4 9:36:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1771246&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：47415 评论：35 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1771246#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Tue, 04 Sep 2007 09:36:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/1771246</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/1771246</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057076/1064678</fs:itemid></item><item><title>[原]大四了</title><link>http://blog.csdn.net/liang13664759/article/details/1769941</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;br /&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp; 我还没有回家，别人都来上课了。明天就要上课，感觉我好像还没有调整好，因为感觉有点想jia。明天真的是要拿着书去教室上课嘛？？？？？？？看心情吧！大一军训的情景还历历在目，转眼间大四了，没多久就要离开学了。刚来的时候学校还像个工地（南昌航空工业学院前湖校区2004）。刚进学校，一看还不错一条笔直的路，而且绿化也不错。再往里走就是一个大坑，一个非常大的坑估计比两个足球场小不了多少。我妈说在修个足球场，你还&lt;br /&gt;能踢球。后来在这个大坑上建了一个很大很大的图书馆，听说好像是27层，具体有没有这么多或者比这个好要多，那我就不知道的，因为我没有去数过。还有这个图书馆是我看着建起来的，感觉有点自豪，去年暑假，它竣工的时候我们坐在办公室编程，大白天还礼花庆祝这个然我觉得有些诧异。三年了，我在这里待了三年。三年间我把我的大多数时间都给了计算机。在我的记忆中有那么一段时间编程几乎成了我生活的全部，每天除了吃饭睡觉，上课，坐在电脑前就编程，不打游戏，不上网（是不上网娱乐，google,baidu还是每天必不可少的）。我不知那段时间我是在借它在逃避什么，还是迷上了。现在想想应该是两者皆有吧。对计算机有了深入的了解，有了自己的思想。我的思考都是用它的模式，这样不好，会丢点一些更重要的东西，比如创造力。所以现在我也刻意让自己不要这样去做，从多角度去思考，不要总是用计算机的思想去思考。我觉得我做得还不错。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;br /&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 三年间，我会学了很多东西，不仅是计算机。这里要再一次的提到思想这个词。这个词很普通，很平常。很多人也把它挂在嘴上，但是真正深刻的领悟的可能不多。一个人的思想是一个人财富。很多很多人没有思想。一旦有了思想你就进了一大步，上了一个台阶，看问题的高度，角度都不一样了。但是思想到底是什么东西？你的阅历，价值观，人生观，你的胸襟，大概就是这些东西吧！这是一个比较抽象的东西，需要去体会，开窍了就好，不然很难理解。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;br /&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 三年前高考完了以后我很失望，完全不是我的水准，我不是在这里为自己辩解什么，我想说的是现在的我没有后悔来南昌，来到南昌航空大学。感谢我的家人都不同意我复读。如果考到了更好的学校，如果我的生活是一帆风顺，现在的我是什么样，那就不好说了。也许更好，也许只是一个中庸的人。对于现在的我，我的状态，我还是满意的。但是不会就这样停下来，继续一步一个脚印的走，一个台阶一个台阶的上。每当又上一个台阶，再看世界都&lt;br /&gt;不一样。更多色彩，更精彩，同样也更黑暗。让自己保持积极向上的激情，保持清醒，让自己活得健康一些。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;br /&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 三年来也建立了友谊，郑鑫一个非常扯蛋家伙，思想很活跃。付俊伟原来是一个很老实憨厚的家伙，现在我不这么认为了，好像现在比我还扯，潜力无限啊。表哥嘛还那样，有些放不开，好像还没有找到自信。徐黎一个很善良的家伙。周元晨一个不错的北方大个。还有花哥，在他那学到很多东西，人不错。徐昭鹤海军，一个不错的家伙，有事找他帮忙一般没有问题，靠的主（完达山，伊立酸酸乳，蒙牛）。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &lt;br /&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 关于爱情，本以在大学是不会来了。结果有些事情就是这样，很难预料。在隔壁学校（南昌大学），有两条正气龙在湖里，湖中间有一条小路，小路在正气龙的后面，正气龙的前面有一个圆广场，广场周围有很多石梯是个看台。傍晚，灯光照着正气龙，广场中间有烟花，我和一个女生坐在看台上，我拉着她的手，我们在看手相（我故意的），然后她就成了我的女朋友，我们就这样恋爱了。哦，对了那个烟花不是我安排的，好像是有人过生日。正确的时间，正确的地点，正确的人。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 大四了，我和我周围的人奋斗中！！！&lt;br /&gt;&amp;nbsp;&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-9-3 11:04:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1769941&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：1072 评论：0 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1769941#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Mon, 03 Sep 2007 11:04:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/1769941</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/1769941</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057075/1064678</fs:itemid></item><item><title>[原]大三总结</title><link>http://blog.csdn.net/liang13664759/article/details/1765279</link><description>&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 已经不止一次的感叹时光飞逝！又是一个夏秋冬春。去年的暑假我们在学校敲击了两个月的键盘，还记得刚开始写MVC程序的时候，所用的东西都是懵懵懂懂的。不懂得JSP，SERVLET的工作原理，只知道找着框架写代码。写了一个两层的网上报名，还僻陋百出。一年的时间里成长了不少，也清晰的看到了进步。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 这一年的时间里了做了很多东西，看了很多的书。研究过windows下的NDIS中间成数据包拦截，这个是我比较感兴趣的一个东西，可是由于时间上的关系，又被叫去做一些其它的东西，所以这一块只是学习了理论，没有做出实际的东西，不过也还算长了很多知识。有时间我会去写一些东西，希望能写一个防火墙。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; &amp;nbsp;然后就是J2EE方面的东西做了很多，计算机学院网站，计算机教务系统，江西广电，江西省教育技术委员会网站，当然都只是其中的部分模块。现在对JSP、SERVLET的原理比较熟悉，也看过一些模式，当然也自己写过框架。有关J2EE的东西，反正就是HTML，JAVASCRIPT，JSTL（我比较喜欢用这套标签，当然也可以用JSP）,SERVLET,还有就是持久层。也用过STRUTS，其实这些东西也就是用SERVLET，BEEN，JSP，做更高一层的封装。我自己用接口写过可以用一个SERVLET实现一个应用的架子。我觉得这样代码更清晰，特别在维护的时候表方便，因为这一年的时间，有一半上的时间用在了维护别人写代码上，也用不同的架子写过实际的项目，所以我知道维护是一件比较痛苦的事情，特别遇到编码风格比较差的时候真的让人觉得想哭，我自己写的这套架子现在用于江西广电客户服务系统上，也算是比较骄傲的事情。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 接下来这个是我最喜欢的一个，我们团队用J2ME在支持JSR-82的蓝牙手机上实现Ad Hoc网络的搭建和拓扑。我们自己制定了路由协议，设计了整个程序。从构思到实现全部由我们自己搞定。我对这个项目感觉到很满意。但是遗憾的是我们只有一台满足要求的试验设备，杨老师的那部手机。学校给我们的经费迟迟没有下来。所以我们只在模拟器上测试，由于经费还没有下来，我们决定用USB接口的蓝牙设备在PC上模拟手机，可是又遇到了一些问题。当这些问题似乎都要被解决的时候，暑假开始了，一个暑假我都在改江西广电客户服务系统至今。希望学校的经费能快点下来，不然我都要出去找工作了。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 还有更重要的是找到了一个叫圆圆女人。&lt;/p&gt;
&lt;p&gt;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp;&amp;nbsp; 还有一个就是比较失败的FREERADIUS，因为广电要AAA但是在windows2003下的那个服务器抗不住压力，所以想要换freeradius。可是在freeradius在自己扩展它的功能的时候，自己添加的进程和freeradius的进程的执行顺序是不确定的。所以这个东西算是失败，但是了解了RADIUS，在linux下也搭上了服务器，freeradius也跑起来了。但是没有用上。&lt;/p&gt;
            &lt;div&gt;
                作者：liang13664759 发表于2007-8-30 13:28:00 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1765279&quot;&gt;原文链接&lt;/a&gt;
            &lt;/div&gt;
            &lt;div&gt;
            阅读：2976 评论：0 &lt;a href=&quot;http://blog.csdn.net/liang13664759/article/details/1765279#comments&quot; target=&quot;_blank&quot;&gt;查看评论&lt;/a&gt;
            &lt;/div&gt;</description><pubDate>Thu, 30 Aug 2007 13:28:00 +0800</pubDate><author>liang13664759</author><guid isPermaLink="false">http://blog.csdn.net/liang13664759/article/details/1765279</guid><dc:creator>liang13664759</dc:creator><fs:srclink>http://blog.csdn.net/liang13664759/article/details/1765279</fs:srclink><fs:srcfeed>http://blog.csdn.net/liang13664759/feed.aspx</fs:srcfeed><fs:itemid>csdn.net/liang13664759/~1064678/638057074/1064678</fs:itemid></item></channel></rss>
