﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>C++博客-zhulf753</title><link>http://www.cppblog.com/zhulf753/</link><description /><language>zh-cn</language><lastBuildDate>Tue, 14 Apr 2026 23:08:09 GMT</lastBuildDate><pubDate>Tue, 14 Apr 2026 23:08:09 GMT</pubDate><ttl>60</ttl><item><title>10年初搭建的原始的AOSP开发环境</title><link>http://www.cppblog.com/zhulf753/archive/2020/12/24/217547.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Thu, 24 Dec 2020 04:34:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2020/12/24/217547.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/217547.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2020/12/24/217547.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/217547.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/217547.html</trackback:ping><description><![CDATA[<div header"="" style="box-sizing: inherit; border: none; margin: 0px; padding: 0em; line-height: 1.33em;"><div style="box-sizing: inherit; margin-bottom: 10px;"><span style="font-size: 14.135px;"><strong>10年初搭建的原始的AOSP工程。不知道官网现在还能不能下载到。从内核到开发调试工具的所有东西都可以编译出来，包括：内核、虚拟机、Framework、Java应用、手机模拟器、log工具、调试工具、开发用的Eclipse插件、SDK、、NDK等。<br />链接：</strong></span>https://gitee.com/zhulf753/repo/tree/master/Android/Ubuntu10.04</div></div><div id="weava-ui-wrapper" style="font-family: Tahoma; font-size: 11px;"><div><div></div> <div>Drop here!</div></div></div><img src ="http://www.cppblog.com/zhulf753/aggbug/217547.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2020-12-24 12:34 <a href="http://www.cppblog.com/zhulf753/archive/2020/12/24/217547.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>UCOS2移植到cortex-M0</title><link>http://www.cppblog.com/zhulf753/archive/2018/06/23/215740.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Sat, 23 Jun 2018 02:41:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2018/06/23/215740.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/215740.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2018/06/23/215740.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/215740.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/215740.html</trackback:ping><description><![CDATA[&nbsp;&nbsp;&nbsp;&nbsp; 摘要: 经过别人在板子上几天几夜的测试，跑了6、7个任务同时开了4、5个中断，CPU满负荷。完全没问题，已经用在产品上。主要就修改一个文件os_cpu_a.asm和任务创建时堆栈初始化代码。花了两天时间（以前没接触过cortex-m0也没写过汇编，多年没接触过单片机了）。以下是os_cpu_a.asm。有不少的草稿代码，应该不影响阅读的，请自行忽略。;***************************...&nbsp;&nbsp;<a href='http://www.cppblog.com/zhulf753/archive/2018/06/23/215740.html'>阅读全文</a><img src ="http://www.cppblog.com/zhulf753/aggbug/215740.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2018-06-23 10:41 <a href="http://www.cppblog.com/zhulf753/archive/2018/06/23/215740.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>yuv格式解析</title><link>http://www.cppblog.com/zhulf753/archive/2016/11/01/214371.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Tue, 01 Nov 2016 10:01:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2016/11/01/214371.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/214371.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2016/11/01/214371.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/214371.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/214371.html</trackback:ping><description><![CDATA[&nbsp;&nbsp;&nbsp;&nbsp; 摘要: 相关标准描述参考网站：http://www.fourcc.org/参考解析工具和图片参考网站：http://www.sunrayimage.com/examples.html网上有很多解析YUV格式的工具，觉得还是sunrayimage的工具比较全还有示例图片。只有代码，花半天时间折腾出来的。Code highlighting produced by Actipro CodeHig...&nbsp;&nbsp;<a href='http://www.cppblog.com/zhulf753/archive/2016/11/01/214371.html'>阅读全文</a><img src ="http://www.cppblog.com/zhulf753/aggbug/214371.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2016-11-01 18:01 <a href="http://www.cppblog.com/zhulf753/archive/2016/11/01/214371.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>递归的非递归写法</title><link>http://www.cppblog.com/zhulf753/archive/2008/01/11/40967.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Fri, 11 Jan 2008 09:15:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2008/01/11/40967.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/40967.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2008/01/11/40967.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/40967.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/40967.html</trackback:ping><description><![CDATA[<p>#include&lt;iostream&gt;<br>#include&lt;deque&gt;<br>#include &lt;ctime&gt;<br>using namespace std;<br>template&lt;class _Ty, class _C = deque&lt;_Ty&gt; &gt;<br>class zlfStack {<br>public:<br>&nbsp;typedef unsigned _Ty;<br>&nbsp;typedef _C::allocator_type allocator_type;<br>&nbsp;typedef _C::value_type value_type;<br>&nbsp;typedef _C::size_type size_type;<br>&nbsp;typedef _C::iterator zlfIterator;<br>protected:<br>&nbsp;_C c;<br>public:<br>inline<br>&nbsp;const value_type&amp; zlfTop2(){<br>&nbsp;&nbsp;return *(c.end()-2);<br>&nbsp;}<br>inline<br>&nbsp;const value_type&amp; zlfTop3(){<br>&nbsp;&nbsp;return *(c.end()-3);<br>&nbsp;}<br>inline<br>&nbsp;void top_3(value_type&amp; x,value_type&amp; y,value_type&amp; b)<br>&nbsp;{<br>&nbsp;&nbsp;b=*(c.end()-1);<br>&nbsp;&nbsp;y=*(c.end()-2);<br>&nbsp;&nbsp;x=*(c.end()-3);<br>&nbsp;}<br>inline<br>void top_2(value_type&amp; x,value_type&amp; y)<br>{<br>&nbsp;y=*(c.end()-2);<br>&nbsp;x=*(c.end()-3);<br>}</p>
<p>&nbsp;//zlfStack(){&nbsp;}<br>&nbsp;explicit zlfStack(const allocator_type&amp; _Al = allocator_type())<br>&nbsp;&nbsp;:c(_Al){}<br>&nbsp;allocator_type get_allocator() const<br>&nbsp;{return (c.get_allocator()); }<br>&nbsp;bool empty() const<br>&nbsp;{return (c.empty()); }<br>&nbsp;size_type size() const<br>&nbsp;{return (c.size()); }<br>&nbsp;value_type&amp; top()<br>&nbsp;{return (c.back()); }<br>&nbsp;const value_type&amp; top() const<br>&nbsp;{return (c.back()); }<br>&nbsp;void push(const value_type&amp; _X)<br>&nbsp;{c.push_back(_X); }<br>inline<br>&nbsp;void push_3(const value_type&amp; x,const value_type&amp; y,const value_type&amp; b)<br>&nbsp;{<br>&nbsp;&nbsp;c.push_back(x);<br>&nbsp;&nbsp;c.push_back(y);<br>&nbsp;&nbsp;c.push_back(b);<br>&nbsp;}<br>inline<br>&nbsp;void pop()<br>&nbsp;{c.pop_back(); }<br>&nbsp;};///<br>enum{B0=0,B1=1,B2=2,B3=3};<br>int A(unsigned x,unsigned y)<br>{<br>&nbsp;static count=0;&nbsp;<br>&nbsp;if (!x&amp;&amp;!y) {return ++count;return count;}<br>&nbsp;if (x==0xffff) {count=0;return 0;}<br>&nbsp;if (x) A(--x,y);<br>AB1:&nbsp;if(y) A(x,--y);<br>AB2:<br>&nbsp;&nbsp;return count;<br>&nbsp;&nbsp;<br>}<br>inline<br>void clear(){A(0xffff,0);}<br>zlfStack&lt;unsigned&gt; s;<br>inline<br>void push(unsigned x,unsigned y,unsigned b)<br>{<br>&nbsp;s.push(x);<br>&nbsp;s.push(y);<br>&nbsp;s.push(b);<br>}<br>inline<br>void pop(unsigned&amp; x,unsigned&amp; y,unsigned&amp; b)<br>{<br>&nbsp;b=s.top();<br>&nbsp;s.pop();<br>//&nbsp;y=s.top();<br>&nbsp;s.pop();<br>//&nbsp;x=s.top();<br>&nbsp;s.pop();<br>}</p>
<p><br>int main()<br>{<br>&nbsp;unsigned x=1,y=1,b=1,c=0,z=0;<br>&nbsp;unsigned temp=0;<br>&nbsp;clock_t t1,t2;<br>&nbsp;unsigned k=1;<br>&nbsp;unsigned long sum1=0,sum2=0,time1=0,time2=0;</p>
<p>&nbsp;cout&lt;&lt;"AAAA"&lt;&lt;endl;<br>&nbsp;t1=clock();<br>&nbsp;for (x=1;x&lt;10;x++) {<br>&nbsp;&nbsp;for (y=1;y&lt;10;y++) {&nbsp;<br>&nbsp;&nbsp;&nbsp;clear();<br>&nbsp;&nbsp;&nbsp;k=A(x,y);<br>&nbsp;&nbsp;&nbsp;sum1+=k;<br>&nbsp;&nbsp;&nbsp;cout&lt;&lt;k&lt;&lt;"&nbsp;";<br>&nbsp;&nbsp;&nbsp;cout&lt;&lt;"x="&lt;&lt;x&lt;&lt;"&nbsp;"&lt;&lt;"y="&lt;&lt;y&lt;&lt;endl;<br>&nbsp;&nbsp;}<br>&nbsp;}<br>&nbsp;t2=clock();<br>&nbsp;time1=t2-t1;<br>&nbsp;cout&lt;&lt;endl;</p>
<p><br>&nbsp;if (!x&amp;&amp;!y) return 0;//exit<br>&nbsp;sum2 = 0;<br>&nbsp;t1=clock();<br>&nbsp;for (x=1;x&lt;10;x++) {&nbsp;<br>&nbsp;&nbsp;for (y=1;y&lt;10;y++) {//&nbsp;push(x,y,B3);<br>&nbsp;&nbsp;s.push_3(x,y,B3);<br>&nbsp;&nbsp;c=0;<br>&nbsp;&nbsp;b=B0;<br>&nbsp;&nbsp;while (!s.empty()) {<br>&nbsp;&nbsp;&nbsp;switch(b) {<br>&nbsp;&nbsp;&nbsp;case B0:if(x) {//push(--x,y,B1);<br>&nbsp;&nbsp;&nbsp;&nbsp;s.push_3(--x,y,B1);<br>&nbsp;&nbsp;&nbsp;&nbsp;b=B0;continue;}<br>&nbsp;&nbsp;&nbsp;case B1:if(y) {//push(x,--y,B2);<br>&nbsp;&nbsp;&nbsp;&nbsp;s.push_3(x,--y,B2);<br>&nbsp;&nbsp;&nbsp;&nbsp;b=B0;continue;}<br>&nbsp;&nbsp;&nbsp;case B2:if (!x&amp;&amp;!y) c++;<br>&nbsp;&nbsp;&nbsp;default:;<br>&nbsp;&nbsp;&nbsp;}//switch<br>&nbsp;&nbsp;//&nbsp;pop(x,y,b);<br>&nbsp;&nbsp;&nbsp;b=s.top();<br>&nbsp;&nbsp;&nbsp;s.pop();<br>&nbsp;&nbsp;&nbsp;s.pop();&nbsp;<br>&nbsp;&nbsp;&nbsp;s.pop();<br>&nbsp;&nbsp;&nbsp;if(b==B3) break;//return to main<br>&nbsp;&nbsp;//&nbsp;pop(x,y,temp);<br>&nbsp;&nbsp;//&nbsp;push(x,y,temp);<br>&nbsp;&nbsp;//&nbsp;y=s.zlfTop2();<br>&nbsp;&nbsp;//&nbsp;x=s.zlfTop3();<br>&nbsp;&nbsp;&nbsp;s.top_2(x,y);<br>&nbsp;&nbsp;}//while<br>&nbsp;&nbsp;sum2+=c;<br>&nbsp;//&nbsp;cout&lt;&lt;"c="&lt;&lt;c&lt;&lt;"&nbsp;"&lt;&lt;"x="&lt;&lt;x&lt;&lt;"&nbsp;"&lt;&lt;"y="&lt;&lt;y&lt;&lt;endl;<br>&nbsp;&nbsp;}//y<br>&nbsp;}//x<br>&nbsp;t2=clock();<br>&nbsp;time2=t2-t1;<br>&nbsp;cout&lt;&lt;"time used :"&lt;&lt;time2&lt;&lt;"ms"&lt;&lt;endl;<br>&nbsp;cout&lt;&lt;"routines :"&lt;&lt;sum2&lt;&lt;endl;<br>&nbsp;cout&lt;&lt;endl&lt;&lt;endl;<br>&nbsp;double t;<br>&nbsp;cout&lt;&lt;"routines: "&lt;&lt;sum1&lt;&lt;"&nbsp;&nbsp;time1: "&lt;&lt;time1&lt;&lt;endl;<br>&nbsp;t=sum1/time1;<br>&nbsp;cout&lt;&lt;t&lt;&lt;" rps"&lt;&lt;endl;<br>&nbsp;cout&lt;&lt;"routines: "&lt;&lt;sum2&lt;&lt;"&nbsp;&nbsp;time2: "&lt;&lt;time2&lt;&lt;endl;<br>&nbsp;t=sum2/time2;<br>&nbsp;cout&lt;&lt;t&lt;&lt;" rps"&lt;&lt;endl;<br>&nbsp;return 0;<br>}</p>
<img src ="http://www.cppblog.com/zhulf753/aggbug/40967.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2008-01-11 17:15 <a href="http://www.cppblog.com/zhulf753/archive/2008/01/11/40967.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>各搜索引擎搜索结果的获取</title><link>http://www.cppblog.com/zhulf753/archive/2008/01/10/40925.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Thu, 10 Jan 2008 12:50:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2008/01/10/40925.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/40925.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2008/01/10/40925.html#Feedback</comments><slash:comments>1</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/40925.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/40925.html</trackback:ping><description><![CDATA[<p>用http的get方法，构造要查询的url，get下来，分析结果页面即可<br>首先是构造url，以下是一些示例，主要看清楚?号后面的参数所代表的意思即可：<br><a href="http://www.google.cn/search?num=100&amp;&amp;q=%E5%85%83%E6%90%9C%E7%B4%A2&amp;start=10">http://www.google.cn/search?num=100&amp;&amp;q=%E5%85%83%E6%90%9C%E7%B4%A2&amp;start=10</a></p>
<p><a href="http://www.baidu.com/s?wd=%D4%AA%CB%D1%CB%F7&amp;rn=100&amp;pn=10">http://www.baidu.com/s?wd=%D4%AA%CB%D1%CB%F7&amp;rn=100&amp;pn=10</a>&nbsp; //第二页pn</p>
<p><a href="http://www.yahoo.cn/s?p=%E5%85%83%E6%90%9C%E7%B4%A2&amp;b=10">http://www.yahoo.cn/s?p=%E5%85%83%E6%90%9C%E7%B4%A2&amp;b=10</a>&nbsp; //第二页b</p>
<p><a href="http://search.yahoo.com/search?n=100&amp;p=%E5%85%83%E6%90%9C%E7%B4%A2&amp;b=101">http://search.yahoo.com/search?n=100&amp;p=%E5%85%83%E6%90%9C%E7%B4%A2&amp;b=101</a></p>
<p><a href="http://cnweb.search.live.com/results.aspx?q=%E5%85%83%E6%90%9C%E7%B4%A2&amp;first=51">http://cnweb.search.live.com/results.aspx?q=%E5%85%83%E6%90%9C%E7%B4%A2&amp;first=51</a>&nbsp; //第二页first=51</p>
<p><a href="http://p.zhongsou.com/p?w=%D4%AA%CB%D1%CB%F7&amp;b=3">http://p.zhongsou.com/p?w=%D4%AA%CB%D1%CB%F7&amp;b=3</a>&nbsp; //b=3表示第三页</p>
<p><a href="http://www.soso.com/q?w=%D4%AA%CB%D1%CB%F7&amp;num=20&amp;pg=1">http://www.soso.com/q?w=%D4%AA%CB%D1%CB%F7&amp;num=20&amp;pg=1</a> //第一页，每页20个<br><br>第二步是解释搜索结果页面：<br><br>&lt;meta http-equiv="content-type" content="text/html;charset=gb2312"&gt;</p>
<p>Google<br>搜索结果个数的字符串前缀：约有&lt;b&gt;&nbsp;//获取个数用字符串定位的方式<br>搜索结果开始的标签：&lt;div id=res&gt;&nbsp;//也可以用字符串定位的方式，要准确就用查找标签定位的方式<br>&nbsp;各个搜索结果的开始标签：&lt;div class=g&gt;&nbsp;//字符串定位的方式<br>&nbsp;<br>&nbsp;&nbsp;搜索结果的url在第一个&lt;a target=_blank class=l&gt;标签里头<br>&nbsp;&nbsp;搜索结果的标题在&lt;a&gt;&lt;/a&gt;的标签之间</p>
<p>&nbsp;&nbsp;搜索结果的摘要在接下来的&lt;table&gt;&lt;tr&gt;&lt;td&gt;标签里头直到&lt;b&gt;...&lt;b&gt;&lt;br&gt;<br>&nbsp;&nbsp;搜索结果的重写的url在&lt;b&gt;...&lt;b&gt;&lt;br&gt;之后的&lt;span&gt;标签里头，格式为：url，一个空格，网页大小<br>&nbsp;&nbsp;搜索结果的网页快照在接下来的&lt;a class=fl&gt;的标签里头，属性中有url,标签之间有网页快照文字<br>&nbsp;&nbsp;接下来还有类似网页等，都在&lt;a class=fl&gt;标签里头</p>
<p>&nbsp;各个搜索结果的结束标签是&lt;/td&gt;&lt;/tr&gt;&lt;/table&gt;&lt;/div&gt;</p>
<p>......................</p>
<p>相关搜索的开始标签：&lt;p class=e&gt;<br>在接下来的各个&lt;a&gt;&lt;/a&gt;标签之间的内容就是相关搜索的内容<br>直到标签&lt;br clear=all&gt;就可以结束了</p>
<p>&nbsp;</p>
<p>Baidu<br>搜索结果个数的字符串前缀：&lt;td align=\"righ，在定位该字符串后，直到&lt;/td&gt;,即在这个td标签之内含有的字符串包含相关网页数和用时<br>搜索结果开始的标签：&lt;DIV id=ScriptDiv&gt;&lt;/DIV&gt;<br>&nbsp;各个搜索结果的开始标签：&lt;table</p>
<p>&nbsp;&nbsp;搜索结果的url在第一个&lt;a target=_blank class=l&gt;标签里头<br>&nbsp;&nbsp;搜索结果的标题在&lt;a&gt;&lt;/a&gt;的标签之间,以&lt;br&gt;标签结束<br>&nbsp;&nbsp;<br>&nbsp;&nbsp;搜索结果的摘要以&lt;br&gt;开始直到下一个&lt;br&gt;标签<br>&nbsp;&nbsp;<br>&nbsp;&nbsp;接下来的一行（&lt;br&gt;换行）的font标签中有搜索结果url的重写，一个空格，网页大小，网页时间<br>&nbsp;&nbsp;在接下来会有一些&lt;a&gt;标签如百度快照，直到又一个&lt;br&gt;</p>
<p>&nbsp;然后搜索结果的结束标签&lt;/table&gt;</p>
<p>.........................</p>
<p>导航条的开始标签：&lt;br clear=all&gt;<br>导航条的内容在开始标签之后的&lt;div class="p"&gt;&lt;/div&gt;标签之间<br>相关搜索在接下来的&lt;div&gt;标签之间的各个&lt;a&gt;标签之内<br><br>其他考虑：对于字符串的匹配可以利用kmp，注意到匹配搜索结果各部分的时候所用到的模式字符串的最大前缀字符串最多是一个字符，这样可以避免求取最大前缀字符串从而提高效率；如果要精确地匹配还需要弄两个函数，一个用来构造标签，一个用来读取标签之间的文本。</p>
<img src ="http://www.cppblog.com/zhulf753/aggbug/40925.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2008-01-10 20:50 <a href="http://www.cppblog.com/zhulf753/archive/2008/01/10/40925.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>序列化探讨 </title><link>http://www.cppblog.com/zhulf753/archive/2007/10/08/33745.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Mon, 08 Oct 2007 03:00:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2007/10/08/33745.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/33745.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2007/10/08/33745.html#Feedback</comments><slash:comments>1</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/33745.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/33745.html</trackback:ping><description><![CDATA[<h2><a id=viewpost1_TitleUrl href="http://www.cppblog.com/zhulf753/articles/32067.html">序列化探讨</a> </h2>
<div>主要就是解释readobject和writeobject函数,应该够了，至于在DOC/VIEW模型种使用的话,应该很简单的
<p><span><br>0---</span><span>空指针</span><span><span>&nbsp;&nbsp; </span>7FFF---</span><span>大索引号标志，即后面的索引号是</span><span>32</span><span>位的</span></p>
<p><span>0X8000---</span><span>保留以后使用</span><span><span>&nbsp;&nbsp; </span>0XFFFF---</span><span>新类的定义</span></p>
<p><span>小索引对象或者类的索引号</span><span>:1~~~7FFE,</span><span>但是类的最高位是</span><span>1</span></p>
<p><span>对象的插入：</span><span>writeobject</span><span>函数：（全局插入</span><span>&lt;&lt;</span><span>函数只是调用了这个函数）首先插入类信息，然后是对象信息。每次写入（即一次</span><span>writeobject</span><span>函数的执行流程）是下面三种的之一：</span></p>
<p><span>1<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></span><span>若是未写过的类并且未被写过的对象：</span></p>
<p><span>1.1 </span><span>写入新类标志：</span><span>0XFFFF<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>*this &lt;&lt; wNewClassTag;</span></p>
<p><span>1.2 </span><span>写入类的版本号，写入类名的字符串长度：</span><span>ar &lt;&lt; (WORD)m_wSchema &lt;&lt; nLen;</span></p>
<p><span>1.3 </span><span>写入类名</span><span>&nbsp;ar.Write(m_lpszClassName, nLen*sizeof(char));</span></p>
<p><span>1.4&nbsp;</span><span>写入对象</span><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>((CObject*)pOb)-&gt;Serialize(*this);</span></p>
<p><span>2<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></span><span>若是曾经写过的类并且未写过的对象：</span></p>
<p><span>2.1 </span><span>写入类的索引号</span> <span>nClassIndex = (DWORD)(*m_pStoreMap)[(void*)pClassRef]</span></p>
<p><span>如果是小索引类：则写入类</span><span>*this &lt;&lt; (WORD)(wClassTag | nClassIndex);</span></p>
<p><span>&nbsp;</span><span>如果是大索引类：则写入大索引号标志（</span><span>7FFF</span><span>）和</span><span>32</span><span>位的类索引号（最高位是</span><span>1</span><span>）</span><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>*this &lt;&lt; wBigObjectTag;</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>*this &lt;&lt; (dwBigClassTag | nClassIndex);</span></p>
<p><span>2.2<span>&nbsp;&nbsp; </span></span><span>写入对象</span><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>((CObject*)pOb)-&gt;Serialize(*this);</span></p>
<p><span>3<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></span><span>若是曾经写过的类并且曾经写过的对象：</span></p>
<p><span>3.1&nbsp;</span><span>写入对象的索引号</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>&nbsp;</span><span>如果是小索引对象：则直接写入索引号</span><span>*this &lt;&lt; (WORD)nObIndex;</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>&nbsp;</span><span>如果是大索引对象：则写入大索引号标志和</span><span>32</span><span>位的对象索引号（最高位是</span><span>0</span><span>）</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>*this &lt;&lt; wBigObjectTag;</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>*this &lt;&lt; nObIndex;</span></p>
<p><span>以上</span><span>3</span><span>种情况的写入都是首先写入一个字，这个字的内容就代表了之后字节即类信息的意义或者可以只是一个对象的索引号（情况三），即是新类（字节为</span><span>0xFFFF</span><span>）还是已经定过的小索引类（索引号从</span><span>0x8001—0xFFFE</span><span>）又或者是已经定义过的大索引类以</span><span>(</span><span>字节</span><span>0x7FFF</span><span>后续两个字节为索引号</span><span>)</span><span>。</span></p>
<p>&nbsp;</p>
<p><span>对于读取上面文件格式的信息并且区分出将要读取的是类还是对象，是索引还是对象数据，在</span><span>readObject</span><span>中</span></p>
<p><span>首先读取一个字<span>如果</span>是</span><span>0XFFFF</span><span>则显然对应的是第一种情况，此时可以容易地读取数据；<span>如果</span>第一个字的最高位是</span><span>1</span><span>的话，很显然此时对应的就是第二种情况，即该字是一个类的索引号，而且是小索引类；<span>如果</span>是</span><span>0x7FFF</span><span>则此时对应的就是第三种情况大索引号对象或者第二种情况大索引号类；<span>如果</span>最高位不是</span><span>1</span><span>则此时对应的也是第三种情况但是小索引对象；在区分了第一个字以后就可以容易地读取后面的数据。这样每次的</span><span>readObject</span><span>函数的调用都只是提取以往某次</span><span>writeObject</span><span>函数写入的数据。</span></p>
<p>&nbsp;</p>
<p><span>对象的提取</span><span>:ReadObject</span><span>函数，因为在宏</span><span>IMPLEMENT_SERIAL</span><span>里定义的提取操作符只是简单地调用了这个函数。首先提取类信息，以便正确地动态生成对象，然后是对象信息。</span></p>
<p><span>PS:m_pStoreMap</span><span>里即包含了已经序列化的类（</span><span>CRuntimeClass</span><span>）和对象的指针。</span></p>
<p>&nbsp;</p>
<p><span>UINT CArchive::GetObjectSchema()</span><span>函数只能调用一次，一般该函数在某个类（</span><span>ar</span><span>正在序列化的类）的</span><span>Serialize</span><span>函数里头调用，它返回读取的类的版本号。以下几行来自</span><span>readObject: </span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>UINT nSchemaSave = m_nObjectSchema;</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>m_nObjectSchema = nSchema;</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>pOb-&gt;Serialize(*this);</span></p>
<p><span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>m_nObjectSchema = nSchemaSave;</span></p>
<p><span>一般地，也正是在</span><span>serialize</span><span>里头来处理各种版本的序列化。</span></p>
<p><span>FAQ:</span></p>
<p><span>1．&nbsp;</span><span>为什么可以定义全局的插入操作符，而提取操作符却要对每个类在宏里头声明？</span></p>
<p><span>插入操作的是在已知对象所有信息的情况下的操作，包括对象的类型和状态，类信息的写入使用</span><span>CruntimeClass</span><span>非静态</span><span>成员函数</span><span>Store</span><span>来实现的，而</span><span>GetCRuntime</span><span>成员函数又是虚函数所以可以用指向</span><span>COBJECT</span><span>的指针来正确地获取，然后正确地调用</span><span>STORE</span><span>函数；而对于提取操作，将要提取出的对象的类型和状态都是未知，提取类信息主要是用</span><span>CruntimeClass</span><span>的</span><span>静态</span><span>成员</span><span>LOAD</span><span>来获取，该成员获得文件中类信息之后通过查找全局的类型链表可以唯一地确定一个</span><span>CrutimeClass</span><span>类型的静态对象，通过该对象的</span><span>createObject</span><span>函数可以构造出即将提取的对象类型，然后利用动态构造的对象的序列化函数就可以正确地提取出对象，因为</span></p>
<p><span>1</span><span>．</span><span>1<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></span><span>在宏定义的提取操作符中</span><span>classname</span><span>参数是无法用</span><span>COBJECT</span><span>类来替换，因为如果替换的话则在提取信息过程中即使出现错误，比如说提取出的类型根本就不是可序列化的但是如果继承自</span><span>COBJECT</span><span>的话仍然可以通过错误检查。</span></p>
</div>
<img src ="http://www.cppblog.com/zhulf753/aggbug/33745.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2007-10-08 11:00 <a href="http://www.cppblog.com/zhulf753/archive/2007/10/08/33745.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>AVL树的简单实现</title><link>http://www.cppblog.com/zhulf753/archive/2007/10/05/33504.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Fri, 05 Oct 2007 07:47:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2007/10/05/33504.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/33504.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2007/10/05/33504.html#Feedback</comments><slash:comments>2</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/33504.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/33504.html</trackback:ping><description><![CDATA[<p>#include &lt;math.h&gt;<br>#include &lt;vector&gt;<br>using namespace std;<br>#define max(a,b)&nbsp; ((a)&gt;(b)?(a):(b))<br>template&lt;typename E&gt;<br>class AVL_TMP<br>{&nbsp;<br>&nbsp;template &lt;typename E&gt;<br>&nbsp;class AVL_NODE<br>&nbsp;{<br>&nbsp;public:<br>&nbsp;&nbsp;AVL_NODE():ln(0),rn(0),depth(0){}<br>&nbsp;&nbsp;AVL_NODE( const E&amp; e):data(e),ln(0),rn(0),depth(0){}<br>&nbsp;&nbsp;~AVL_NODE(){&nbsp;if (ln) delete ln;&nbsp;if (rn) delete rn;&nbsp;}</p>
<p>&nbsp;&nbsp;bool operator &lt; (E&amp; e){&nbsp;&nbsp;return data &lt; e;&nbsp;}<br>&nbsp;&nbsp;bool operator &gt; (E&amp; e){&nbsp;&nbsp;return data &gt; e;&nbsp;}<br>&nbsp;&nbsp;bool operator == (E&amp; e){&nbsp;return data == e;&nbsp;}<br>&nbsp;&nbsp;bool operator != (E&amp; e){&nbsp;return data != e;&nbsp;}</p>
<p>&nbsp;&nbsp;E getdata(){return data;}<br>&nbsp;&nbsp;<br>&nbsp;&nbsp;E data;<br>&nbsp;&nbsp;int depth;<br>&nbsp;&nbsp;AVL_NODE&lt;E&gt; *ln,*rn;<br>&nbsp;};<br>&nbsp;<br>public:&nbsp;<br>&nbsp;typedef E dataType;<br>&nbsp;typedef AVL_TMP&lt;E&gt; Myt;<br>&nbsp;typedef AVL_NODE&lt;E&gt; n;<br>&nbsp;typedef n* npos;<br>&nbsp;typedef npos iterator;<br>&nbsp;enum unbalanceType {LL,RR,LR,RL};<br>&nbsp;AVL_TMP():root(0),size(0),depth(-1){}<br>&nbsp;~AVL_TMP(){ if(root) delete root; }</p>
<p>&nbsp;iterator begin(){return root;}<br>&nbsp;bool insert(const E&amp; e);<br>&nbsp;npos find(const E&amp; e);<br>&nbsp;npos findpre(const E&amp; e);<br>&nbsp;bool del(dataType);<br>&nbsp;bool balance(AVL_TMP&lt;E&gt;::iterator pos){ <br>&nbsp;&nbsp;if(pos == NULL) throw 0;<br>&nbsp;&nbsp;int lh,rh;<br>&nbsp;&nbsp;if(pos-&gt;ln == NULL ) lh = -1;<br>&nbsp;&nbsp;else lh = pos-&gt;ln-&gt;depth;<br>&nbsp;&nbsp;if(pos-&gt;rn == NULL ) rh = -1;<br>&nbsp;&nbsp;else rh = pos-&gt;rn-&gt;depth;<br>&nbsp;&nbsp;return abs( lh - rh ) &lt; 2 ;<br>&nbsp;}<br>&nbsp;virtual&nbsp;void frontOrder(){};<br>&nbsp;virtual void midOrder(){ };</p>
<p>protected:<br>&nbsp;void LLr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos);<br>&nbsp;void LRr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos);<br>&nbsp;void RRr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos);<br>&nbsp;void RLr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos);<br>&nbsp;void updateDepth(AVL_TMP&lt;E&gt;::iterator pos);<br>&nbsp;bool delAux(E const&amp; e,AVL_TMP&lt;E&gt;::iterator pos = NULL);<br>&nbsp;iterator findMax(iterator );<br>&nbsp;iterator findMin(iterator );<br>&nbsp;bool upTree(int iDepth,iterator itRoot,unsigned long iSize){depth = iDepth;root = itRoot;size = iSize; return true;}<br>&nbsp;bool upRoutineDepth(vector&lt;iterator&gt;&amp;);<br>&nbsp;bool adjust(iterator a,iterator b,iterator c,iterator prePos = NULL);<br>&nbsp;npos root;<br>&nbsp;int depth;<br>&nbsp;unsigned long size;<br>};<br>template&lt;typename E&gt;<br>bool AVL_TMP&lt;E&gt;::adjust(iterator a,iterator b,iterator c,iterator prePos){<br>&nbsp;bool b1,b2;<br>&nbsp;b1 = b == a-&gt;ln;<br>&nbsp;b2 = c == b-&gt;ln;<br>&nbsp;unbalanceType ub;<br>&nbsp;if(b1&amp;&amp;!b2)&nbsp;&nbsp; ub = LR;<br>&nbsp;if(!b1&amp;&amp;b2)&nbsp;&nbsp; ub = RL;<br>&nbsp;if(b1&amp;&amp;b2)&nbsp;&nbsp;&nbsp; ub = LL;<br>&nbsp;if(!b1&amp;&amp;!b2)&nbsp; ub = RR;<br>&nbsp;switch(ub) {<br>&nbsp;&nbsp;case&nbsp; LL :LLr(a,prePos);<br>&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;case&nbsp; LR :LRr(a, prePos);<br>&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;case&nbsp; RR :RRr(a,prePos);<br>&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;case&nbsp; RL :RLr(a,prePos);<br>&nbsp;&nbsp;&nbsp;break;<br>&nbsp;}&nbsp; //end switch<br>&nbsp;return true;<br>}<br>template&lt;typename E&gt;<br>bool AVL_TMP&lt;E&gt;::upRoutineDepth(vector&lt;iterator&gt;&amp;routine){<br>&nbsp;//该函数主要是将路径节点的深度更新并且使得那些不平衡的节点平衡<br>&nbsp;int size = routine.size();<br>&nbsp;while (size--) {<br>&nbsp;&nbsp;updateDepth(routine[size]);<br>&nbsp;&nbsp;if (!balance(routine[size])) {//不平衡得调整<br>&nbsp;&nbsp;&nbsp;iterator cur = routine[size],prePos = NULL;<br>&nbsp;&nbsp;&nbsp;if(size-1&gt;=0)<br>&nbsp;&nbsp;&nbsp;&nbsp;prePos = routine[size-1];<br>&nbsp;&nbsp;&nbsp;//检查当前不平衡节点的哪颗子树的高度更高<br>&nbsp;&nbsp;&nbsp;bool bl = cur-&gt;ln != NULL;<br>&nbsp;&nbsp;&nbsp;bool br = cur-&gt;rn != NULL;<br>&nbsp;&nbsp;&nbsp;if (!bl) {//肯定有右孩子<br>&nbsp;&nbsp;&nbsp;&nbsp;if(cur-&gt;rn-&gt;ln) RLr(cur,prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;else RRr(cur,prePos);<br>&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;&nbsp;else{//有左孩子<br>&nbsp;&nbsp;&nbsp;&nbsp;if (!br) {//没右孩子<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if (cur-&gt;ln-&gt;ln) LLr(cur,prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else LRr(cur,prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;&nbsp;&nbsp;else{ //有右孩子,此时需要检查左右孩子的高度，则右子树高度至少为1<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//因此左子树高度至少为3，则左子树的节点个数肯定大于4<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if (cur-&gt;ln-&gt;depth &gt; cur-&gt;rn-&gt;depth) LLr(cur,prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else RRr(cur,prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;}<br>&nbsp;}<br>&nbsp;return true;<br>}<br>template&lt;typename E&gt;<br>AVL_TMP&lt;E&gt;::iterator AVL_TMP&lt;E&gt;::findMax(AVL_TMP&lt;E&gt;::iterator pos){//以pos为根的树的最大值的节点<br>&nbsp;if (!pos) return NULL;<br>&nbsp;iterator p = pos;<br>&nbsp;while(p-&gt;rn) p = p-&gt;rn;<br>&nbsp;return p;<br>}<br>template&lt;typename E&gt;<br>AVL_TMP&lt;E&gt;::iterator AVL_TMP&lt;E&gt;::findMin(AVL_TMP&lt;E&gt;::iterator pos){<br>&nbsp;iterator p = pos;<br>&nbsp;while (p-&gt;ln) p = p-&gt;ln;<br>&nbsp;return p;<br>}<br>template&lt;typename E&gt;<br>void AVL_TMP&lt;E&gt;::updateDepth(AVL_TMP&lt;E&gt;::iterator pos){<br>&nbsp;bool b1 = pos-&gt;ln == NULL,b2 = pos-&gt;rn ==NULL;<br>&nbsp;switch(b1) {<br>&nbsp;case true:<br>&nbsp;&nbsp;if(b2) pos-&gt;depth = 0;<br>&nbsp;&nbsp;else pos-&gt;depth = pos-&gt;rn-&gt;depth+1;<br>&nbsp;&nbsp;break;<br>&nbsp;default: //false<br>&nbsp;&nbsp;if(b2)&nbsp; pos-&gt;depth = pos-&gt;ln-&gt;depth+1;<br>&nbsp;&nbsp;else pos-&gt;depth = max(pos-&gt;ln-&gt;depth , pos-&gt;rn-&gt;depth )+1;<br>&nbsp;}<br>&nbsp;if(pos == root) depth = pos-&gt;depth;<br>}<br>template&lt;typename E&gt;<br>void AVL_TMP&lt;E&gt;::LLr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos){<br>&nbsp;typename AVL_TMP&lt;E&gt;::iterator t, a = pos, b = t = pos-&gt;ln ;<br>&nbsp;pos-&gt;ln = t-&gt;rn;<br>&nbsp;t-&gt;rn = pos;<br>&nbsp;if(root == a) root = b;<br>&nbsp;if(prePos != NULL) <br>&nbsp;&nbsp;if(prePos-&gt;ln == a) prePos-&gt;ln = b;<br>&nbsp;&nbsp;else prePos-&gt;rn =&nbsp; b;<br>&nbsp;updateDepth(a);updateDepth(b);<br>}<br>template&lt;typename E&gt;<br>void AVL_TMP&lt;E&gt;::LRr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos){<br>&nbsp;AVL_TMP&lt;E&gt;::iterator a = pos,b = pos -&gt;ln, c = b-&gt;rn;<br>&nbsp;b-&gt;rn = c-&gt;ln ; a-&gt;ln = c-&gt;rn;<br>&nbsp;c-&gt;ln = b;&nbsp; c-&gt;rn =a;<br>&nbsp;if(a == root ) root = c ;<br>&nbsp;if(prePos != NULL) <br>&nbsp;&nbsp;if(prePos-&gt;ln == a) prePos-&gt;ln = c;<br>&nbsp;&nbsp;else prePos-&gt;rn =&nbsp; c;<br>&nbsp;updateDepth(a);updateDepth(b);updateDepth(c);<br>}<br>template&lt;typename E&gt;<br>void AVL_TMP&lt;E&gt;::RRr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos ){<br>&nbsp;AVL_TMP&lt;E&gt;::iterator a = pos ,t, b = t = pos-&gt;rn ;<br>&nbsp;pos-&gt;rn = t-&gt;ln; <br>&nbsp;t-&gt;ln = pos;<br>&nbsp;if(prePos != NULL) <br>&nbsp;&nbsp;if(prePos-&gt;ln == a) prePos-&gt;ln = b;<br>&nbsp;&nbsp;else prePos-&gt;rn =&nbsp; b;<br>&nbsp;if(root == a) root = b;<br>&nbsp;updateDepth(a);updateDepth(b);<br>}<br>template&lt;typename E&gt;<br>void AVL_TMP&lt;E&gt;::RLr(AVL_TMP&lt;E&gt;::iterator pos,AVL_TMP&lt;E&gt;::iterator prePos){<br>&nbsp;AVL_TMP&lt;E&gt;::iterator a = pos, b = pos-&gt;rn , c = b-&gt;ln;<br>&nbsp;a-&gt;rn = c-&gt;ln ;&nbsp; b-&gt;ln = c-&gt;rn;<br>&nbsp;c-&gt;ln = a; c-&gt;rn = b;<br>&nbsp;if(prePos != NULL) <br>&nbsp;&nbsp;if(prePos-&gt;ln == a) prePos-&gt;ln = c;<br>&nbsp;&nbsp;else prePos-&gt;rn =&nbsp; c;<br>&nbsp;if( a == root) root = c;<br>&nbsp;updateDepth(a);updateDepth(b);updateDepth(c);<br>}<br>template&lt;typename E&gt;<br>bool AVL_TMP&lt;E&gt;::insert(const E&amp; e){<br>&nbsp;if(root == NULL) {root = new AVL_NODE&lt;E&gt;(e); size++; depth = root-&gt;depth;return true;}<br>&nbsp;bool bUpdateDepth = false;<br>&nbsp;vector&lt;AVL_TMP&lt;E&gt;::iterator&gt; routin;<br>&nbsp;typename AVL_TMP&lt;E&gt;::iterator p = root,pos,prePos;<br>&nbsp;for (int i = 0 ; i &lt; size ;i++ ) {<br>&nbsp;&nbsp;routin.push_back(p);<br>&nbsp;&nbsp;if(p-&gt;data &gt; e){<br>&nbsp;&nbsp;&nbsp;if ( p-&gt;ln == NULL ) {<br>&nbsp;&nbsp;&nbsp;&nbsp;p-&gt;ln = pos = new AVL_NODE&lt;E&gt;(e);<br>&nbsp;&nbsp;&nbsp;&nbsp;bUpdateDepth = p-&gt;rn == NULL;<br>&nbsp;&nbsp;&nbsp;&nbsp;break; <br>&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;&nbsp;else { p = p-&gt;ln ; continue;}<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;if(p-&gt;data&nbsp; &lt; e){<br>&nbsp;&nbsp;&nbsp;if (p-&gt;rn == NULL) { <br>&nbsp;&nbsp;&nbsp;&nbsp;p-&gt;rn = pos = new AVL_NODE&lt;E&gt;(e) ;<br>&nbsp;&nbsp;&nbsp;&nbsp;bUpdateDepth = p-&gt;ln == NULL;<br>&nbsp;&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;&nbsp;else {&nbsp; p = p-&gt;rn ; continue;}<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;return false;&nbsp;&nbsp; //already exists<br>&nbsp;}&nbsp;&nbsp;//insertion finished<br>&nbsp;size++;<br>&nbsp;if(size &lt;= 2 ) {<br>&nbsp;&nbsp;updateDepth(root);<br>&nbsp;&nbsp;return true;<br>&nbsp;}<br>&nbsp;if(!bUpdateDepth) return true;&nbsp;&nbsp;&nbsp;//balance<br>&nbsp;<br>&nbsp;bool unAdjusted = true;<br>&nbsp;// check for balance and adjust depth<br>&nbsp;for (i = routin.size()-1; i&nbsp; &gt;=0 ; i-- ) {<br>&nbsp;&nbsp;if(!balance(routin.at(i))) <br>&nbsp;&nbsp;&nbsp;if(unAdjusted) {&nbsp; //&nbsp; unbalance! get unbalance type<br>&nbsp;&nbsp;&nbsp;&nbsp;if(i-1 &gt;= 0) prePos = routin.at(i-1);<br>&nbsp;&nbsp;&nbsp;&nbsp;else prePos = NULL;<br>&nbsp;&nbsp;&nbsp;&nbsp;AVL_TMP&lt;E&gt;::iterator a = routin.at(i) , b = routin.at(i+1) , c;<br>&nbsp;&nbsp;&nbsp;&nbsp;if(i+2 &gt;= routin.size() ) c = pos;<br>&nbsp;&nbsp;&nbsp;&nbsp;else c = routin.at(i+2);<br>&nbsp;&nbsp;&nbsp;&nbsp;bool b1,b2;<br>&nbsp;&nbsp;&nbsp;&nbsp;b1 = b == a-&gt;ln;<br>&nbsp;&nbsp;&nbsp;&nbsp;b2 = c == b-&gt;ln;<br>&nbsp;&nbsp;&nbsp;&nbsp;unbalanceType ub;<br>&nbsp;&nbsp;&nbsp;&nbsp;if(b1&amp;&amp;!b2)&nbsp;&nbsp; ub = LR;<br>&nbsp;&nbsp;&nbsp;&nbsp;if(!b1&amp;&amp;b2)&nbsp;&nbsp; ub = RL;<br>&nbsp;&nbsp;&nbsp;&nbsp;if(b1&amp;&amp;b2)&nbsp;&nbsp;&nbsp; ub = LL;<br>&nbsp;&nbsp;&nbsp;&nbsp;if(!b1&amp;&amp;!b2)&nbsp; ub = RR;</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;switch(ub) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;case&nbsp; LL :LLr(routin.at(i),prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;case&nbsp; LR :LRr(routin.at(i),prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;case&nbsp; RR :RRr(routin.at(i),prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;case&nbsp; RL :RLr(routin.at(i),prePos);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp; //end switch<br>&nbsp;&nbsp;&nbsp;&nbsp;unAdjusted = false;<br>&nbsp;&nbsp;&nbsp;}&nbsp; //end if<br>&nbsp;updateDepth(routin.at(i));&nbsp;&nbsp;//update the depth of the node in the routin<br>&nbsp;depth = root-&gt;depth;<br>&nbsp;}//end for<br>&nbsp;return true;<br>};<br>template&lt;typename E&gt;<br>AVL_TMP&lt;E&gt;::npos AVL_TMP&lt;E&gt;::find(const E&amp; e){//search for position<br>&nbsp;&nbsp;&nbsp;npos p=root;<br>&nbsp;&nbsp;&nbsp;while (p&amp;&amp;p-&gt;data!=e) <br>&nbsp;&nbsp;&nbsp;&nbsp;if(e&gt;p-&gt;data) p=p-&gt;rn;<br>&nbsp;&nbsp;&nbsp;&nbsp;else p= p-&gt;ln;<br>&nbsp;&nbsp;&nbsp;return p;<br>}<br>template&lt;typename E&gt;<br>AVL_TMP&lt;E&gt;::npos AVL_TMP&lt;E&gt;::findpre(const E&amp; e){//search for parent node position<br>&nbsp;&nbsp;&nbsp;npos p,pre;<br>&nbsp;&nbsp;&nbsp;p=pre=root;<br>&nbsp;&nbsp;&nbsp;while (p&amp;&amp;p-&gt;data!=e) {<br>&nbsp;&nbsp;&nbsp;&nbsp;pre = p;<br>&nbsp;&nbsp;&nbsp;&nbsp;if (e&gt;p-&gt;data) p=p-&gt;rn;<br>&nbsp;&nbsp;&nbsp;&nbsp;else p = p-&gt;ln;<br>&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;&nbsp;if(p) if(p-&gt;data==e) return NULL;//already existed <br>&nbsp;&nbsp;&nbsp;return pre;<br>}<br>template&lt;typename E&gt;<br>bool AVL_TMP&lt;E&gt;::delAux(E const&amp; e,AVL_TMP&lt;E&gt;::iterator pos){<br>&nbsp;// 1.递归删除节点，直到删除的是叶子节点&nbsp;<br>&nbsp;// 2. 删除叶子节点,更新树的数据成员 <br>&nbsp;// 3. 更新路径上的节点深度并且检查平衡因子&nbsp;<br>&nbsp;static vector&lt;iterator&gt; routine;<br>&nbsp;iterator p = pos;<br>&nbsp;bool bUpdate = false;<br>&nbsp;if(!pos){//第一次调用<br>&nbsp;&nbsp;p = root;<br>&nbsp;&nbsp;while (p&amp;&amp;e!=p-&gt;data) {//找到节点，并且将寻找路径存入表中<br>&nbsp;&nbsp;&nbsp;routine.push_back(p);<br>&nbsp;&nbsp;&nbsp;if(p-&gt;data &gt; e) p = p-&gt;ln;<br>&nbsp;&nbsp;&nbsp;else p = p-&gt;rn;<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;if(p == NULL){ //没找到<br>&nbsp;&nbsp;&nbsp;routine.clear();&nbsp;<br>&nbsp;&nbsp;&nbsp;return false;<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;else pos = p;<br>&nbsp;}<br>&nbsp;if (pos-&gt;ln||pos-&gt;rn) {//不是叶子节点，则该节点有孩子节点，可能是一个或者两个<br>&nbsp;&nbsp;routine.push_back(pos);//还得往下删除<br>&nbsp;&nbsp;if (pos-&gt;ln&amp;&amp;!pos-&gt;rn){ //情况一: 只有有左孩子<br>&nbsp;&nbsp;&nbsp;//找到左子树中的最大值的位置<br>&nbsp;&nbsp;&nbsp;iterator max = pos-&gt;ln;<br>&nbsp;&nbsp;&nbsp;while (max-&gt;rn) { routine.push_back(max); max = max-&gt;rn;}<br>&nbsp;&nbsp;&nbsp;bUpdate = false;<br>&nbsp;&nbsp;&nbsp;//伪删除<br>&nbsp;&nbsp;&nbsp;pos-&gt;data = max-&gt;data;<br>&nbsp;&nbsp;&nbsp;delAux(max-&gt;data,max);<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;else if (!pos-&gt;ln&amp;&amp;pos-&gt;rn) { //情况二：只有右孩子<br>&nbsp;&nbsp;&nbsp;//找到右子树中的最小值<br>&nbsp;&nbsp;&nbsp;iterator min = pos-&gt;rn;<br>&nbsp;&nbsp;&nbsp;while (min-&gt;ln) { routine.push_back(min); min = min-&gt;ln;}<br>&nbsp;&nbsp;&nbsp;bUpdate = false;<br>&nbsp;&nbsp;&nbsp;//伪删除<br>&nbsp;&nbsp;&nbsp;pos-&gt;data = min-&gt;data;<br>&nbsp;&nbsp;&nbsp;delAux(min-&gt;data,min);<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;else //情况三：有俩个孩子<br>&nbsp;&nbsp;{<br>&nbsp;&nbsp;&nbsp;//找到左子树中的最大值<br>&nbsp;&nbsp;&nbsp;iterator max = pos-&gt;ln;<br>&nbsp;&nbsp;&nbsp;while (max-&gt;rn) { routine.push_back(max); max = max-&gt;rn;}<br>&nbsp;&nbsp;&nbsp;bUpdate = false;<br>&nbsp;&nbsp;&nbsp;//伪删除<br>&nbsp;&nbsp;&nbsp;pos-&gt;data = max-&gt;data;<br>&nbsp;&nbsp;&nbsp;delAux(max-&gt;data,max);<br>&nbsp;&nbsp;}<br>&nbsp;} <br>&nbsp;else <br>&nbsp;{//是叶子节点<br>&nbsp;&nbsp;//有三种情况，是其父节点的左子树且没有兄弟，是其父节点的右子树且没有兄弟，有兄弟<br>&nbsp;&nbsp;//取得其父节点<br>&nbsp;&nbsp;iterator parent = NULL;<br>&nbsp;&nbsp;if (routine.size()) //有父节点<br>&nbsp;&nbsp;&nbsp;parent = routine[routine.size()-1];<br>&nbsp;&nbsp;else{//即该节点是根节点,无根节点<br>&nbsp;&nbsp;&nbsp;delete root;<br>&nbsp;&nbsp;&nbsp;routine.clear();<br>&nbsp;&nbsp;&nbsp;upTree(-1,NULL,0);<br>&nbsp;&nbsp;&nbsp;return true;<br>&nbsp;&nbsp;}&nbsp; //完成根节点的删除<br>&nbsp;&nbsp;//有父节点<br>&nbsp;&nbsp;if (pos == parent-&gt;ln&amp;&amp;!parent-&gt;rn) {//情况一：是父节点的左孩子且没兄弟<br>&nbsp;&nbsp;&nbsp;//删除节点<br>&nbsp;&nbsp;&nbsp;parent-&gt;ln = NULL;<br>&nbsp;&nbsp;&nbsp;delete pos;<br>&nbsp;&nbsp;&nbsp;//需要更新路径上的节点的深度<br>&nbsp;&nbsp;&nbsp;bUpdate = true;<br>&nbsp;&nbsp;&nbsp;upRoutineDepth(routine);<br>&nbsp;&nbsp;&nbsp;upTree(root-&gt;depth,root,size-1);<br>&nbsp;&nbsp;&nbsp;routine.clear();<br>&nbsp;&nbsp;&nbsp;//改写父节点的孩子指针<br>&nbsp;&nbsp;}//完成情况一叶子节点的删除<br>&nbsp;&nbsp;else{<br>&nbsp;&nbsp;&nbsp;if (pos == parent-&gt;rn &amp;&amp; !parent-&gt;ln ) { //情况二：是父节点的右孩子且没兄弟<br>&nbsp;&nbsp;&nbsp;&nbsp;parent-&gt;rn = NULL;<br>&nbsp;&nbsp;&nbsp;&nbsp;delete pos;&nbsp;<br>&nbsp;&nbsp;&nbsp;&nbsp;bUpdate = true;<br>&nbsp;&nbsp;&nbsp;&nbsp;upRoutineDepth(routine);<br>&nbsp;&nbsp;&nbsp;&nbsp;upTree(root-&gt;depth,root,size-1);<br>&nbsp;&nbsp;&nbsp;&nbsp;routine.clear();<br>&nbsp;&nbsp;&nbsp;}//完成情况二叶子节点的删除<br>&nbsp;&nbsp;&nbsp;else{//情况三：有兄弟<br>&nbsp;&nbsp;&nbsp;&nbsp;//只需要将节点删除，并清理路径表就可以了<br>&nbsp;&nbsp;&nbsp;&nbsp;if (pos == parent-&gt;ln) parent-&gt;ln = NULL;<br>&nbsp;&nbsp;&nbsp;&nbsp;else parent-&gt;rn = NULL;<br>&nbsp;&nbsp;&nbsp;&nbsp;delete pos;<br>&nbsp;&nbsp;&nbsp;&nbsp;routine.clear();<br>&nbsp;&nbsp;&nbsp;}//完成情况三的叶子节点删除<br>&nbsp;&nbsp;}<br>&nbsp;}<br>&nbsp;return true;<br>}</p>
<p>template&lt;typename E&gt;<br>bool AVL_TMP&lt;E&gt;::del(dataType e){<br>&nbsp;return delAux(e);<br>}<br></p>
<img src ="http://www.cppblog.com/zhulf753/aggbug/33504.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2007-10-05 15:47 <a href="http://www.cppblog.com/zhulf753/archive/2007/10/05/33504.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>八皇后的简单解法</title><link>http://www.cppblog.com/zhulf753/archive/2007/09/14/32210.html</link><dc:creator>zlf</dc:creator><author>zlf</author><pubDate>Fri, 14 Sep 2007 06:41:00 GMT</pubDate><guid>http://www.cppblog.com/zhulf753/archive/2007/09/14/32210.html</guid><wfw:comment>http://www.cppblog.com/zhulf753/comments/32210.html</wfw:comment><comments>http://www.cppblog.com/zhulf753/archive/2007/09/14/32210.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/zhulf753/comments/commentRss/32210.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/zhulf753/services/trackbacks/32210.html</trackback:ping><description><![CDATA[<p>#include &lt;iostream.h&gt;<br>#include&lt;io.h&gt;<br>const int QUENS = 7;<br>int getcol(int q[],int n,const int firstCol)<br>{//q表示每个皇后的列位置，firstCol是搜索列位置时的起始位置<br>//n表示正在搜索第n个皇后的列位置,皇后的名称从0--n编号，也是皇后的行号<br>&nbsp;bool b = true;<br>&nbsp;int i = firstCol;<br>&nbsp;for (i = firstCol; i&lt;=QUENS ; i++) {<br>&nbsp;&nbsp;for (int j=0; j&lt;n; j++) {<br>&nbsp;&nbsp;&nbsp;if(q[j] == i || (n+q[j]==i+j)||(n+i == j+q[j])){ <br>&nbsp;&nbsp;&nbsp;&nbsp;b = false;<br>&nbsp;&nbsp;&nbsp;&nbsp;break;<br>&nbsp;&nbsp;&nbsp;}<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;if(j == n) <br>&nbsp;&nbsp;&nbsp; return i;<br>&nbsp;&nbsp;else if(firstCol &gt; QUENS) return firstCol;<br>&nbsp;}<br>&nbsp;if(!b) return QUENS+1;<br>}<br>void EQ(int q[],int n){<br>&nbsp;void disp(int[]);<br>&nbsp;int col = QUENS+1;<br>&nbsp;bool b = true;<br>&nbsp;int firstCol = 0;<br>&nbsp;while (QUENS &gt;= (col=getcol(q,n,firstCol))){<br>&nbsp;&nbsp;if(QUENS == n){<br>&nbsp;&nbsp;&nbsp;q[n] = col;<br>&nbsp;&nbsp;&nbsp;disp(q);<br>&nbsp;&nbsp;&nbsp;return ;<br>&nbsp;&nbsp;}<br>&nbsp;&nbsp;else{<br>&nbsp;&nbsp;&nbsp;q[n] = col;<br>&nbsp;&nbsp;&nbsp;firstCol = col +1;<br>&nbsp;&nbsp;&nbsp;EQ(q,n+1);<br>&nbsp;&nbsp;}<br>&nbsp;}<br>&nbsp;<br>}<br>void disp(int q[])<br>{//显示一种排列<br>&nbsp;static count = 0;<br>&nbsp;count++;<br>&nbsp;cout&lt;&lt;"number "&lt;&lt;count&lt;&lt;"&nbsp;: ";<br>&nbsp;for (int i=0; i&lt;=QUENS; i++) <br>&nbsp;&nbsp;cout&lt;&lt;q[i]+1&lt;&lt;" ";<br>&nbsp;cout&lt;&lt;endl;<br>}<br>void outTofile()<br>{//由于结果比较多，所以把结果重定向输出到文件里头了，文件名是EightQuen.txt<br>&nbsp;int old = _dup(1);<br>&nbsp;FILE* pf;<br>&nbsp;pf = fopen("EightQuen.txt","w");<br>&nbsp;if(!pf)&nbsp; throw 0;<br>&nbsp;_dup2((fileno(pf)),_fileno(stdout));<br>&nbsp;int q[8];<br>&nbsp;EQ(q,0);<br>fclose(pf);<br>&nbsp;_dup2(old,_fileno(stdout));</p>
<p>}<br>void main()<br>{<br>outTofile();<br>}</p>
<img src ="http://www.cppblog.com/zhulf753/aggbug/32210.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/zhulf753/" target="_blank">zlf</a> 2007-09-14 14:41 <a href="http://www.cppblog.com/zhulf753/archive/2007/09/14/32210.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>