﻿<?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++博客-Kisser Leon</title><link>http://www.cppblog.com/jacky2019/</link><description>这个kisser不太冷</description><language>zh-cn</language><lastBuildDate>Sat, 04 Apr 2026 17:17:12 GMT</lastBuildDate><pubDate>Sat, 04 Apr 2026 17:17:12 GMT</pubDate><ttl>60</ttl><item><title>dxtrans.h missing in Microsoft DirectX SDK (November 2007)</title><link>http://www.cppblog.com/jacky2019/archive/2010/04/08/111974.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 08 Apr 2010 08:31:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2010/04/08/111974.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/111974.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2010/04/08/111974.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/111974.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/111974.html</trackback:ping><description><![CDATA[<p><font style="BACKGROUND-COLOR: #c0c0c0">解决办法（来自 <a href="http://social.msdn.microsoft.com/forums/en-US/windowssdk/thread/ed097d2c-3d68-4f48-8448-277eaaf68252/"><u><font color=#0000ff>http://social.msdn.microsoft.com/forums/en-US/windowssdk/thread/ed097d2c-3d68-4f48-8448-277eaaf68252/</font></u></a>）<br><br><span style="WIDOWS: 2; TEXT-TRANSFORM: none; TEXT-INDENT: 0px; BORDER-COLLAPSE: separate; FONT: medium Simsun; WHITE-SPACE: normal; ORPHANS: 2; LETTER-SPACING: normal; COLOR: rgb(0,0,0); WORD-SPACING: 0px; -webkit-border-horizontal-spacing: 0px; -webkit-border-vertical-spacing: 0px; -webkit-text-decorations-in-effect: none; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px" class=Apple-style-span>1.使用&nbsp;<span style="LINE-HEIGHT: 16px; FONT-FAMILY: 'Segoe UI', 'Lucida Grande', Verdana, Arial, Helvetica, sans-serif; COLOR: rgb(51,51,51); FONT-SIZE: 13px" class=Apple-style-span><a style="BORDER-BOTTOM-STYLE: none; PADDING-BOTTOM: 0px; BORDER-RIGHT-STYLE: none; LIST-STYLE-TYPE: none; MARGIN: 0px; PADDING-LEFT: 0px; OUTLINE-WIDTH: 0px; PADDING-RIGHT: 0px; FONT-FAMILY: inherit; BORDER-TOP-STYLE: none; COLOR: rgb(0,102,221); BORDER-LEFT-STYLE: none; CURSOR: pointer; TEXT-DECORATION: none; PADDING-TOP: 0px" title=http://www.microsoft.com/downloads/details.aspx?familyid=529F03BE-1339-48C4-BD5A-8506E5ACF571&amp;displaylang=en href="http://www.microsoft.com/downloads/details.aspx?familyid=529F03BE-1339-48C4-BD5A-8506E5ACF571&amp;displaylang=en"><font style="PADDING-BOTTOM: 0px; BORDER-RIGHT-WIDTH: 0px; MARGIN: 0px; PADDING-LEFT: 0px; OUTLINE-WIDTH: 0px; PADDING-RIGHT: 0px; FONT-FAMILY: inherit; BORDER-TOP-WIDTH: 0px; BORDER-BOTTOM-WIDTH: 0px; BORDER-LEFT-WIDTH: 0px; PADDING-TOP: 0px" face=Verdana>August 2007 DirectX SDK</font></a><font style="PADDING-BOTTOM: 0px; BORDER-RIGHT-WIDTH: 0px; MARGIN: 0px; PADDING-LEFT: 0px; OUTLINE-WIDTH: 0px; PADDING-RIGHT: 0px; FONT-FAMILY: inherit; BORDER-TOP-WIDTH: 0px; BORDER-BOTTOM-WIDTH: 0px; BORDER-LEFT-WIDTH: 0px; PADDING-TOP: 0px" face=Verdana>.<br><br>2. <span style="WIDOWS: 2; TEXT-TRANSFORM: none; TEXT-INDENT: 0px; BORDER-COLLAPSE: separate; FONT: 16px Simsun; WHITE-SPACE: normal; ORPHANS: 2; LETTER-SPACING: normal; COLOR: rgb(0,0,0); WORD-SPACING: 0px; -webkit-border-horizontal-spacing: 0px; -webkit-border-vertical-spacing: 0px; -webkit-text-decorations-in-effect: none; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px" class=Apple-style-span><span style="TEXT-ALIGN: left; LINE-HEIGHT: 16px; FONT-FAMILY: 'Segoe UI', 'Lucida Grande', Verdana, Arial, Helvetica, sans-serif; COLOR: rgb(51,51,51); FONT-SIZE: 13px" class=Apple-style-span>移除所有和dxtrans.h有关的<br><br>Remove anything to do with "dxtrans.h" and "IDXEffect" .<br><br>Say, for file "qedit.h"<br><br>//#include "dxtrans.h" -- Line 498<span class=Apple-converted-space>&nbsp;</span><br><br>// IDxtCompositor //: public IDXEffect -- Line 837<br>// IDxtAlphaSetter //: public IDXEffect -- Line 1151<br>// IDxtJpeg //: public IDXEffect -- Line 1345<br>// IDxtKey //: public IDXEffect -- Line 1735<br></span></span><br><br><br></font></span></span><br></font></p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/111974.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2010-04-08 16:31 <a href="http://www.cppblog.com/jacky2019/archive/2010/04/08/111974.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>c++ volatile关键字</title><link>http://www.cppblog.com/jacky2019/archive/2010/04/01/111249.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 01 Apr 2010 02:54:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2010/04/01/111249.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/111249.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2010/04/01/111249.html#Feedback</comments><slash:comments>7</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/111249.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/111249.html</trackback:ping><description><![CDATA[<span style="WIDOWS: 2; TEXT-TRANSFORM: none; TEXT-INDENT: 0px; BORDER-COLLAPSE: separate; FONT: medium Simsun; WHITE-SPACE: normal; ORPHANS: 2; LETTER-SPACING: normal; COLOR: rgb(0,0,0); WORD-SPACING: 0px; -webkit-border-horizontal-spacing: 0px; -webkit-border-vertical-spacing: 0px; -webkit-text-decorations-in-effect: none; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px" class=Apple-style-span>c++中的volatile关键字一直没有使用过。最近使用了一下。该关键字的意思就是告诉编译器不要对该变量进行优化，使用地址对该变量进行取值而不要直接使用该变量已经存在cach/寄存器中的值。<br><br>测试代码如下：<br>
<div style="BORDER-BOTTOM: #cccccc 1px solid; BORDER-LEFT: #cccccc 1px solid; PADDING-BOTTOM: 4px; BACKGROUND-COLOR: #eeeeee; PADDING-LEFT: 4px; WIDTH: 98%; PADDING-RIGHT: 5px; FONT-SIZE: 13px; WORD-BREAK: break-all; BORDER-TOP: #cccccc 1px solid; BORDER-RIGHT: #cccccc 1px solid; PADDING-TOP: 4px"><span style="COLOR: #008080">&nbsp;1</span><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"><span style="COLOR: #008000">//</span><span style="COLOR: #008000">&nbsp;volatile.cpp&nbsp;:&nbsp;Defines&nbsp;the&nbsp;entry&nbsp;point&nbsp;for&nbsp;the&nbsp;console&nbsp;application.<br></span><span style="COLOR: #008080">&nbsp;2</span><span style="COLOR: #008000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"></span><span style="COLOR: #008000">//<br></span><span style="COLOR: #008080">&nbsp;3</span><span style="COLOR: #008000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"></span><span style="COLOR: #000000"><br></span><span style="COLOR: #008080">&nbsp;4</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif">#include&nbsp;</span><span style="COLOR: #000000">"</span><span style="COLOR: #000000">stdafx.h</span><span style="COLOR: #000000">"</span><span style="COLOR: #000000"><br></span><span style="COLOR: #008080">&nbsp;5</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"><br></span><span style="COLOR: #008080">&nbsp;6</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"><br></span><span style="COLOR: #008080">&nbsp;7</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"></span><span style="COLOR: #0000ff">int</span><span style="COLOR: #000000">&nbsp;_tmain(</span><span style="COLOR: #0000ff">int</span><span style="COLOR: #000000">&nbsp;argc,&nbsp;_TCHAR</span><span style="COLOR: #000000">*</span><span style="COLOR: #000000">&nbsp;argv[])<br></span><span style="COLOR: #008080">&nbsp;8</span><span style="COLOR: #000000"><img id=Codehighlighter1_134_292_Open_Image onclick="this.style.display='none'; Codehighlighter1_134_292_Open_Text.style.display='none'; Codehighlighter1_134_292_Closed_Image.style.display='inline'; Codehighlighter1_134_292_Closed_Text.style.display='inline';" align=top src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockStart.gif"><img style="DISPLAY: none" id=Codehighlighter1_134_292_Closed_Image onclick="this.style.display='none'; Codehighlighter1_134_292_Closed_Text.style.display='none'; Codehighlighter1_134_292_Open_Image.style.display='inline'; Codehighlighter1_134_292_Open_Text.style.display='inline';" align=top src="http://www.cppblog.com/Images/OutliningIndicators/ContractedBlock.gif"></span><span style="BORDER-BOTTOM: #808080 1px solid; BORDER-LEFT: #808080 1px solid; BACKGROUND-COLOR: #ffffff; DISPLAY: none; BORDER-TOP: #808080 1px solid; BORDER-RIGHT: #808080 1px solid" id=Codehighlighter1_134_292_Closed_Text><img src="http://www.cppblog.com/Images/dot.gif"></span><span id=Codehighlighter1_134_292_Open_Text><span style="COLOR: #000000">{<br></span><span style="COLOR: #008080">&nbsp;9</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="COLOR: #008000">//</span><span style="COLOR: #008000">volatile&nbsp;int&nbsp;a&nbsp;=&nbsp;1;</span><span style="COLOR: #008000"><br></span><span style="COLOR: #008080">10</span><span style="COLOR: #008000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif"></span><span style="COLOR: #000000">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="COLOR: #0000ff">int</span><span style="COLOR: #000000">&nbsp;a&nbsp;</span><span style="COLOR: #000000">=</span><span style="COLOR: #000000">&nbsp;</span><span style="COLOR: #000000">1</span><span style="COLOR: #000000">;<br></span><span style="COLOR: #008080">11</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="COLOR: #000000">"</span><span style="COLOR: #000000">a&nbsp;=&nbsp;%d\n</span><span style="COLOR: #000000">"</span><span style="COLOR: #000000">,&nbsp;a);<br></span><span style="COLOR: #008080">12</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif"><br></span><span style="COLOR: #008080">13</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;__asm<br></span><span style="COLOR: #008080">14</span><span style="COLOR: #000000"><img id=Codehighlighter1_206_238_Open_Image onclick="this.style.display='none'; Codehighlighter1_206_238_Open_Text.style.display='none'; Codehighlighter1_206_238_Closed_Image.style.display='inline'; Codehighlighter1_206_238_Closed_Text.style.display='inline';" align=top src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedSubBlockStart.gif"><img style="DISPLAY: none" id=Codehighlighter1_206_238_Closed_Image onclick="this.style.display='none'; Codehighlighter1_206_238_Closed_Text.style.display='none'; Codehighlighter1_206_238_Open_Image.style.display='inline'; Codehighlighter1_206_238_Open_Text.style.display='inline';" align=top src="http://www.cppblog.com/Images/OutliningIndicators/ContractedSubBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="BORDER-BOTTOM: #808080 1px solid; BORDER-LEFT: #808080 1px solid; BACKGROUND-COLOR: #ffffff; DISPLAY: none; BORDER-TOP: #808080 1px solid; BORDER-RIGHT: #808080 1px solid" id=Codehighlighter1_206_238_Closed_Text><img src="http://www.cppblog.com/Images/dot.gif"></span><span id=Codehighlighter1_206_238_Open_Text><span style="COLOR: #000000">{<br></span><span style="COLOR: #008080">15</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;mov&nbsp;DWORD&nbsp;ptr&nbsp;[ebp</span><span style="COLOR: #000000">-</span><span style="COLOR: #000000">4</span><span style="COLOR: #000000">],&nbsp;10h // 修改a的值为0x10<br></span><span style="COLOR: #008080">16</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedSubBlockEnd.gif">&nbsp;&nbsp;&nbsp;&nbsp;}</span></span><span style="COLOR: #000000"><br></span><span style="COLOR: #008080">17</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif"><br></span><span style="COLOR: #008080">18</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="COLOR: #000000">"</span><span style="COLOR: #000000">a&nbsp;=&nbsp;%d\n</span><span style="COLOR: #000000">"</span><span style="COLOR: #000000">,&nbsp;a);<br></span><span style="COLOR: #008080">19</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif"><br></span><span style="COLOR: #008080">20</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;getchar();<br></span><span style="COLOR: #008080">21</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif"><br></span><span style="COLOR: #008080">22</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="COLOR: #0000ff">return</span><span style="COLOR: #000000">&nbsp;</span><span style="COLOR: #000000">0</span><span style="COLOR: #000000">;<br></span><span style="COLOR: #008080">23</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockEnd.gif">}</span></span><span style="COLOR: #000000"><br></span><span style="COLOR: #008080">24</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"><br></span><span style="COLOR: #008080">25</span><span style="COLOR: #000000"><img align=top src="http://www.cppblog.com/Images/OutliningIndicators/None.gif"></span></div>
</span><br>如果编译器没有优化的话，打印出来的值应该是1和16。反之，如果没有使用volatile，优化之后应该打印出来的是1和1<br><br>我在vc6和vc2008下测试了一下：<br>-----------------------------------------------------------------------------------------<br><br>vc2008<br>-----------------------------------------------------------------------------------------<br>没有关键字volatile<br>debug下打印的是 1 和 1 ---为什么会被优化了呢？我的优化编译选项是disabled的<br>release下打印的是 1 和 1<br>有关键字volatile<br><font face=宋体>debug下打印的是 1 和 1 --- 很奇怪！<br>release下打印的是 1 和 16<br><br>我想可能是编译器对debug下作了特殊处理。而且该程序非常简单</font><br><br>vc6.0<br><font face=宋体>-----------------------------------------------------------------------------------------<br>没有关键字volatile<br><font face=宋体>debug下打印的是 1 和 16<br>release下打印的是 1 和 1</font><br>有关键字volatile<br><font face=宋体>debug下打印的是 1 和 16<br>release下打印的是 1 和 1 --- 很奇怪！<br><br><br>小结：<br>在多线程多核的情况下，要防止被编译器优化！</font></font>
<img src ="http://www.cppblog.com/jacky2019/aggbug/111249.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2010-04-01 10:54 <a href="http://www.cppblog.com/jacky2019/archive/2010/04/01/111249.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>环形缓冲</title><link>http://www.cppblog.com/jacky2019/archive/2010/04/01/111243.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 01 Apr 2010 01:17:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2010/04/01/111243.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/111243.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2010/04/01/111243.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/111243.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/111243.html</trackback:ping><description><![CDATA[<pre>环形缓冲是可以不用加锁的。下面是一个实现的环形缓冲。使用c实现，可能会出现write满情况，这种情况导致的结果就是write不成功，数据没写进去。</pre><pre>我这里只是简单的丢了。</pre><pre><div style="background-color: rgb(238, 238, 238); font-size: 13px; border-left-color: rgb(204, 204, 204); padding-right: 5px; padding-bottom: 4px; padding-left: 4px; padding-top: 4px; width: 98%; word-break: break-all; "><!--<br><br>Code highlighting produced by Actipro CodeHighlighter (freeware)<br>http://www.CodeHighlighter.com/<br><br>--><span style="color: #008080; ">&nbsp;&nbsp;1</span>&nbsp;<span style="color: #008000; ">//<br></span><span style="color: #008080; ">&nbsp;&nbsp;2</span>&nbsp;<span style="color: #008000; "></span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;3</span>&nbsp;<span style="color: #000000; ">#include&nbsp;</span><span style="color: #000000; ">"</span><span style="color: #000000; ">stdafx.h</span><span style="color: #000000; ">"</span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;4</span>&nbsp;<span style="color: #000000; ">#include&nbsp;</span><span style="color: #000000; ">&lt;</span><span style="color: #000000; ">windows.h</span><span style="color: #000000; ">&gt;</span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;5</span>&nbsp;<span style="color: #000000; ">#include&nbsp;</span><span style="color: #000000; ">&lt;</span><span style="color: #000000; ">time.h</span><span style="color: #000000; ">&gt;</span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;6</span>&nbsp;<span style="color: #000000; ">#include&nbsp;</span><span style="color: #000000; ">&lt;</span><span style="color: #000000; ">stdlib.h</span><span style="color: #000000; ">&gt;</span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;7</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;8</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">#define</span><span style="color: #000000; ">&nbsp;MAX_BUF_LEN&nbsp;20&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;buf&nbsp;len&nbsp;//&nbsp;必须要大于1</span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;&nbsp;9</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;nRead&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;read&nbsp;pos</span><span style="color: #008000; "><br></span><span style="color: #008080; ">&nbsp;10</span>&nbsp;<span style="color: #008000; "></span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;nWrite&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;write&nbsp;pos</span><span style="color: #008000; "><br></span><span style="color: #008080; ">&nbsp;11</span>&nbsp;<span style="color: #008000; "></span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;buf[MAX_BUF_LEN]&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;{</span><span style="color: #000000; ">0</span><span style="color: #000000; ">};&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;loop&nbsp;buf</span><span style="color: #008000; "><br></span><span style="color: #008080; ">&nbsp;12</span>&nbsp;<span style="color: #008000; "></span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;13</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">#define</span><span style="color: #000000; ">&nbsp;NN&nbsp;20000</span><span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;14</span>&nbsp;<span style="color: #000000; ">__int64&nbsp;sum&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;15</span>&nbsp;<span style="color: #000000; ">__int64&nbsp;lose&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;16</span>&nbsp;<span style="color: #000000; ">__int64&nbsp;total&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;(__int64)(</span><span style="color: #000000; ">1</span><span style="color: #000000; ">+</span><span style="color: #000000; ">NN)</span><span style="color: #000000; ">*</span><span style="color: #000000; ">(__int64)NN</span><span style="color: #000000; ">/</span><span style="color: #000000; ">2</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;17</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">bool</span><span style="color: #000000; ">&nbsp;bWriteFinished&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">false</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;18</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;19</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">void</span><span style="color: #000000; ">&nbsp;WriteBuf(</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;n)<br></span><span style="color: #008080; ">&nbsp;20</span>&nbsp;<span style="color: #000000; ">{<br></span><span style="color: #008080; ">&nbsp;21</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;rpos&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;nRead;<br></span><span style="color: #008080; ">&nbsp;22</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;wpos&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;nWrite;<br></span><span style="color: #008080; ">&nbsp;23</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;wpos</span><span style="color: #000000; ">++</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;24</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;buf满（即write又追上了read）,&nbsp;注为了避免与初始状态重复判断<br></span><span style="color: #008080; ">&nbsp;25</span>&nbsp;<span style="color: #008000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;所以这里最后一个buf没有写就认为已经满了。所以MAX_BUF_LEN不能为1<br></span><span style="color: #008080; ">&nbsp;26</span>&nbsp;<span style="color: #008000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;这里，直接丢弃该buf</span><span style="color: #008000; "><br></span><span style="color: #008080; ">&nbsp;27</span>&nbsp;<span style="color: #008000; "></span><span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(wpos&nbsp;</span><span style="color: #000000; ">%</span><span style="color: #000000; ">&nbsp;MAX_BUF_LEN&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;rpos&nbsp;</span><span style="color: #000000; ">%</span><span style="color: #000000; ">&nbsp;MAX_BUF_LEN)<br></span><span style="color: #008080; ">&nbsp;28</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;29</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;lose&nbsp;</span><span style="color: #000000; ">+=</span><span style="color: #000000; ">&nbsp;n;<br></span><span style="color: #008080; ">&nbsp;30</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">return</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;31</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;32</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;33</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;buf[nWrite]&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;n;<br></span><span style="color: #008080; ">&nbsp;34</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;nWrite</span><span style="color: #000000; ">++</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;35</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(nWrite&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;MAX_BUF_LEN)<br></span><span style="color: #008080; ">&nbsp;36</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;nWrite&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;37</span>&nbsp;<span style="color: #000000; ">}<br></span><span style="color: #008080; ">&nbsp;38</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;39</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;ReadBuf()<br></span><span style="color: #008080; ">&nbsp;40</span>&nbsp;<span style="color: #000000; ">{<br></span><span style="color: #008080; ">&nbsp;41</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(nWrite&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;nRead)<br></span><span style="color: #008080; ">&nbsp;42</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">return</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">-</span><span style="color: #000000; ">1</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;43</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;44</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;n&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;buf[nRead];<br></span><span style="color: #008080; ">&nbsp;45</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;nRead</span><span style="color: #000000; ">++</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;46</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(nRead&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;MAX_BUF_LEN)<br></span><span style="color: #008080; ">&nbsp;47</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;nRead&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;48</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">return</span><span style="color: #000000; ">&nbsp;n;<br></span><span style="color: #008080; ">&nbsp;49</span>&nbsp;<span style="color: #000000; ">}<br></span><span style="color: #008080; ">&nbsp;50</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;51</span>&nbsp;<span style="color: #000000; ">DWORD&nbsp;WINAPI&nbsp;ReadThread(LPVOID&nbsp;lpParameter)<br></span><span style="color: #008080; ">&nbsp;52</span>&nbsp;<span style="color: #000000; ">{<br></span><span style="color: #008080; ">&nbsp;53</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;tmp&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;54</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;count&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;55</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">while</span><span style="color: #000000; ">&nbsp;(</span><span style="color: #0000FF; ">true</span><span style="color: #000000; ">)<br></span><span style="color: #008080; ">&nbsp;56</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;57</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000; ">//</span><span style="color: #008000; ">printf("ReadThread&nbsp;=&nbsp;%d\n",&nbsp;ReadBuf());</span><span style="color: #008000; "><br></span><span style="color: #008080; ">&nbsp;58</span>&nbsp;<span style="color: #008000; "></span><span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;tmp&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;ReadBuf();<br></span><span style="color: #008080; ">&nbsp;59</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(</span><span style="color: #000000; ">-</span><span style="color: #000000; ">1</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">!=</span><span style="color: #000000; ">&nbsp;tmp)<br></span><span style="color: #008080; ">&nbsp;60</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;61</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;count</span><span style="color: #000000; ">++</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;62</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;sum&nbsp;</span><span style="color: #000000; ">+=</span><span style="color: #000000; ">&nbsp;tmp;<br></span><span style="color: #008080; ">&nbsp;63</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(count&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;NN)<br></span><span style="color: #008080; ">&nbsp;64</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;65</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">ReadThread&nbsp;finished!\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">&nbsp;66</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">break</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;67</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;68</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;69</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">else</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(bWriteFinished)<br></span><span style="color: #008080; ">&nbsp;70</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;71</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">ReadThread&nbsp;finished222!\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">&nbsp;72</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">break</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;73</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;74</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;75</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Sleep(rand()&nbsp;</span><span style="color: #000000; ">%</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">10</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">&nbsp;76</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;77</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">return</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;78</span>&nbsp;<span style="color: #000000; ">}<br></span><span style="color: #008080; ">&nbsp;79</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;80</span>&nbsp;<span style="color: #000000; ">DWORD&nbsp;WINAPI&nbsp;WriteThread(LPVOID&nbsp;lpParameter)<br></span><span style="color: #008080; ">&nbsp;81</span>&nbsp;<span style="color: #000000; ">{<br></span><span style="color: #008080; ">&nbsp;82</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;n&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;83</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">while</span><span style="color: #000000; ">&nbsp;(</span><span style="color: #0000FF; ">true</span><span style="color: #000000; ">)<br></span><span style="color: #008080; ">&nbsp;84</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;85</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;n</span><span style="color: #000000; ">++</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;86</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;WriteBuf(n);<br></span><span style="color: #008080; ">&nbsp;87</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(n&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;NN)<br></span><span style="color: #008080; ">&nbsp;88</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">&nbsp;89</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">WriteThread&nbsp;finished!\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">&nbsp;90</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;bWriteFinished&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #0000FF; ">true</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;91</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">break</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;92</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;93</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;94</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Sleep(rand()&nbsp;</span><span style="color: #000000; ">%</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">10</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">&nbsp;95</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">&nbsp;96</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">return</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">&nbsp;97</span>&nbsp;<span style="color: #000000; ">}<br></span><span style="color: #008080; ">&nbsp;98</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">&nbsp;99</span>&nbsp;<span style="color: #000000; "></span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;main(</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;argc,&nbsp;</span><span style="color: #0000FF; ">char</span><span style="color: #000000; ">*</span><span style="color: #000000; ">&nbsp;argv[])<br></span><span style="color: #008080; ">100</span>&nbsp;<span style="color: #000000; ">{<br></span><span style="color: #008080; ">101</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;srand(time(NULL));<br></span><span style="color: #008080; ">102</span>&nbsp;<span style="color: #000000; "></span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;(int&nbsp;i&nbsp;=&nbsp;0;&nbsp;i&nbsp;&lt;&nbsp;10;&nbsp;i++)<br></span><span style="color: #008080; ">103</span>&nbsp;<span style="color: #008000; "></span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">104</span>&nbsp;<span style="color: #008000; "></span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;("rand&nbsp;=&nbsp;%d\n",&nbsp;rand()&nbsp;%&nbsp;10);<br></span><span style="color: #008080; ">105</span>&nbsp;<span style="color: #008000; "></span><span style="color: #008000; ">//</span><span style="color: #008000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}</span><span style="color: #008000; "><br></span><span style="color: #008080; ">106</span>&nbsp;<span style="color: #008000; "></span><span style="color: #000000; "><br></span><span style="color: #008080; ">107</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;DWORD&nbsp;threadid[</span><span style="color: #000000; ">2</span><span style="color: #000000; ">]&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;{</span><span style="color: #000000; ">0</span><span style="color: #000000; ">};<br></span><span style="color: #008080; ">108</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;HANDLE&nbsp;hThread[</span><span style="color: #000000; ">2</span><span style="color: #000000; ">]&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;{</span><span style="color: #000000; ">0</span><span style="color: #000000; ">};<br></span><span style="color: #008080; ">109</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">int</span><span style="color: #000000; ">&nbsp;param&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">110</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;hThread[</span><span style="color: #000000; ">0</span><span style="color: #000000; ">]&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;CreateThread(NULL,&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">,&nbsp;ReadThread,&nbsp;</span><span style="color: #000000; ">&amp;</span><span style="color: #000000; ">param,&nbsp;CREATE_SUSPENDED,&nbsp;</span><span style="color: #000000; ">&amp;</span><span style="color: #000000; ">threadid[</span><span style="color: #000000; ">0</span><span style="color: #000000; ">]);<br></span><span style="color: #008080; ">111</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;ResumeThread(hThread[</span><span style="color: #000000; ">0</span><span style="color: #000000; ">]);<br></span><span style="color: #008080; ">112</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;hThread[</span><span style="color: #000000; ">1</span><span style="color: #000000; ">]&nbsp;</span><span style="color: #000000; ">=</span><span style="color: #000000; ">&nbsp;CreateThread(NULL,&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">,&nbsp;WriteThread,&nbsp;</span><span style="color: #000000; ">&amp;</span><span style="color: #000000; ">param,&nbsp;CREATE_SUSPENDED,&nbsp;</span><span style="color: #000000; ">&amp;</span><span style="color: #000000; ">threadid[</span><span style="color: #000000; ">1</span><span style="color: #000000; ">]);<br></span><span style="color: #008080; ">113</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;ResumeThread(hThread[</span><span style="color: #000000; ">1</span><span style="color: #000000; ">]);<br></span><span style="color: #008080; ">114</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">115</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;getchar();<br></span><span style="color: #008080; ">116</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">117</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">total&nbsp;=&nbsp;%lld\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">,&nbsp;total);<br></span><span style="color: #008080; ">118</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">lose&nbsp;=&nbsp;%lld\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">,&nbsp;lose);<br></span><span style="color: #008080; ">119</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">sum&nbsp;=&nbsp;%lld\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">,&nbsp;sum);<br></span><span style="color: #008080; ">120</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">121</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">if</span><span style="color: #000000; ">&nbsp;(total&nbsp;</span><span style="color: #000000; ">==</span><span style="color: #000000; ">&nbsp;lose&nbsp;</span><span style="color: #000000; ">+</span><span style="color: #000000; ">&nbsp;sum)<br></span><span style="color: #008080; ">122</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">123</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">loop&nbsp;buf&nbsp;works&nbsp;well!\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">124</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">125</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">else</span><span style="color: #000000; "><br></span><span style="color: #008080; ">126</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;{<br></span><span style="color: #008080; ">127</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;(</span><span style="color: #000000; ">"</span><span style="color: #000000; ">loop&nbsp;buf&nbsp;works&nbsp;wrong!\n</span><span style="color: #000000; ">"</span><span style="color: #000000; ">);<br></span><span style="color: #008080; ">128</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;}<br></span><span style="color: #008080; ">129</span>&nbsp;<span style="color: #000000; "><br></span><span style="color: #008080; ">130</span>&nbsp;<span style="color: #000000; ">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000FF; ">return</span><span style="color: #000000; ">&nbsp;</span><span style="color: #000000; ">0</span><span style="color: #000000; ">;<br></span><span style="color: #008080; ">131</span>&nbsp;<span style="color: #000000; ">}<br></span><span style="color: #008080; ">132</span>&nbsp;<span style="color: #000000; "></span></div>上面是加了测试代码的，我测试了一下似乎没问题。</pre><pre>注意：这里只允许一个线程读，一个线程写。如果是多个线程读写的话，需要锁！</pre><pre>另外，很多细节都没考虑进去</pre><img src ="http://www.cppblog.com/jacky2019/aggbug/111243.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2010-04-01 09:17 <a href="http://www.cppblog.com/jacky2019/archive/2010/04/01/111243.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>人生最高境界</title><link>http://www.cppblog.com/jacky2019/archive/2008/12/19/69790.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Fri, 19 Dec 2008 00:07:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2008/12/19/69790.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/69790.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2008/12/19/69790.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/69790.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/69790.html</trackback:ping><description><![CDATA[人生最高境界：<br>　　　　拿沙特工资，住英国房子，用瑞典手机，戴瑞士手表，娶韩国女人，包日本二奶，做泰国按摩，开德国轿车，坐美国飞机，喝法国红酒，吃澳洲海鲜，抽古巴雪茄，穿意大利皮鞋，玩西班牙女郎，看奥地利歌剧，买俄罗斯别墅，雇菲律宾女佣，配以色列保镖，洗土耳其桑拿，当中国干部.<br>
<img src ="http://www.cppblog.com/jacky2019/aggbug/69790.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2008-12-19 08:07 <a href="http://www.cppblog.com/jacky2019/archive/2008/12/19/69790.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>存储过程 stored procedure</title><link>http://www.cppblog.com/jacky2019/archive/2007/11/06/35937.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Tue, 06 Nov 2007 02:37:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/11/06/35937.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/35937.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/11/06/35937.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/35937.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/35937.html</trackback:ping><description><![CDATA[<span style="COLOR: red">&nbsp;
<p><font color=#000000><font size=3><strong>定义：<br></strong>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 将常用的或很复杂的工作，预先用SQL语句写好并用一个指定的名称存储起来,&nbsp;&nbsp; 那么以后要叫<a onclick="javascript:tagshow(event, '%CA%FD%BE%DD%BF%E2');" href="javascript:;" target=_self><u><strong><font color=#800080>数据库</font></strong></u></a>提供与已定义好的存储过程的功能相同的服务时,只需调用execute,即可自动完成命令。<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 讲到这里,可能有人要问：这么说存储过程就是一堆SQL语句而已啊？<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Microsoft公司为什么还要添加这个技术呢?<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 那么存储过程与一般的SQL语句有什么区别呢?</font></font></p>
<p><font color=#000000 size=3>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </font><font color=#000000><font size=3><strong>存储过程的优点：<br></strong>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 1.存储过程只在创造时进行编译，以后每次执行存储过程都不需再重新编译，而一般SQL语句每执行一次就编译一次,所以使用存储过程可提高数据库执行速度。<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 2.当对数据库进行复杂操作时(如对多个表进行Update,Insert,Query,Delete时），可将此复杂操作用存储过程封装起来与数据库提供的事务处理结合一起使用。<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 3.存储过程可以重复使用,可减少数据库开发人员的工作量<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 4.安全性高,可设定只有某此用户才具有对指定存储过程的使用权<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <strong>存储过程的种类：</strong><br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 1.系统存储过程：以sp_开头,用来进行系统的各项设定.取得信息.相关管理工作,<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 如&nbsp;&nbsp; sp_help就是取得指定对象的相关信息<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 2.扩展存储过程&nbsp;&nbsp; 以XP_开头,用来调用操作系统提供的功能<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; exec&nbsp;&nbsp; master..xp_cmdshell&nbsp;&nbsp; 'ping&nbsp;&nbsp; 10.8.16.1'<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 3.用户自定义的存储过程,这是我们所指的存储过程<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 常用格式<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Create&nbsp;&nbsp; procedure&nbsp;&nbsp; procedue_name<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; [@parameter&nbsp;&nbsp; data_type][output]<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; [with]{recompile|encryption}<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; as<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; sql_statement<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 解释:&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; output：表示此参数是可传回的<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; with&nbsp;&nbsp; {recompile|encryption}<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; recompile:表示每次执行此存储过程时都重新编译一次<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; encryption:所创建的存储过程的内容会被加密<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 如:<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 表book的内容如下<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 编号&nbsp;&nbsp; 书名&nbsp;&nbsp; 价格<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 001&nbsp;&nbsp; C语言入门&nbsp;&nbsp; $30<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 002&nbsp;&nbsp; PowerBuilder报表开发&nbsp;&nbsp; $52<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 实例1:查询表Book的内容的存储过程<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; create&nbsp;&nbsp; proc&nbsp;&nbsp; query_book<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; as&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; select&nbsp;&nbsp; *&nbsp;&nbsp; from&nbsp;&nbsp; book<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; go<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; exec&nbsp;&nbsp; query_book<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 实例2:加入一笔记录到表book,并查询此表中所有书籍的总金额<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Create&nbsp;&nbsp; proc&nbsp;&nbsp; insert_book<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; @param1&nbsp;&nbsp; char(10),@param2&nbsp;&nbsp; varchar(20),@param3&nbsp;&nbsp; money,@param4&nbsp;&nbsp; money&nbsp;&nbsp; output<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; with&nbsp;&nbsp; encryption&nbsp;&nbsp; ---------加密<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; as<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; insert&nbsp;&nbsp; book(编号,书名，价格）&nbsp;&nbsp; Values(@param1,@param2,@param3)<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; select&nbsp;&nbsp; @param4=sum(价格)&nbsp;&nbsp; from&nbsp;&nbsp; book<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; go<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 执行例子:&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; declare&nbsp;&nbsp; @total_price&nbsp;&nbsp; money&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; exec&nbsp;&nbsp; insert_book&nbsp;&nbsp; '003','Delphi&nbsp;&nbsp; 控件开发指南',$100,@total_price<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; print&nbsp;&nbsp; '总金额为'+convert(varchar,@total_price)<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; go<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </font></font><font color=#000000><font size=3><strong>存储过程的3种传回值:<br></strong>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 1.以Return传回整数<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 2.以output格式传回参数<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 3.Recordset<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 传回值的区别:<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; output和return都可在批次程式中用变量接收,而recordset则传回到执行批次的客户端中&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 实例3：设有两个表为Product,Order,其表内容如下：<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Product<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 产品编号&nbsp;&nbsp; 产品名称&nbsp;&nbsp; 客户订数&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 001&nbsp;&nbsp; 钢笔&nbsp;&nbsp; 30&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 002&nbsp;&nbsp; 毛笔&nbsp;&nbsp; 50&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 003&nbsp;&nbsp; 铅笔&nbsp;&nbsp; 100&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Order&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 产品编号&nbsp;&nbsp; 客户名&nbsp;&nbsp; 客户订金<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 001&nbsp;&nbsp; 南山区&nbsp;&nbsp; $30<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 002&nbsp;&nbsp; 罗湖区&nbsp;&nbsp; $50<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 003&nbsp;&nbsp; 宝安区&nbsp;&nbsp; $4<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 请实现按编号为连接条件,将两个表连接成一个临时表,该表只含编号.产品名.客户名.订金.总金额,<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 总金额=订金*订数,临时表放在存储过程中<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 代码如下:<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Create&nbsp;&nbsp; proc&nbsp;&nbsp; temp_sale<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; as<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; select&nbsp;&nbsp; a.产品编号,a.产品名称,b.客户名,b.客户订金,a.客户订数*&nbsp;&nbsp; b.客户订金&nbsp;&nbsp; as总金额<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; into&nbsp;&nbsp; #temptable&nbsp;&nbsp; from&nbsp;&nbsp; Product&nbsp;&nbsp; a&nbsp;&nbsp; inner&nbsp;&nbsp; join&nbsp;&nbsp; Order&nbsp;&nbsp; b&nbsp;&nbsp; on&nbsp;&nbsp; a.产品编号=b.产品编号<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if&nbsp;&nbsp; @@error=0&nbsp;&nbsp; <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; print&nbsp;&nbsp; 'Good'<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; print&nbsp;&nbsp; 'Fail'<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; go </font></font></p>
<p><font color=#000000 size=3>--------------------------------------------------------------------------------------------------------------------</font></p>
<p dir=ltr><font color=#000000><font size=3><strong>存储过程介绍</strong>&nbsp;&nbsp; </font></font></p>
<blockquote dir=ltr style="MARGIN-RIGHT: 0px">
<p><font color=#000000><font size=3><strong>一、先介绍一下什么是存储过程&nbsp;&nbsp; <br></strong>存储过程是利用SQL&nbsp;&nbsp; Server所提供的Tranact-SQL语言所编写的程序。Tranact-SQL语言是SQL&nbsp;&nbsp; Server提供专为设计数据库应用程序的语言，它是应用程序和SQL&nbsp;&nbsp; Server数据库间的主要程序式设计界面。它好比Oracle数据库系统中的Pro-SQL和Informix的数据库系统能够中的Informix-4GL语言一样。这类语言主要提供以下功能，让用户可以设计出符合引用需求的程序：&nbsp;&nbsp; <br>1)、变量说明&nbsp;&nbsp; <br>2)、ANSI兼容的SQL命令(如Select,Update&#8230;.)&nbsp;&nbsp; <br>3)、一般流程控制命令(if&#8230;else&#8230;、while&#8230;.)&nbsp;&nbsp; <br>4)、内部函数&nbsp;&nbsp; </font></font></p>
<p><strong><font color=#000000 size=3>二、存储过程的书写格式&nbsp;&nbsp; </font></strong></p>
<p><font color=#000000 size=3>CREATE&nbsp;&nbsp; PROCEDURE&nbsp;&nbsp; [拥有者.]存储过程名[;程序编号]&nbsp;&nbsp; <br>[(参数#1,&#8230;参数#1024)]&nbsp;&nbsp; <br>[WITH&nbsp;&nbsp; <br>{RECOMPILE&nbsp;&nbsp; |&nbsp;&nbsp; ENCRYPTION&nbsp;&nbsp; |&nbsp;&nbsp; RECOMPILE,&nbsp;&nbsp; ENCRYPTION}&nbsp;&nbsp; <br>]&nbsp;&nbsp; <br>[FOR&nbsp;&nbsp; REPLICATION]&nbsp;&nbsp; <br>AS&nbsp;&nbsp; 程序行&nbsp;&nbsp; </font></p>
<p><font color=#000000 size=3>其中存储过程名不能超过128个字。每个存储过程中最多设定1024个参数&nbsp;&nbsp; <br>(SQL&nbsp;&nbsp; Server&nbsp;&nbsp; 7.0以上版本),参数的使用方法如下:&nbsp;&nbsp; </font></p>
<p><font color=#000000 size=3>@参数名&nbsp;&nbsp; 数据类型&nbsp;&nbsp; [VARYING]&nbsp;&nbsp; [=内定值]&nbsp;&nbsp; [OUTPUT]&nbsp;&nbsp; </font></p>
<p><font color=#000000 size=3>每个参数名前要有一个&#8220;@&#8221;符号,每一个存储过程的参数仅为该程序内部使用,参数的类型除了IMAGE外，其他SQL&nbsp;&nbsp; Server所支持的数据类型都可使用。&nbsp;&nbsp; <br>[=内定值]相当于我们在建立数据库时设定一个字段的默认值，这里是为这个参数设定默认值。[OUTPUT]是用来指定该参数是既有输入又有输出值的，也就是在调用了这个存储过程时，如果所指定的参数值是我们需要输入的参数，同时也需要在结果中输出的，则该项必须为OUTPUT，而如果只是做输出参数用，可以用CURSOR，同时在使用该参数时，必须指定VARYING和OUTPUT这两个语句。&nbsp;&nbsp; </font></p>
<p><font color=#000000 size=3>例子:&nbsp;&nbsp; <br>CREATE&nbsp;&nbsp; PROCEDURE&nbsp;&nbsp; order_tot_amt&nbsp;&nbsp; @o_id&nbsp;&nbsp; int,@p_tot&nbsp;&nbsp; int&nbsp;&nbsp; output&nbsp;&nbsp; AS&nbsp;&nbsp; <br>SELECT&nbsp;&nbsp; @p_tot&nbsp;&nbsp; =&nbsp;&nbsp; sum(Unitprice*Quantity)&nbsp;&nbsp; <br>FROM&nbsp;&nbsp; orderdetails&nbsp;&nbsp; <br>WHERE&nbsp;&nbsp; ōrdered=@o_id&nbsp;&nbsp; </font></p>
<p><font color=#000000 size=3>例子说明:&nbsp;&nbsp; <br>该例子是建立一个简单的存储过程order_tot_amt,这个存储过程根据用户输入的定单ID号码(@o_id),由定单明细表(orderdetails)中计算该定单销售总额[单价(Unitprice)*数量(Quantity)],这一金额通过@p_tot这一参数输出给调用这一存储过程的程序&nbsp;&nbsp; </font></p>
<p><font color=#000000><font size=3><strong>三、在SQL&nbsp;&nbsp; Server中执行存储过程</strong>&nbsp;&nbsp; </font></font></p>
<p><font color=#000000 size=3>在SQL&nbsp;&nbsp; Server的查询分析器中，输入以下代码:&nbsp;&nbsp; <br>declare&nbsp;&nbsp; @tot_amt&nbsp;&nbsp; int&nbsp;&nbsp; <br>execute&nbsp;&nbsp; order_tot_amt&nbsp;&nbsp; 1,@tot_amt&nbsp;&nbsp; output&nbsp;&nbsp; <br>select&nbsp;&nbsp; @tot_amt&nbsp;&nbsp; </font></p>
<p><font color=#000000 size=3>以上代码是执行order_tot_amt这一存储过程，以计算出定单编号为1的定单销售金额，我们定义@tot_amt为输出参数，用来承接我们所要的结果 </font></p>
</blockquote></span>
<img src ="http://www.cppblog.com/jacky2019/aggbug/35937.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-11-06 10:37 <a href="http://www.cppblog.com/jacky2019/archive/2007/11/06/35937.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>汉字编码问题</title><link>http://www.cppblog.com/jacky2019/archive/2007/11/01/35693.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 01 Nov 2007 06:53:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/11/01/35693.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/35693.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/11/01/35693.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/35693.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/35693.html</trackback:ping><description><![CDATA[<font face=宋体><a href="http://www.css8.cn/css8_document/gb2312.htm">http://www.css8.cn/css8_document/gb2312.htm</a><br><br>由于常常要和汉字处理打交道，因此，我常常受到汉字编码问题的困扰。在不断的打击与坚持中，也积累了一点汉字编码方面的经验，想和大家一起分享。</font>
<p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">一、汉字编码的种类</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="COLOR: black; FONT-FAMILY: 宋体">汉字编码中现在主要用到的有三类，包括<span lang=EN-US>GBK</span>，<span lang=EN-US>GB2312</span>和<span lang=EN-US>Big5</span>。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; 1</span><span style="COLOR: black; FONT-FAMILY: 宋体">、<span lang=EN-US>GB2312</span>又称国标码，</span><span style="FONT-FAMILY: 宋体">由国家标准总局发布，</span><span lang=EN-US>1981</span><span style="FONT-FAMILY: 宋体">年</span><span lang=EN-US>5</span><span style="FONT-FAMILY: 宋体">月</span><span lang=EN-US>1</span><span style="FONT-FAMILY: 宋体">日实施，通行于大陆。新加坡等地也使用此编码。它是一个简化字的编码规范，当然也包括其他的符号、字母、日文假名等，共</span><span lang=EN-US>7445</span><span style="FONT-FAMILY: 宋体">个图形字符，其中汉字占</span><span lang=EN-US>6763</span><span style="FONT-FAMILY: 宋体">个。我们平时说</span><span lang=EN-US>6768</span><span style="FONT-FAMILY: 宋体">个汉字，实际上里边有</span><span lang=EN-US>5</span><span style="FONT-FAMILY: 宋体">个编码为空白，所以总共有</span><span lang=EN-US>6763</span><span style="FONT-FAMILY: 宋体">个汉字。</span></p>
<p class=MsoNormal><span lang=EN-US>&nbsp; &nbsp;&nbsp;&nbsp; GB2312</span><span style="FONT-FAMILY: 宋体">规定&#8220;对任意一个图形字符都采用两个字节表示，每个字节均采用七位编码表示&#8221;，习惯上称第一个字节为&#8220;高字节&#8221;，第二个字节为&#8220;低字节&#8221;。</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">GB2312</span><span style="COLOR: black; FONT-FAMILY: 宋体">中汉字的编码范围为，第一字节<span lang=EN-US>0xB0-0xF7(</span>对应十进制为<span lang=EN-US>176-247)</span>，第二个字节<span lang=EN-US>0xA0-0xFE</span>（对应十进制为<span lang=EN-US>160-254</span>）。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span lang=EN-US>GB2312</span><span style="FONT-FAMILY: 宋体">将代码表分为</span><span lang=EN-US>94</span><span style="FONT-FAMILY: 宋体">个区，对应第一字节（</span><span lang=EN-US style="FONT-FAMILY: 宋体">0xa1-0xfe</span><span style="FONT-FAMILY: 宋体">）；每个区</span><span lang=EN-US>94</span><span style="FONT-FAMILY: 宋体">个位（</span><span lang=EN-US style="FONT-FAMILY: 宋体">0xa1-0xfe</span><span style="FONT-FAMILY: 宋体">），对应第二字节，两个字节的值分别为区号值和位号值加</span><span lang=EN-US>32</span><span style="FONT-FAMILY: 宋体">（</span><span lang=EN-US>2OH</span><span style="FONT-FAMILY: 宋体">），因此也称为区位码。</span><span lang=EN-US>01-09</span><span style="FONT-FAMILY: 宋体">区为符号、数字区，</span><span lang=EN-US>16-87</span><span style="FONT-FAMILY: 宋体">区为汉字区（</span><span lang=EN-US style="FONT-FAMILY: 宋体">0xb0-0xf7</span><span style="FONT-FAMILY: 宋体">），</span><span lang=EN-US>10-15</span><span style="FONT-FAMILY: 宋体">区、</span><span lang=EN-US>88-94</span><span style="FONT-FAMILY: 宋体">区是有待进一步标准化的空白区。</span></p>
<p class=MsoNormal><span lang=EN-US></span>&nbsp;</p>
<p class=MsoNormal><span lang=EN-US>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 2</span><span style="FONT-FAMILY: 宋体">、</span><span lang=EN-US>Big5</span><span style="FONT-FAMILY: 宋体">又称大五码，主要为香港与台湾使用，即是一个繁体字编码。</span><span style="COLOR: black; FONT-FAMILY: 宋体">每个汉字由两个字节构成，第一个字节的范围从</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">0X81</span><span style="COLOR: black; FONT-FAMILY: 宋体">－</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">0XFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（即</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">129-255</span><span style="COLOR: black; FONT-FAMILY: 宋体">），共</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">126</span><span style="COLOR: black; FONT-FAMILY: 宋体">种。第二个字节的范围不连续，分别为</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">0X40</span><span style="COLOR: black; FONT-FAMILY: 宋体">－</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">0X7E</span><span style="COLOR: black; FONT-FAMILY: 宋体">（即</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">64-126</span><span style="COLOR: black; FONT-FAMILY: 宋体">），</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">0XA1</span><span style="COLOR: black; FONT-FAMILY: 宋体">－</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">0XFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（即</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">161-254</span><span style="COLOR: black; FONT-FAMILY: 宋体">），共</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: Arial">157</span><span style="COLOR: black; FONT-FAMILY: 宋体">种。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体"></span>&nbsp;</p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; 3</span><span style="COLOR: black; FONT-FAMILY: 宋体">、<span lang=EN-US>GBK</span>是<span lang=EN-US>GB2312</span>的扩展，是向上兼容的，因此<span lang=EN-US>GB2312</span>中的汉字的编码与<span lang=EN-US>GBK</span>中汉字的相同。另外，<span lang=EN-US>GBK</span>中还包含繁体字的编码，它与<span lang=EN-US>Big5</span>编码之间的关系我还没有弄明白，好像是不一致的。<span lang=EN-US>GBK</span>中每个汉字仍然包含两个字节，第一个字节的范围是<span lang=EN-US>0x81-0xFE</span>（即<span lang=EN-US>129-254</span>），第二个字节的范围是<span lang=EN-US>0x40-0xFE</span>（即<span lang=EN-US>64-254</span>）。<span lang=EN-US>GBK</span>中有码位<span lang=EN-US>23940</span>个，包含汉字<span lang=EN-US>21003</span>个。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></p>
<p class=MsoNormal><span lang=en-us style="FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span style="COLOR: black; FONT-FAMILY: 宋体">表<span lang=EN-US>1 </span>汉字编码范围</span></p>
<table class=MsoTableGrid id=table1 style="BORDER-RIGHT: medium none; BORDER-TOP: medium none; MARGIN-LEFT: 50.4pt; BORDER-LEFT: medium none; BORDER-BOTTOM: medium none; BORDER-COLLAPSE: collapse" cellSpacing=0 cellPadding=0 border=1>
    <tbody>
        <tr>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: windowtext 1pt solid; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: windowtext 1pt solid; WIDTH: 81pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=108>
            <p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">名称</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: windowtext 1pt solid; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 144pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=192>
            <p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">第一字节</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: windowtext 1pt solid; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 153pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=204>
            <p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">第二字节</span></p>
            </td>
        </tr>
        <tr>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: windowtext 1pt solid; WIDTH: 81pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=108>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">GB2312</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 144pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=192>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0xB0-0xF7(176-247) </span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 153pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=204>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0xA0-0xFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（</span><span lang=EN-US style="COLOR: black">160-254</span><span style="COLOR: black; FONT-FAMILY: 宋体">）</span></p>
            </td>
        </tr>
        <tr>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: windowtext 1pt solid; WIDTH: 81pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=108>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">GBK</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 144pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=192>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0x81-0xFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（</span><span lang=EN-US style="COLOR: black">129-254</span><span style="COLOR: black; FONT-FAMILY: 宋体">）</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 153pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=204>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0x40-0xFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（</span><span lang=EN-US style="COLOR: black">64-254</span><span style="COLOR: black; FONT-FAMILY: 宋体">）</span></p>
            </td>
        </tr>
        <tr>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: windowtext 1pt solid; WIDTH: 81pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=108>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">Big5</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 144pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=192>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0x81-0xFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（</span><span lang=EN-US style="COLOR: black">129-255</span><span style="COLOR: black; FONT-FAMILY: 宋体">）</span></p>
            </td>
            <td style="BORDER-RIGHT: windowtext 1pt solid; PADDING-RIGHT: 5.4pt; BORDER-TOP: medium none; PADDING-LEFT: 5.4pt; PADDING-BOTTOM: 0cm; BORDER-LEFT: medium none; WIDTH: 153pt; PADDING-TOP: 0cm; BORDER-BOTTOM: windowtext 1pt solid" vAlign=top width=204>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0x40-0x7E</span><span style="COLOR: black; FONT-FAMILY: 宋体">（</span><span lang=EN-US style="COLOR: black">64-126</span><span style="COLOR: black; FONT-FAMILY: 宋体">）</span></p>
            <p class=MsoNormal><span lang=EN-US style="COLOR: black">0xA1</span><span style="COLOR: black; FONT-FAMILY: 宋体">－</span><span lang=EN-US style="COLOR: black">0xFE</span><span style="COLOR: black; FONT-FAMILY: 宋体">（</span><span lang=EN-US style="COLOR: black">161-254</span><span style="COLOR: black; FONT-FAMILY: 宋体">）</span></p>
            </td>
        </tr>
    </tbody>
</table>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体"></span>&nbsp;</p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体"></span>&nbsp;</p>
<p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">二、对汉字进行<span lang=EN-US>hash</span></span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="COLOR: black; FONT-FAMILY: 宋体">为了处理汉字的方便，在查找汉字的时候，我们通常会用到<span lang=EN-US>hash</span>的方法，那怎么来确定一个汉字位置呢？这就和每种编码的排列有关了，这里主要给出一种<span lang=EN-US>hash</span>函数的策略。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="COLOR: black; FONT-FAMILY: 宋体">对于<span lang=EN-US>GB2312</span>编码，设输入的汉字为<span lang=EN-US>GBword</span>，我们可以采用公式<span lang=EN-US>(C1-176)*94 + (C2-161)</span>确定<span lang=EN-US>GBindex</span>。其中，<span lang=EN-US>C1</span>表示第一字节，<span lang=EN-US>C2</span>表示第二字节。具体如下：</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="COLOR: #3700c8; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; GBindex = ((unsigned char)GBword.at(0)-176)*94 + (unsigned char)GBword.at(1) - 161;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="FONT-FAMILY: 宋体">之所以用<span lang=EN-US>unsigned char</span>类型，是因为<span lang=EN-US>char</span>是一个字节，如果用<span lang=EN-US>unsigend int</span>，因为<span lang=EN-US>int</span>是<span lang=EN-US>4</span>个字节的，所以会造成扩展，导致错误。</span></p>
<p class=MsoNormal style="MARGIN-LEFT: 18.05pt; TEXT-INDENT: -35.9pt; TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span style="FONT-FAMILY: 宋体">对于<span lang=EN-US>GBK</span>编码，设输入的汉字为<span lang=EN-US>GBKword</span>，则可以采用公式<span lang=EN-US>&nbsp;&nbsp; </span></span><span lang=EN-US style="FONT-FAMILY: 宋体">index=(ch1-0x81)*190+(ch2-0x40)-(ch2/128)</span><span style="FONT-FAMILY: 宋体">，其中<span lang=EN-US>ch1</span>是第一字节，<span lang=EN-US>ch2</span>是第二字节。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="COLOR: black; FONT-FAMILY: 宋体">具体的，</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: #3700c8; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; GBKindex = ((unsigned char)GBKword[0]-129)*190 +</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: #3700c8; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ((unsigned char)GBKword[1]-64) - (unsigned char)GBKword[1]/128;</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体"></span>&nbsp;</p>
<p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">三、怎样判断一个汉字的是什么编码</span></p>
<p class=MsoNormal><span style="COLOR: black; FONT-FAMILY: 宋体">直接根据汉字的编码范围判断，对于<span lang=EN-US>GB2312</span>和<span lang=EN-US>GBK</span>可用下面两个程序实现。</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">1</span><span style="COLOR: black; FONT-FAMILY: 宋体">、判断是否是<span lang=EN-US>GB2312</span></span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">bool isGBCode(const string&amp; strIn)</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">{</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; unsigned char ch1;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; unsigned char ch2;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; </span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; if (strIn.size() &gt;= 2)</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; {</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ch1 = (unsigned char)strIn.at(0);</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ch2 = (unsigned char)strIn.at(1);</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (ch1&gt;=176 &amp;&amp; ch1&lt;=247 &amp;&amp; ch2&gt;=160 &amp;&amp; ch2&lt;=254)</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return true;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else return false;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; }</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; else return false;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; FONT-FAMILY: Fixedsys">}</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">2</span><span style="COLOR: black; FONT-FAMILY: 宋体">、判断是否是<span lang=EN-US>GBK</span>编码</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">bool isGBKCode(const string&amp; strIn)</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">{</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; unsigned char ch1;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; unsigned char ch2;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; </span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; </span><span style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">if (strIn.size() &gt;= 2)</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; {</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">ch1 = (unsigned char)strIn.at(0);</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ch2 = (unsigned char)strIn.at(1);</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (ch1&gt;=129 &amp;&amp; ch1&lt;=254 &amp;&amp; ch2&gt;=64 &amp;&amp; ch2&lt;=254)</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return true;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else return false;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; }</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span lang=EN-US style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">&nbsp;&nbsp;&nbsp; else return false;</span></p>
<p class=MsoNormal style="TEXT-ALIGN: left" align=left><span style="FONT-SIZE: 10pt; COLOR: black; FONT-FAMILY: Fixedsys">}</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体"></span>&nbsp;</p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">3</span><span style="COLOR: black; FONT-FAMILY: 宋体">、对于<span lang=EN-US>Big5</span></span></p>
<p class=MsoPlainText><span lang=EN-US>&nbsp;&nbsp;&nbsp; </span>它的范围为：高字节从<span lang=EN-US>0xA0</span>到<span lang=EN-US>0xFE</span>，低字节从<span lang=EN-US>0x40</span>到<span lang=EN-US>0x7E</span>，和<span lang=EN-US>0xA1</span>到<span lang=EN-US>0xFE</span>两部分。判断一个汉字是否是<span lang=EN-US>BIG5</span>编码，可以如上对字符的编码范围判断即可。如何定位呢？那么也想象所有编码排列为一个二维坐标，纵坐标是高字节，横坐标是低字节。这样一行上的汉字个数：<span lang=EN-US>(0x7E-0x40+1)+(0xFE-0xA1+1)</span>＝<span lang=EN-US>157</span>。那么定位算法分两块，为<span lang=EN-US>: &nbsp;</span></p>
<p class=MsoPlainText><span lang=EN-US>&nbsp;&nbsp;&nbsp; if 0x40&lt;=ch2&lt;=0x7E: #is big5 char </span></p>
<p class=MsoPlainText>&nbsp;&nbsp;&nbsp; <span lang=PT-BR>index=((ch1-0xA1)*157+(ch2-0x40))*2 </span></p>
<p class=MsoPlainText><span lang=EN-US>&nbsp;&nbsp;&nbsp; elif 0xA1&lt;=ch2&lt;=0xFE: #is big5 char </span></p>
<p class=MsoPlainText>&nbsp;&nbsp;&nbsp; <span lang=PT-BR>index=((ch1-0xA1)*157+(ch2-0xA1+63))*2 </span></p>
<p class=MsoPlainText><span lang=PT-BR></span>&nbsp;</p>
<p class=MsoNormal><span style="FONT-FAMILY: 宋体">对于第二块，计算偏移量时因为有两块数值，所以在计算后面一段值时，不要忘了前面还有一段值。<span lang=EN-US>0x7E-0x40+1=63</span>。</span></p>
<p class=MsoNormal><span lang=EN-US style="FONT-FAMILY: 宋体"></span>&nbsp;</p>
<p class=MsoNormal><span style="FONT-FAMILY: 宋体">四、如果判断一个字符是西文字符还是中文字符</span></p>
<p class=MsoNormal><span lang=EN-US style="FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="FONT-FAMILY: 宋体">大家知道西文字符主要是指</span><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">ASCII</span><span style="COLOR: black; FONT-FAMILY: 宋体">码，它用一个字节表示。且这个字符转换成数字之后，该数字是大于<span lang=EN-US>0</span>的，而汉字是两个字节的，第一个字节的转化为数字之后应该是小于<span lang=EN-US>0</span>的，因此可以根据每个字节转化为数字之后是否小于<span lang=EN-US>0</span>，判断它是否是汉字。</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp; </span><span style="COLOR: black; FONT-FAMILY: 宋体">例如，设输入字为<span lang=EN-US>strin</span>，则，</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp;&nbsp; If (strin.at(0) &lt; 0)</span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; cout &lt;&lt; &#8221;</span><span style="COLOR: black; FONT-FAMILY: 宋体">是汉字<span lang=EN-US>&#8221; &lt;&lt; endl;</span></span></p>
<p class=MsoNormal><span lang=EN-US style="COLOR: black; FONT-FAMILY: 宋体">&nbsp;&nbsp;&nbsp;&nbsp; else cout &lt;&lt; &#8221;</span><span style="COLOR: black; FONT-FAMILY: 宋体">不是汉字<span lang=EN-US>&#8221; &lt;&lt; endl;</span></span></p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/35693.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-11-01 14:53 <a href="http://www.cppblog.com/jacky2019/archive/2007/11/01/35693.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>常用数据类型使用转换详解</title><link>http://www.cppblog.com/jacky2019/archive/2007/11/01/35657.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 01 Nov 2007 01:30:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/11/01/35657.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/35657.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/11/01/35657.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/35657.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/35657.html</trackback:ping><description><![CDATA[<p><font style="BACKGROUND-COLOR: #ffffff" color=#009900><a href="http://www.vckbase.com/study/article/data_convert.htm">http://www.vckbase.com/study/article/data_convert.htm</a></font><font color=#009900></p>
<p align=left><br><br><br><strong>常用数据类型使用转换详解<br></strong></font>作者：程佩君</p>
<hr noShade SIZE=1>
<table class=big height=2155 cellSpacing=0 cellPadding=0 width="100%" border=0>
    <tbody>
        <tr>
            <td vAlign=top height=2418>
            <p>读者层次：<font color=#006699>初学</font><br><br>刚接触VC编程的朋友往往对许多数据类型的转换感到迷惑不解，本文将介绍一些常用数据类型的使用。<br><br>我们先定义一些常见类型变量借以说明</p>
            <p>int i = 100;<br>long l = 2001;<br>float f=300.2;<br>double d=12345.119;<br>char username[]="程佩君";<br>char temp[200];<br>char *buf;<br>CString str;<br>_variant_t v1;<br>_bstr_t v2;<br><br><font color=#6699ff><strong>一、其它数据类型转换为字符串</strong></font><br></p>
            <ul>
                <li><font color=#6699ff>短整型(int)</font><br>itoa(i,temp,10);///将i转换为字符串放入temp中,最后一个数字表示十进制<br>itoa(i,temp,2); ///按二进制方式转换
                <li><font color=#6699ff>长整型(long)</font><br>ltoa(l,temp,10);
                <li><font color=#6699ff>浮点数(float,double)</font><br>用fcvt可以完成转换,这是MSDN中的例子:<br>int decimal, sign; <br>char *buffer; <br>double source = 3.1415926535; <br>buffer = _fcvt( source, 7, &amp;decimal, &amp;sign ); <br>运行结果:source: 3.1415926535 buffer: '31415927' decimal: 1 sign: 0<br>decimal表示小数点的位置,sign表示符号:0为正数，1为负数
                <li><font color=#6699ff>CString变量</font><br>str = "2008北京奥运";<br>buf = (LPSTR)(LPCTSTR)str;
                <li><font color=#6699ff>BSTR变量</font><br>BSTR bstrValue = ::SysAllocString(L"程序员"); <br>char * buf = _com_util::ConvertBSTRToString(bstrValue); <br>SysFreeString(bstrValue); <br>AfxMessageBox(buf); <br>delete(buf);
                <li><font color=#6699ff>CComBSTR变量</font><br>CComBSTR bstrVar("test"); <br>char *buf = _com_util::ConvertBSTRToString(bstrVar.m_str); <br>AfxMessageBox(buf); <br>delete(buf); <br>
                <li><font color=#6699ff>_bstr_t变量</font><br>_bstr_t类型是对BSTR的封装，因为已经重载了=操作符，所以很容易使用<br>_bstr_t bstrVar("test"); <br>const char *buf = bstrVar;///不要修改buf中的内容 <br>AfxMessageBox(buf); <br><br>
                <li><font color=#6699ff>通用方法(针对非COM数据类型)</font><br>用sprintf完成转换<br>
                <pre>char  buffer[200];
                char  c = '1';
                int   i = 35;
                long  j = 1000;
                float f = 1.7320534f;
                sprintf( buffer, "%c",c);
                sprintf( buffer, "%d",i);
                sprintf( buffer, "%d",j);
                sprintf( buffer, "%f",f);
                </pre>
                </li>
            </ul>
            <p><strong><font color=#6699ff>二、字符串转换为其它数据类型</font></strong><br>strcpy(temp,"123"); </p>
            <ul>
                <li><font color=#6699ff>短整型(int)</font><br>i = atoi(temp);
                <li><font color=#6699ff>长整型(long)</font><br>l = atol(temp);
                <li><font color=#6699ff>浮点(double)</font><br>d = atof(temp);
                <li><font color=#6699ff>CString变量</font><br>CString name = temp;
                <li><font color=#6699ff>BSTR变量</font> <br>BSTR bstrValue = ::SysAllocString(L"程序员"); <br>...///完成对bstrValue的使用<br>SysFreeString(bstrValue); <br>
                <li><font color=#6699ff>CComBSTR变量</font><br>CComBSTR类型变量可以直接赋值<br>CComBSTR bstrVar1("test");<br>CComBSTR bstrVar2(temp);<br>
                <li><font color=#6699ff>_bstr_t变量</font><br>_bstr_t类型的变量可以直接赋值<br>_bstr_t bstrVar1("test"); <br>_bstr_t bstrVar2(temp); <br><br></li>
            </ul>
            <p><strong><font color=#6699ff>三、其它数据类型转换到CString</font></strong><br>使用CString的成员函数Format来转换,例如:<br></p>
            <ul>
                <li>整数(int)<br>str.Format("%d",i);
                <li>浮点数(float)<br>str.Format("%f",i);
                <li>字符串指针(char *)等已经被CString构造函数支持的数据类型可以直接赋值<br>str = username;
                <li>对于Format所不支持的数据类型，可以通过上面所说的关于其它数据类型转化到char *的方法先转到char *，然后赋值给CString变量。<br></li>
            </ul>
            <p><strong><font color=#6699ff>四、BSTR、_bstr_t与CComBSTR</font></strong><br></p>
            <ul>
                <li>CComBSTR 是ATL对BSTR的封装，_bstr_t是C++对BSTR的封装,BSTR是32位指针,但并不直接指向字串的缓冲区。<br>char *转换到BSTR可以这样: <br>BSTR b=_com_util::ConvertStringToBSTR("数据");///使用前需要加上comutil.h和comsupp.lib<br>SysFreeString(bstrValue); <br>反之可以使用<br>char *p=_com_util::ConvertBSTRToString(b);<br>delete p;<br>具体可以参考一，二段落里的具体说明。<br><br>CComBSTR与_bstr_t对大量的操作符进行了重载，可以直接进行=,!=,==等操作，所以使用非常方便。<br>特别是_bstr_t,建议大家使用它。<br></li>
            </ul>
            <p>&#160;</p>
            <p><strong><font color=#6699ff>五、VARIANT 、_variant_t 与 COleVariant</font></strong><br></p>
            <ul>
                <li>VARIANT的结构可以参考头文件VC98\Include\OAIDL.H中关于结构体tagVARIANT的定义。<br>对于VARIANT变量的赋值：首先给vt成员赋值，指明数据类型，再对联合结构中相同数据类型的变量赋值，举个例子：<br>VARIANT va;<br>int a=2001;<br>va.vt=VT_I4;///指明整型数据<br>va.lVal=a; ///赋值<br><br>对于不马上赋值的VARIANT，最好先用Void VariantInit(VARIANTARG FAR* pvarg);进行初始化,其本质是将vt设置为VT_EMPTY,下表我们列举vt与常用数据的对应关系:<br><br>
                <table cellSpacing=1 cellPadding=0 width=792 bgColor=#333333 border=0>
                    <tbody>
                        <tr bgColor=#ffffff>
                            <td width=442>Byte bVal; </td>
                            <td width=338>// VT_UI1.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>Short iVal; </td>
                            <td width=338>// VT_I2.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>long lVal; </td>
                            <td width=338>// VT_I4.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>float fltVal; </td>
                            <td width=338>// VT_R4.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>double dblVal; </td>
                            <td width=338>// VT_R8.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>VARIANT_BOOL boolVal; </td>
                            <td width=338>// VT_BOOL.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>SCODE scode; </td>
                            <td width=338>// VT_ERROR.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>CY cyVal; </td>
                            <td width=338>// VT_CY.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>DATE date; </td>
                            <td width=338>// VT_DATE.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>BSTR bstrVal; </td>
                            <td width=338>// VT_BSTR.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>DECIMAL FAR* pdecVal </td>
                            <td width=338>// VT_BYREF|VT_DECIMAL.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>IUnknown FAR* punkVal; </td>
                            <td width=338>// VT_UNKNOWN.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>IDispatch FAR* pdispVal; </td>
                            <td width=338>// VT_DISPATCH.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>SAFEARRAY FAR* parray; </td>
                            <td width=338>// VT_ARRAY|*.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>Byte FAR* pbVal; </td>
                            <td width=338>// VT_BYREF|VT_UI1.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>short FAR* piVal; </td>
                            <td width=338>// VT_BYREF|VT_I2.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>long FAR* plVal; </td>
                            <td width=338>// VT_BYREF|VT_I4.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>float FAR* pfltVal; </td>
                            <td width=338>// VT_BYREF|VT_R4.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>double FAR* pdblVal; </td>
                            <td width=338>// VT_BYREF|VT_R8.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>VARIANT_BOOL FAR* pboolVal; </td>
                            <td width=338>// VT_BYREF|VT_BOOL.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>SCODE FAR* pscode; </td>
                            <td width=338>// VT_BYREF|VT_ERROR.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>CY FAR* pcyVal; </td>
                            <td width=338>// VT_BYREF|VT_CY.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>DATE FAR* pdate; </td>
                            <td width=338>// VT_BYREF|VT_DATE.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>BSTR FAR* pbstrVal; </td>
                            <td width=338>// VT_BYREF|VT_BSTR.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>IUnknown FAR* FAR* ppunkVal; </td>
                            <td width=338>// VT_BYREF|VT_UNKNOWN.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>IDispatch FAR* FAR* ppdispVal; </td>
                            <td width=338>// VT_BYREF|VT_DISPATCH.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>SAFEARRAY FAR* FAR* pparray; </td>
                            <td width=338>// VT_ARRAY|*.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>VARIANT FAR* pvarVal; </td>
                            <td width=338>// VT_BYREF|VT_VARIANT.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>void FAR* byref; </td>
                            <td width=338>// Generic ByRef.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>char cVal; </td>
                            <td width=338>// VT_I1.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>unsigned short uiVal; </td>
                            <td width=338>// VT_UI2.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>unsigned long ulVal; </td>
                            <td width=338>// VT_UI4.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>int intVal; </td>
                            <td width=338>// VT_INT.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>unsigned int uintVal; </td>
                            <td width=338>// VT_UINT.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>char FAR * pcVal; </td>
                            <td width=338>// VT_BYREF|VT_I1.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>unsigned short FAR * puiVal; </td>
                            <td width=338>// VT_BYREF|VT_UI2.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>unsigned long FAR * pulVal; </td>
                            <td width=338>// VT_BYREF|VT_UI4.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>int FAR * pintVal; </td>
                            <td width=338>// VT_BYREF|VT_INT.</td>
                        </tr>
                        <tr bgColor=#ffffff>
                            <td width=442>unsigned int FAR * puintVal; </td>
                            <td width=338>//VT_BYREF|VT_UINT.</td>
                        </tr>
                    </tbody>
                </table>
                <br>
                <li>_variant_t是VARIANT的封装类，其赋值可以使用强制类型转换，其构造函数会自动处理这些数据类型。<br>使用时需加上#include &lt;comdef.h&gt;<br>例如：<br>long l=222;<br>ing i=100;<br>_variant_t lVal(l);<br>lVal = (long)i;<br><br>
                <li>COleVariant的使用与_variant_t的方法基本一样，请参考如下例子：<br>COleVariant v3 = "字符串", v4 = (long)1999;<br>CString str =(BSTR)v3.pbstrVal;<br>long i = v4.lVal;<br><br></li>
            </ul>
            <p><font color=#6699ff><strong>六、其它一些COM数据类型</strong></font></p>
            <ul>
                <li>根据ProgID得到CLSID<br>HRESULT CLSIDFromProgID( LPCOLESTR lpszProgID,LPCLSID pclsid);<br>CLSID clsid;<br>CLSIDFromProgID( L"MAPI.Folder",&amp;clsid);<br>
                <li>根据CLSID得到ProgID<br>WINOLEAPI ProgIDFromCLSID( REFCLSID clsid,LPOLESTR * lplpszProgID); <br>例如我们已经定义了 CLSID_IApplication,下面的代码得到ProgID<br>LPOLESTR pProgID = 0;<br>ProgIDFromCLSID( CLSID_IApplication,&amp;pProgID);<br>...///可以使用pProgID <br>CoTaskMemFree(pProgID);//不要忘记释放 <br></li>
            </ul>
            <p><font color=#6699ff><strong>七、ANSI与Unicode<br></strong></font>Unicode称为宽字符型字串,COM里使用的都是Unicode字符串。</p>
            <ul>
                <li>将ANSI转换到Unicode<br>(1)通过L这个宏来实现，例如: CLSIDFromProgID( L"MAPI.Folder",&amp;clsid);<br>(2)通过MultiByteToWideChar函数实现转换,例如:<br>char *szProgID = "MAPI.Folder";<br>WCHAR szWideProgID[128];<br>CLSID clsid;<br>long lLen = MultiByteToWideChar(CP_ACP,0,szProgID,strlen(szProgID),szWideProgID,sizeof(szWideProgID));<br>szWideProgID[lLen] = '\0'; <br>(3)通过A2W宏来实现,例如: <br>USES_CONVERSION; <br>CLSIDFromProgID( A2W(szProgID),&amp;clsid);
                <li>将Unicode转换到ANSI<br>(1)使用WideCharToMultiByte,例如:<br>// 假设已经有了一个Unicode 串 wszSomeString... <br>char szANSIString [MAX_PATH]; <br>WideCharToMultiByte ( CP_ACP, WC_COMPOSITECHECK, wszSomeString, -1, szANSIString, sizeof(szANSIString), NULL, NULL ); <br>(2)使用W2A宏来实现,例如:<br>USES_CONVERSION;<br>pTemp=W2A(wszSomeString); </li>
            </ul>
            <p><font color=#6699ff><strong>八、其它</strong></font></p>
            <ul>
                <li>对消息的处理中我们经常需要将WPARAM或LPARAM等32位数据（DWORD)分解成两个16位数据（WORD),例如：<br>LPARAM lParam;<br>WORD loValue = LOWORD(lParam);///取低16位<br>WORD hiValue = HIWORD(lParam);///取高16位<br><br>
                <li>对于16位的数据(WORD)我们可以用同样的方法分解成高低两个8位数据(BYTE),例如:<br>WORD wValue;<br>BYTE loValue = LOBYTE(wValue);///取低8位<br>BYTE hiValue = HIBYTE(wValue);///取高8位<br><br>
                <li>两个16位数据（WORD）合成32位数据(DWORD,LRESULT,LPARAM,或WPARAM)<br>LONG MAKELONG( WORD wLow, WORD wHigh );<br>WPARAM MAKEWPARAM( WORD wLow, WORD wHigh ); <br>LPARAM MAKELPARAM( WORD wLow, WORD wHigh );<br>LRESULT MAKELRESULT( WORD wLow, WORD wHigh ); <br><br>
                <li>两个8位的数据(BYTE)合成16位的数据(WORD)<br>WORD MAKEWORD( BYTE bLow, BYTE bHigh ); <br><br>
                <li>从R(red),G(green),B(blue)三色得到COLORREF类型的颜色值<br>COLORREF RGB( BYTE byRed,BYTE byGreen,BYTE byBlue );<br>例如COLORREF bkcolor = RGB(0x22,0x98,0x34);<br><br>
                <li>从COLORREF类型的颜色值得到RGB三个颜色值<br>BYTE Red = GetRValue(bkcolor); ///得到红颜色<br>BYTE Green = GetGValue(bkcolor); ///得到绿颜色<br>BYTE Blue = GetBValue(bkcolor); ///得到兰颜色<br></li>
            </ul>
            <p><font color=#6699ff><strong>九、注意事项</strong></font><br>假如需要使用到ConvertBSTRToString此类函数,需要加上头文件comutil.h,并在setting中加入comsupp.lib或者直接加上#pragma comment( lib, "comsupp.lib" )</p>
            <p>后记：本文匆匆写成，错误之处在所难免，欢迎指正.</p>
            </td>
        </tr>
    </tbody>
</table>
<img src ="http://www.cppblog.com/jacky2019/aggbug/35657.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-11-01 09:30 <a href="http://www.cppblog.com/jacky2019/archive/2007/11/01/35657.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>“休息”了半年</title><link>http://www.cppblog.com/jacky2019/archive/2007/10/29/35443.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Mon, 29 Oct 2007 05:34:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/10/29/35443.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/35443.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/10/29/35443.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/35443.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/35443.html</trackback:ping><description><![CDATA[我又回来了。<img height=20 src="http://www.cppblog.com/Emoticons/QQ/06.gif" width=20 border=0><br><br>人生就像龟兔赛跑。一开始领先不代表你会一直领先。同样，一开始落后不代表你会一直落后。但是如果你下不了决心，不可能努力，没有自制力，没毅力的话，你肯定不会领先的。<br><br>乐观，开朗，积极的态度。好长时间没有体会到领先带给我的乐趣了；好长时间没有体会到学习技术带给我的快乐了。该重新归队，好好学习，好好工作了。<br><br>废话不多说，开工。<img height=20 src="http://www.cppblog.com/Emoticons/QQ/12.gif" width=20 border=0>
<img src ="http://www.cppblog.com/jacky2019/aggbug/35443.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-10-29 13:34 <a href="http://www.cppblog.com/jacky2019/archive/2007/10/29/35443.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>python与c++交互学习入门之5</title><link>http://www.cppblog.com/jacky2019/archive/2007/06/06/25662.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Wed, 06 Jun 2007 08:25:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/06/06/25662.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/25662.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/06/06/25662.html#Feedback</comments><slash:comments>7</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/25662.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/25662.html</trackback:ping><description><![CDATA[<p>这次讲讲，如何扩展c++库。通过boost.python把c++库编译成python能够调用的dll。</p>
<p><br>通过上一次的教程后，大家都应该会使用boost.python了。把c++程序编译成pyd文件。由于c++有很多特性，所以，如果你的程</p>
<p>序用了很多的c++特性的话，那么你必须做很多工作了。像虚拟函数，函数重载，继承，默认值等等。具体如何转化，请参</p>
<p>boost.python的文档了。</p>
<p><br>这几天尝试着把c++程序库编译成python可调用的dll，不知道为什么一直不可用。。很是郁闷。老是显示如下的错误：</p>
<p>Traceback (most recent call last):<br>&nbsp; File "&lt;pyshell#3&gt;", line 1, in &lt;module&gt;<br>&nbsp;&nbsp;&nbsp; import pydll<br>ImportError: No module named pydll</p>
<p>意思是说找不到dll。我把dll都copy到python/dlls下了还是不行，而且我确定python的sys.path包含了python/dlls目录了。</p>
<p>很是不解。网上也很难找到资料，google了很长时间找不到有用的资料，好像中文方面的资料很少的。今天尝试了一下google</p>
<p>英文资料，终于有了新的发现：<br><a href="http://mail.python.org/pipermail/c++-sig/2007-February/011971.html">http://mail.python.org/pipermail/c++-sig/2007-February/011971.html</a><br>You are using Python2.5. In this version of Python you have to have<br>file extension<br>to be "pyd" - sge.pyd</p>
<p>-- <br>Roman Yakovenko<br>C++ Python language binding<br><a href="http://www.language-binding.net/">http://www.language-binding.net/</a></p>
<p>有人碰到的问题跟我的是一样的。后面那个Roman回答了一下，是文件扩展名的问题！！！为什么不支持dll呢？不解。回去试</p>
<p>了一下把后缀名改了就成功了。。。why???</p>
<p><br>下面来看一下我的那个简单的例子：<br>这个例子来自于网上，<br><a href="http://www.vckbase.com/document/viewdoc/?id=1540">http://www.vckbase.com/document/viewdoc/?id=1540</a><br>C++ 扩展和嵌入 Python<br>作者：胡金山<br>源码下载地址：<a href="http://www.vckbase.com/code/downcode.asp?id=2777">http://www.vckbase.com/code/downcode.asp?id=2777</a></p>
<p><br>这是一个非常简单的dll工程。给python提供了一个函数static PyObject* Recognise(PyObject *self, PyObject *args)。</p>
<p><br>1、不使用boost.python库来直接构建dll<br>接下来，我们来用C++为Python编写扩展模块(动态链接库)，并在Python程序中调用C++开发的扩展功能函数。生成一个取名为</p>
<p>pyUtil的Win32 DLL工程，除了pyUtil.cpp文件以外，从工程中移除所有其它文件，并填入如下的代码： </p>
<p>// pyUtil.cpp<br>#ifdef PYUTIL_EXPORTS<br>#define PYUTIL_API __declspec(dllexport)<br>#else<br>#define PYUTIL_API __declspec(dllimport)<br>#endif</p>
<p>#include&lt;windows.h&gt;<br>#include&lt;string&gt;<br>#include&lt;Python.h&gt;<br>BOOL APIENTRY DllMain( HANDLE hModule, <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; DWORD&nbsp; ul_reason_for_call, <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; LPVOID lpReserved<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; )<br>{<br>&nbsp;&nbsp;&nbsp; switch (ul_reason_for_call)<br>&nbsp;&nbsp;&nbsp; {<br>&nbsp;&nbsp;&nbsp; case DLL_PROCESS_ATTACH:<br>&nbsp;&nbsp;&nbsp; case DLL_THREAD_ATTACH:<br>&nbsp;&nbsp;&nbsp; case DLL_THREAD_DETACH:<br>&nbsp;&nbsp;&nbsp; case DLL_PROCESS_DETACH:<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; break;<br>&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp; return TRUE;<br>}<br>std::string Recognise_Img(const std::string url)<br>{<br>&nbsp;&nbsp;&nbsp; //返回结果<br>&nbsp;&nbsp;&nbsp; return "从dll中返回的数据... : " +url;<br>}<br>static PyObject* Recognise(PyObject *self, PyObject *args)<br>{<br>&nbsp;&nbsp;&nbsp; const char *url;<br>&nbsp;&nbsp;&nbsp; std::string sts;<br>&nbsp;&nbsp;&nbsp; if (!PyArg_ParseTuple(args, "s", &amp;url))<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return NULL;<br>&nbsp;&nbsp;&nbsp; sts = Recognise_Img(url);<br>&nbsp;&nbsp;&nbsp; return Py_BuildValue("s", sts.c_str() );<br>}<br>static PyMethodDef AllMyMethods[] = {<br>&nbsp;&nbsp;&nbsp; {"Recognise",&nbsp; Recognise, METH_VARARGS},//暴露给Python的函数<br>&nbsp;&nbsp;&nbsp; {NULL,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; NULL}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; /* Sentinel */<br>};<br>extern "C" PYUTIL_API void initpyUtil()<br>{<br>&nbsp;&nbsp;&nbsp; PyObject *m, *d;<br>&nbsp;&nbsp;&nbsp; m = Py_InitModule("pyUtil", AllMyMethods); //初始化本模块，并暴露函数<br>&nbsp;&nbsp;&nbsp; d = PyModule_GetDict(m);<br>}</p>
<p>在Python代码中调用这个动态链接库： (记得把dll的扩展名改为.pyd，另外dll的路径要能够被检索到)<br>import pyUtil<br>result = pyUtil.Recognise("input url of specific data")<br>print "the result is: "+ result</p>
<p>&nbsp;</p>
<p>2、使用boost.python库来构建dll<br>用C++为Python写扩展时，如果您愿意使用Boost.Python库的话，开发过程会变得更开心J，要编写一个与上述pyUtil同样功能</p>
<p>的动态链接库，只需把文件内容替换为下面的代码。当然，编译需要boost_python.lib支持，运行需要boost_python.dll支持</p>
<p>。 <br>#include&lt;string&gt;<br>#include &lt;boost/python.hpp&gt;<br>using namespace boost::python;<br>#pragma comment(lib, "boost_python.lib")<br>std::string strtmp;<br>char const* Recognise(const char* url)<br>{<br>&nbsp;&nbsp;&nbsp; strtmp ="从dll中返回的数据... : ";<br>&nbsp;&nbsp;&nbsp; strtmp+=url;<br>&nbsp;&nbsp;&nbsp; return strtmp.c_str();<br>}<br>BOOST_PYTHON_MODULE(pyUtil)<br>{<br>&nbsp;&nbsp;&nbsp; def("Recognise", Recognise);<br>}</p>
<p>可以非常明显地看到，用了boost.python库之后，简单了很多。因为boost.python为你做了很多的事情。。恩。</p>
<p>&nbsp;</p>
<p>好像没有讲很多有用的东西，光顾着讲了，呵呵。。。我也还在继续学习之中。下次写点什么呢？继续学习了哦</p>
<p>&nbsp;</p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/25662.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-06-06 16:25 <a href="http://www.cppblog.com/jacky2019/archive/2007/06/06/25662.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>c++中嵌入python入门4 之 Boost.Python</title><link>http://www.cppblog.com/jacky2019/archive/2007/06/01/25254.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Fri, 01 Jun 2007 02:32:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/06/01/25254.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/25254.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/06/01/25254.html#Feedback</comments><slash:comments>5</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/25254.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/25254.html</trackback:ping><description><![CDATA[<p>坏境python25 + vs2005 (2005真耗资源阿。。。)</p>
<p>有一段时间没写blog了。这几天都在研究怎么封装c++，让python可以用c++的库。在网上发现了boost.python这个好咚咚。不</p>
<p>过在使用过程中碰到一点问题。本文教大家如何把</p>
<p>char const* greet()<br>{<br>&nbsp;&nbsp; return "hello, world";<br>}</p>
<p>封装成python。实际上这是python教程里面的咚咚。</p>
<p><br>首先下载Boost，<a href="http://www.boost.org/">www.boost.org</a>。boost.python在boost里面了。在visual studio 2005 command prompt中navigation到</p>
<p>boost\boost_1_34_0\下。记得一定要用visual studio 2005 command prompt这个vs2005带的tools，不要用cmd.exe，否则会</p>
<p>碰到很多错误的。然后就是把bjam.exe拷贝到一个能被找到的目录下，或者直接也拷贝到boost\boost_1_34_0\下即可。然后，</p>
<p>设置python的根目录和python的版本，也可直接把它们加到坏境目录中，那样就不用每次都设置一下。<br>set PYTHON_ROOT=c:/python25<br>set PYTHON_VERSION=2.5</p>
<p>接着就可以直接运行了，bjam -sTOOLS=vc-8_0<br>整个编译过程要很长时间。。。</p>
<p>成功之后，就会有好多个boost_python-vc80-****.dll,.lib的，把他们都拷贝到一个能被系统找到的目录，不妨直接把他们都</p>
<p>扔到c:\windows\system32下。</p>
<p>接着，我们开始编译hello。navigation到boost\boost_1_34_0\libs\python\example\tutorial下，bjam -sTOOLS=vc-8_0运行</p>
<p>，在bin的目录下即会生成hello.pyd。这下就基本成功了，如果没成功的话，check一下上面boost_python的那些dll能否被系</p>
<p>统找到。另外，这里有python25的一个bug。。。我花了很长时间才在python的mail lists中找到了。寒。。。</p>
<p>错误如下所示：<br>D:\Learn\Python\boost\boost_1_34_0\libs\python\example\tutorial&gt;bjam<br>Jamroot:17: in modules.load<br>rule python-extension unknown in module Jamfile&lt;/D:/Learn/Python/boost/boost_1_3<br>4_0/libs/python/example/tutorial&gt;.<br>D:/Learn/Python/boost/boost_1_34_0/tools/build/v2/build\project.jam:312: in load<br>-jamfile<br>D:/Learn/Python/boost/boost_1_34_0/tools/build/v2/build\project.jam:68: in load<br>D:/Learn/Python/boost/boost_1_34_0/tools/build/v2/build\project.jam:170: in proj<br>ect.find<br>D:/Learn/Python/boost/boost_1_34_0/tools/build/v2\build-system.jam:237: in load<br>D:\Learn\Python\boost\boost_1_34_0\libs\python\example\..\..\..\tools\build\v2/k<br>ernel\modules.jam:261: in import<br>D:\Learn\Python\boost\boost_1_34_0\libs\python\example\..\..\..\tools\build\v2/k<br>ernel/bootstrap.jam:132: in boost-build<br>D:\Learn\Python\boost\boost_1_34_0\libs\python\example\boost-build.jam:7: in mod<br>ule scope</p>
<p>解决办法如下：<br>在boost\boost_1_34_0\tools\build\v2\目录下找到user-config.jam文件，打开在<br>import toolset : using ;<br>下面加一行代码：<br>using python ;<br>再重新编译一下boost，然后就没问题了。tutorial里面的hello能顺利编译通过。ps.这个问题困扰了我好长时间。。sigh。。</p>
<p>。</p>
<p>编译成功后会产生一个hello.pyd，在bin的目录下面。</p>
<p><br>有好多办法测试此hello.pyd是否可以用。<br>方法一，把它拷贝到python25\dlls下，打开IDLE，<br>&gt;&gt;&gt; import hello<br>&gt;&gt;&gt; hello.greet()<br>'hello, world'<br>&gt;&gt;&gt; <br>方法二，直接在当前目录下写一个python文件，然后直接调用hello.pyd即可。总之，hello.pyd就是一个python文件了。。嗯</p>
<p>。操作hello.pyd根其他python文件是一样的。<br>这样就成功了。</p>
<p>如果碰到如下错误，是因为系统找不到boost_python的dll。强烈建议把他们都扔到system32下！。</p>
<p>&gt;&gt;&gt; import hello</p>
<p>Traceback (most recent call last):<br>&nbsp; File "&lt;pyshell#0&gt;", line 1, in &lt;module&gt;<br>&nbsp;&nbsp;&nbsp; import hello<br>ImportError: DLL load failed: 找不到指定的模块。<br>&gt;&gt;&gt; </p>
<p><br>说明，hello.cpp在boost\boost_1_34_0\libs\python\example\tutorial目录下。里面的内容是：</p>
<p>//&nbsp; Copyright Joel de Guzman 2002-2004. Distributed under the Boost<br>//&nbsp; Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt <br>//&nbsp; or copy at <a href="http://www.boost.org/LICENSE_1_0.txt">http://www.boost.org/LICENSE_1_0.txt</a>)<br>//&nbsp; Hello World Example from the tutorial<br>//&nbsp; [Joel de Guzman 10/9/2002]</p>
<p>char const* greet()<br>{<br>&nbsp;&nbsp; return "hello, world";<br>}</p>
<p>#include &lt;boost/python/module.hpp&gt;<br>#include &lt;boost/python/def.hpp&gt;<br>using namespace boost::python;</p>
<p>BOOST_PYTHON_MODULE(hello)<br>{<br>&nbsp;&nbsp;&nbsp; def("greet", greet);<br>}</p>
<p><br>其中<br>BOOST_PYTHON_MODULE(hello)<br>{<br>&nbsp;&nbsp;&nbsp; def("greet", greet);<br>}<br>是对greet从c++向python的一个封装声明吧，装换就交给boost了。</p>
<p>&nbsp;</p>
<p>先写到这里了。下次再写。。嗯<br></p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/25254.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-06-01 10:32 <a href="http://www.cppblog.com/jacky2019/archive/2007/06/01/25254.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>笔记本主板坏了</title><link>http://www.cppblog.com/jacky2019/archive/2007/05/31/25203.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 31 May 2007 07:58:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/05/31/25203.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/25203.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/05/31/25203.html#Feedback</comments><slash:comments>2</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/25203.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/25203.html</trackback:ping><description><![CDATA[郁闷，极度郁闷。。。
<img src ="http://www.cppblog.com/jacky2019/aggbug/25203.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-05-31 15:58 <a href="http://www.cppblog.com/jacky2019/archive/2007/05/31/25203.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>c++中嵌入python入门3</title><link>http://www.cppblog.com/jacky2019/archive/2007/05/17/24286.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 17 May 2007 07:16:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/05/17/24286.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/24286.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/05/17/24286.html#Feedback</comments><slash:comments>3</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/24286.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/24286.html</trackback:ping><description><![CDATA[<p>这次主要讲讲怎么把python中的class嵌入到c++中去。<br>顺便讲讲元组的操作和怎么编译python源代码。</p>
<p>1. 首先讲讲元组的操作<br>由于参数是通过元组传进去的，所以我们不能老是通过Py_BuildValue这个函数来操作元组，那样太不方便了。<br>Python提供了元组相关的操作，下面这个例子演示了如何操作。主要是下面几个函数：<br>//new一个元组，传入size<br>pArgs = PyTuple_New(argc - 3);&nbsp;<br>//set元组的直，第一个为元组，第二个为index（从0开始），第三个为value<br>PyTuple_SetItem(pArgs,0,Py_BuildValue("i",2000) );<br>PyTuple_SetItem(pArgs,1,Py_BuildValue("i",8) );</p>
<p>来自python doc的一个例子</p>
<p>#include &lt;Python.h&gt;<br>int<br>main(int argc, char *argv[])<br>{<br>&nbsp;&nbsp;&nbsp; PyObject *pName, *pModule, *pDict, *pFunc;<br>&nbsp;&nbsp;&nbsp; PyObject *pArgs, *pValue;<br>&nbsp;&nbsp;&nbsp; int i;</p>
<p>&nbsp;&nbsp;&nbsp; if (argc &lt; 3) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr,"Usage: call pythonfile funcname [args]\n");<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return 1;<br>&nbsp;&nbsp;&nbsp; }</p>
<p>&nbsp;&nbsp;&nbsp; Py_Initialize();<br>&nbsp;&nbsp;&nbsp; pName = PyString_FromString(argv[1]);<br>&nbsp;&nbsp;&nbsp; /* Error checking of pName left out */</p>
<p>&nbsp;&nbsp;&nbsp; pModule = PyImport_Import(pName);<br>&nbsp;&nbsp;&nbsp; Py_DECREF(pName);</p>
<p>&nbsp;&nbsp;&nbsp; if (pModule != NULL) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; pFunc = PyObject_GetAttrString(pModule, argv[2]);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; /* pFunc is a new reference */</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (pFunc &amp;&amp; PyCallable_Check(pFunc)) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; pArgs = PyTuple_New(argc - 3);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; for (i = 0; i &lt; argc - 3; ++i) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; pValue = PyInt_FromLong(atoi(argv[i + 3]));<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (!pValue) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pArgs);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pModule);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr, "Cannot convert argument\n");<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return 1;<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; /* pValue reference stolen here: */<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; PyTuple_SetItem(pArgs, i, pValue);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; pValue = PyObject_CallObject(pFunc, pArgs);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pArgs);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (pValue != NULL) {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; printf("Result of call: %ld\n", PyInt_AsLong(pValue));<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pValue);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pFunc);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pModule);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; PyErr_Print();<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr,"Call failed\n");<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return 1;<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (PyErr_Occurred())<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; PyErr_Print();<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr, "Cannot find function \"%s\"\n", argv[2]);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_XDECREF(pFunc);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Py_DECREF(pModule);<br>&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp; else {<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; PyErr_Print();<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; fprintf(stderr, "Failed to load \"%s\"\n", argv[1]);<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return 1;<br>&nbsp;&nbsp;&nbsp; }<br>&nbsp;&nbsp;&nbsp; Py_Finalize();<br>&nbsp;&nbsp;&nbsp; return 0;<br>}</p>
<p><br>2. class操作<br>把下面加入到test2.py中去。定义了一个很简单的类，有一个name成员变量，一个printName成员函数<br>class TestClass:<br>&nbsp;&nbsp;&nbsp; def __init__(self,name):<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; self.name = name</p>
<p>&nbsp;&nbsp;&nbsp; def printName(self):<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; print self.name</p>
<p>cpp文件<br>#include &lt;python.h&gt;<br>int main()<br>{<br>&nbsp;Py_Initialize();</p>
<p>&nbsp;PyObject * pModule = NULL;<br>&nbsp;PyObject * pFunc&nbsp;&nbsp; = NULL;<br>&nbsp;PyObject * pArg&nbsp;&nbsp;&nbsp; = NULL;<br>&nbsp;PyObject * pClass&nbsp; = NULL;<br>&nbsp;PyObject * pObject = NULL;</p>
<p>&nbsp;pModule = PyImport_ImportModule("test2");<br>&nbsp;pClass&nbsp; = PyObject_GetAttrString(pModule, "TestClass");//得到那个类<br>&nbsp;pArg = PyTuple_New(1);<br>&nbsp;PyTuple_SetItem(pArg, 0, Py_BuildValue("s", "Jacky"));<br>&nbsp;pObject = PyEval_CallObject(pClass, pArg);//生成一个对象，或者叫作实例</p>
<p>&nbsp;pFunc = PyObject_GetAttrString(pObject, "printName");//得到该实例的成员函数<br>&nbsp;PyEval_CallObject(pFunc, NULL);//执行该实例的成员函数</p>
<p>&nbsp;Py_Finalize();</p>
<p>&nbsp;return 0;<br>}</p>
<p><br>没有什么资料，就先写到这里了。下面介绍一下怎么build python25的源代码</p>
<p>3. 编译python源代码<br>为什么要编译呢？因为没有python25_d.lib！呵呵。顺便可以了解一下代码结构。<br>解压缩后，有好多目录，其中pcbuild和pcbuild8是我们要的。pcbuild对应着vc7.1的,pcbuild8对应着vc8.0的<br>因为在用vc7.1，也就是2003了。所以我就说说怎么用2003来编译吧。事实上是从一位牛人那里学来的</p>
<p><a href="http://blog.donews.com/lemur/archive/2005/12/17/660973.aspx">http://blog.donews.com/lemur/archive/2005/12/17/660973.aspx</a>，那位大哥大概一年半前就在解剖python了，厉害</p>
<p>阿。看来我只能后来居上了，娃哈哈。我按照他说的试了一下，编译成功！</p>
<p>不过遇到一点小问题，用vc2003打开那个solution的时候，发现作者没有把source code control去掉，郁闷！害的我</p>
<p>们打开的时候一堆messagebox。不过不用管它就好了，一直确定。最后试了一下那个python25_d.lib，没问题。不过记</p>
<p>得把python25_d.dll copy到一个能被找到的目录，比如说c:\windows\system32\下面。python25.dll也在这个目录下</p>
<p>面。over。恩。</p>
<p>&nbsp;</p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/24286.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-05-17 15:16 <a href="http://www.cppblog.com/jacky2019/archive/2007/05/17/24286.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>c++中嵌入python入门2</title><link>http://www.cppblog.com/jacky2019/archive/2007/05/17/24276.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 17 May 2007 03:28:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/05/17/24276.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/24276.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/05/17/24276.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/24276.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/24276.html</trackback:ping><description><![CDATA[<p>1. 一个有一个参数的例子</p>
<p>python文件<br>#Filename test2.py</p>
<p>def Hello(s):<br>&nbsp;&nbsp;&nbsp; print "Hello, world!"<br>&nbsp;&nbsp;&nbsp; print s</p>
<p>cpp文件<br>#include &lt;python.h&gt;<br>int main()<br>{<br>&nbsp;Py_Initialize();</p>
<p>&nbsp;PyObject * pModule = NULL;<br>&nbsp;PyObject * pFunc&nbsp;&nbsp; = NULL;<br>&nbsp;PyObject * pArg&nbsp;&nbsp;&nbsp; = NULL;</p>
<p>&nbsp;pModule = PyImport_ImportModule("test2");<br>&nbsp;pFunc&nbsp;&nbsp; = PyObject_GetAttrString(pModule, "Hello");<br>&nbsp;pArg&nbsp;&nbsp;&nbsp; = Py_BuildValue("(s)", "function with argument");</p>
<p>&nbsp;PyEval_CallObject(pFunc, pArg);</p>
<p>&nbsp;Py_Finalize();</p>
<p>&nbsp;return 0;<br>}</p>
<p>注意，参数要以tuple元组形式传入。因为这个函数只要一个参数，所以我们直接使用(s)构造一个元组了。</p>
<p>2. 一个有两个参数的例子</p>
<p>python文件中加入以下代码，一个加函数<br>def Add(a, b):<br>&nbsp;&nbsp;&nbsp; print "a+b=", a+b</p>
<p>cpp文件，只改了两行，有注释的那两行<br>#include &lt;python.h&gt;<br>int main()<br>{<br>&nbsp;Py_Initialize();</p>
<p>&nbsp;PyObject * pModule = NULL;<br>&nbsp;PyObject * pFunc&nbsp;&nbsp; = NULL;<br>&nbsp;PyObject * pArg&nbsp;&nbsp;&nbsp; = NULL;</p>
<p>&nbsp;pModule = PyImport_ImportModule("test2");<br>&nbsp;pFunc&nbsp;&nbsp; = PyObject_GetAttrString(pModule, "Add");//终于告别hello world了，开始使用新的函数<br>&nbsp;pArg&nbsp;&nbsp;&nbsp; = Py_BuildValue("(i,i)", 10, 15);//构造一个元组</p>
<p>&nbsp;PyEval_CallObject(pFunc, pArg);</p>
<p>&nbsp;Py_Finalize();</p>
<p>&nbsp;return 0;<br>}</p>
<p>其它的就类似了。。。基本上，我们知道了怎么在c++中使用python中的函数。接下来学习一下如何使用python中的</p>
<p>class。</p>
<p>附：Py_BuildValue的使用例子，来自python documentation：</p>
<p>&nbsp;&nbsp;&nbsp; Py_BuildValue("")&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; None<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("i", 123)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 123<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("iii", 123, 456, 789)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; (123, 456, 789)<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("s", "hello")&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 'hello'<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("ss", "hello", "world")&nbsp;&nbsp;&nbsp; ('hello', 'world')<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("s#", "hello", 4)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 'hell'<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("()")&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ()<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("(i)", 123)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; (123,)<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("(ii)", 123, 456)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; (123, 456)<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("(i,i)", 123, 456)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; (123, 456)<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("[i,i]", 123, 456)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; [123, 456]<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("{s:i,s:i}",<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; "abc", 123, "def", 456)&nbsp;&nbsp;&nbsp; {'abc': 123, 'def': 456}<br>&nbsp;&nbsp;&nbsp; Py_BuildValue("((ii)(ii)) (ii)",<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 1, 2, 3, 4, 5, 6)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; (((1, 2), (3, 4)), (5, 6))<br></p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/24276.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-05-17 11:28 <a href="http://www.cppblog.com/jacky2019/archive/2007/05/17/24276.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>c++中嵌入python入门1</title><link>http://www.cppblog.com/jacky2019/archive/2007/05/17/24269.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 17 May 2007 03:03:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/05/17/24269.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/24269.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/05/17/24269.html#Feedback</comments><slash:comments>6</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/24269.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/24269.html</trackback:ping><description><![CDATA[<p>本人是用vc2003+python2.5学习的，其它的也应该差不了多少</p>
<p>0. 坏境设置<br>把python的include/libs目录分别加到vc的include/lib directories中去。另外，由于python没有提供debug lib，体地说，就是没有提供python25_d.lib了。你可以自己编译python的源代码来得到python25_d.lib的，偶还没试过，呵呵。而且网上找了一下也没下载到。所以，如果你想要在debug下运行程序的话，你要把pyconfig.h（在python25/include/目录下）的大概是在283行，把pragma comment(lib,"python25_d.lib")改成pragma comment(lib,"python25.lib")，让python都使用非debug lib.</p>
<p>1. 开始编程了<br>#include &lt;python.h&gt;<br>第一步就是包含python的头文件</p>
<p>2. 看一个很简单的例子<br>1)python文件test.py，很简单的定义了一个函数</p>
<p>#Filename test.py<br>def Hello():<br>&nbsp;&nbsp;&nbsp; print "Hello, world!"</p>
<p>这个应该能看懂的吧？否则的话，回去再练练python吧，呵呵。《简明Python教程》Swaroop, C. H. 著。沈洁元&nbsp; 译。</p>
<p><br>2)cpp文件</p>
<p>#include &lt;python.h&gt; //包含头文件，在c++中嵌入python，这是必须的<br>int main()<br>{<br>&nbsp;Py_Initialize();</p>
<p>&nbsp;PyObject * pModule = NULL;<br>&nbsp;PyObject * pFunc&nbsp;&nbsp; = NULL;</p>
<p>&nbsp;pModule = PyImport_ImportModule("test");<br>&nbsp;pFunc&nbsp;&nbsp; = PyObject_GetAttrString(pModule, "Hello");<br>&nbsp;PyEval_CallObject(pFunc, NULL);</p>
<p>&nbsp;Py_Finalize();</p>
<p>&nbsp;return 0;<br>}</p>
<p>第一步还是包含头文件</p>
<p>第二步，使用python之前，要调用Py_Initialize();这个函数进行初始化。<br>帮助文档中如是说：<br>The basic initialization function is Py_Initialize(). This initializes the table of loaded modules, and creates the fundamental modules __builtin__, __main__, sys, and exceptions. It also initializes the module search path (sys.path). </p>
<p>反正，一开始你一定要调用。</p>
<p>第三步，声明一些Python的变量，PyObject类型的。其实声明也可放在前面，这个倒是无所谓的。</p>
<p>第四步，import module，也就是你的脚本名字，不需要加后缀名，否则会出错的。</p>
<p>第五步，从你import进来的module中得到你要的函数<br>&nbsp;pFunc&nbsp;&nbsp; = PyObject_GetAttrString(pModule, "Hello");<br>上面的例子已经够清楚的了，最后一个是你要得到的函数的名字</p>
<p>第六步，调用PyEval_CallObject来执行你的函数，第二个参数为我们要调用的函数的函数，本例子不含参数，所以设置为NULL。</p>
<p>第七步，调用Py_Finalize，这个根Py_Initialize相对应的。一个在最前面，一个在最后面。</p>
<p>&nbsp;</p>
<p>第一次写教程。这个例子非常简单，本人也还在学习当中阿，只能保证大家能够把这个例子运行起来。建议大家去看python的documentaion，里面有讲怎么embedding python的。先写到这里，其实目前也只学到这么多，呵呵。下次学了更多以后再写。Over。恩。</p>
<p>&nbsp;</p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/24269.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-05-17 11:03 <a href="http://www.cppblog.com/jacky2019/archive/2007/05/17/24269.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>sigh</title><link>http://www.cppblog.com/jacky2019/archive/2007/05/08/23647.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Tue, 08 May 2007 13:14:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/05/08/23647.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/23647.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/05/08/23647.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/23647.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/23647.html</trackback:ping><description><![CDATA[sigh sigh sigh ...
<img src ="http://www.cppblog.com/jacky2019/aggbug/23647.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-05-08 21:14 <a href="http://www.cppblog.com/jacky2019/archive/2007/05/08/23647.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>The difference between GetDC  and GetWindowDC</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/21/22467.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Sat, 21 Apr 2007 04:54:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/21/22467.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/22467.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/21/22467.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/22467.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/22467.html</trackback:ping><description><![CDATA[<p><span>GetDC</span></p>
<p>The GetDC function retrieves a handle of a display device context (DC) for the client area of the specified window. The display device context can be used in subsequent GDI functions to draw in the client area of the window. </p>
<p>This function retrieves a common, class, or private device context depending on the class style specified for the specified window. For common device contexts, GetDC assigns default attributes to the device context each time it is retrieved. For class and private device contexts, GetDC leaves the previously assigned attributes unchanged. </p>
<p>HDC GetDC(</p>
<p><span>&nbsp;&nbsp;&nbsp; </span>HWND hWnd <span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>// handle of window&nbsp;</p>
<p><span>&nbsp;&nbsp; </span>);<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></p>
<p>Parameters</p>
<p>hWnd</p>
<p>Identifies the window whose device context is to be retrieved.</p>
<p>&nbsp;</p>
<p><span>GetWindowDC</span></p>
<p>The GetWindowDC function retrieves the device context (DC) for the entire window, including title bar, menus, and scroll bars. A window device context permits painting anywhere in a window, because the origin of the device context is the upper-left corner of the window instead of the client area. </p>
<p>GetWindowDC assigns default attributes to the window device context each time it retrieves the device context. Previous attributes are lost. </p>
<p>HDC GetWindowDC(</p>
<p><span>&nbsp;&nbsp;&nbsp; </span>HWND hWnd <span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span>// handle of window&nbsp;</p>
<p><span>&nbsp;&nbsp; </span>);<span>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; </span></p>
<p>Parameters</p>
<p>hWnd</p>
<p>Identifies the window with a device context that is to be retrieved.</p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/22467.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-21 12:54 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/21/22467.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>求n!的结果中末尾0的个数</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/19/22306.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Thu, 19 Apr 2007 07:30:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/19/22306.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/22306.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/19/22306.html#Feedback</comments><slash:comments>1</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/22306.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/22306.html</trackback:ping><description><![CDATA[<p><span>终于悟到了。</span></p>
<p><span>首先，很容易想到的就是，求</span>0<span>的个数，就是求</span>5<span>的个数（如果这个你都想不明白的话，那就。。。再好好想想吧，呵呵）；</span></p>
<p><span>接下来，如何求</span>5<span>的个数呢？如果遍历一遍的话，那显然是太慢了！因为这种计算题太有规律了！想了好久，终于想出来了：</span></p>
<p>Result = 0; // <span>最后的结果</span></p>
<p>while ( N &gt;= 5 )</p>
<p>{</p>
<p>&nbsp;N /= 5;</p>
<p>&nbsp;Result += N;</p>
<p>}</p>
<p>// <span>结束了。</span></p>
<p>&nbsp;</p>
<p><span>没错，就是这么简单！下面简单说说为什么这样子做是对的（偶小试了一下，没问题，呵呵）：</span></p>
<p><span>第一次除以</span>5<span>表示</span>5<span>的倍数的个数，</span></p>
<p><span>第二次除以</span>5<span>表示</span>5<span>的平方的倍数的个数，（显然，</span>5<span>的平方暗含了两个</span>0<span>）</span></p>
<p><span>。。。依此类推</span></p>
<p><span>最后当</span>N&lt;5<span>了，结束。</span></p>
<p>&nbsp;</p>
<p><span>小小的验证一下：</span></p>
<p>26<span>！</span></p>
<p>26/5 = 5<span>，</span> 5/5 = 1<span>，那么最后</span>0<span>的个数就是</span>6<span>了。用</span>Google<span>算了一下，结果</span>G<span>大叔直接用有效数字表示了，</span>@$%$%@$%<span>。。。不过应该是没错了。恩。</span></p>
<p>Sigh<span>，知道结果后才知道原来这么简单的阿，偶土了。</span></p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/22306.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-19 15:30 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/19/22306.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>48 Ways to Wisdom - Way 4, Introduce Yourself to Yourself</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/18/22222.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Wed, 18 Apr 2007 08:22:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/18/22222.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/22222.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/18/22222.html#Feedback</comments><slash:comments>4</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/22222.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/22222.html</trackback:ping><description><![CDATA[<p><a name=ETFTOP><span>Did you ever get on a train going somewhere, only to find that you're headed in the wrong direction?</span></a></p>
<p><span><span>The same thing happens in life. We set goals and make plans - and sometimes discover that we're on "the wrong train."</span></span></p>
<p><span><em><span>Bi-vinat ha-lave</span></em></span><span><span> literally means "understanding the heart." The heart is the seat of emotions. We say: "My heart is heavy, my heart is lifted, my heart is broken," etc. To understand your heart is to understand your true inner self.</span></span></p>
<p><span><span>Many people go through life making assumptions about who they are. They never take time to "meet" themselves. Don't be afraid of discovering that the "real you" may be different than the "current you."</span></span></p>
<p><span><span>Often a crisis hits at midlife when people ask: "What's my life about? Is this all worth it?" We've heard stories of people who suddenly change direction, quitting their job and getting divorced. You know, like the successful doctor who decides he never wanted to go into medicine in the first place -- so he drops it and becomes an artist.</span></span></p>
<p><span><span>Knowing yourself is the essence of being alive. If you don't know yourself, you are not living. If you don't know what makes you tick, you're a robot, a puppet, a zombie.</span></span></p>
<p><span><span>So don't wait for a crisis. </span></span><span><strong><span>Life is too short to take wrong trains.</span></strong></span><span></span></p>
<p><strong><span>GETTING STARTED</span></strong></p>
<p><span>Think of someone you'd be fascinated to meet, someone you'd really like to find out what makes him tick.</span></p>
<p><span>Now realize the most fascinating person you could ever meet is ... yourself.</span></p>
<p><span>Sit down, say hello, and introduce yourself to yourself. Become familiar with yourself as if you'd just met a long-lost cousin. Interview yourself. Ask questions about your life and the direction you're going. Search out your dreams -- both the ones you're fulfilling and the ones you've pushed to the back of your mind.</span></p>
<p><span>Get down to basics. You want to be rich. You want to be famous. You want to be good. You want to accomplish. You want meaning. You want to be creative. But why do you want all this? What's driving you? What you really want out of life?</span></p>
<p><span>The process of self-discovery involves asking a series of questions, always probing deeper until the underlying truth emerges. Ask yourself 10 questions that you would ask an intimate friend. Then wait for answers. Don't worry, no one is going to poke fun at you.</span></p>
<ol type=1>
    <li><span>What is the <strong>purpose</strong> of life?</span></li>
    <li><span>What is my <strong>goal</strong> in life?</span></li>
    <li><span>Why did I choose this <strong>career</strong>?</span></li>
    <li><span>How do I spend my <strong>spare</strong> <strong>time</strong>?</span></li>
    <li><span>What is my <strong>motivation</strong> for doing what I do?</span></li>
    <li><span>What really makes me <strong>happy</strong>?</span></li>
    <li><strong><span>Am I as happy as I want to be</span></strong><span>?</span></li>
    <li><span>Is it more important to be <strong>rich</strong> or to be <strong>happy</strong>?</span></li>
    <li><span>What are my future <strong>plans</strong>? Why?</span></li>
    <li><span>What are my secret <strong>dreams</strong> and <strong>ambitions</strong>?</span></li>
</ol>
<p><span>Don't be surprised if the answers aren't immediate. This process can take many months. Stick with it and find out what makes you tick. The answers are hiding in there. After all, you have a fascinating partner.</span></p>
<p><span>Finally, the most important question to ask is:</span></p>
<p><strong><span>"What am I living for?"</span></strong></p>
<p><span>It sounds like a simple question, but many are embarrassed to ask it. A voice inside us says, "Nah, why ask such a basic question?" We're resistant because we know this requires a lot of difficult soul-searching. And when you thoroughly know yourself, then you have changed. You've changed your relationship with yourself and the world.</span></p>
<p><strong><span>CONFIDENCE IN DECISION-MAKING</span></strong></p>
<p><span>People often avoid making decisions out of fear of making a mistake.</span></p>
<p><span>Actually, the failure to make decisions is one of life's biggest mistakes.</span></p>
<p><span>Imagine the beggar who receives a letter saying that he's inherited a million dollars. If he doesn't read the letter, is he rich ... or not?</span></p>
<p><span>Similarly, God gave us the free will to make choices in life and achieve greatness. But if we're not aware of our free will, then we don't really have it. And then we wind up blaming others when things go wrong -- even though we know the decision is really up to us.</span></p>
<p><span>If you're not using your potential, it wears away at your confidence. Do you know what your potential is? Have you tried to use it? You have to tackle life. You haven't given up yet, have you? Let's get on with the game, with the business of really living, of not just "going through the motions."</span></p>
<p><span>Know the difference between "making decisions" and just floating, falling into place. Did you choose to go to college? Or perhaps you had nothing to do with the decision. Was it something you just did because you graduated high school and everybody else was doing it? Did you think it through and actually make a decision?</span></p>
<p><span>Imagine this private conversation of a college student:</span></p>
<p><em><span>Why am I going to college?<br>To get a degree.<br>Why?<br>Because I want to get into a good graduate school.<br>Why?<br>So I'll get a good job.<br>Why?<br>So I can pay back my college loans!</span></em></p>
<p><span>Through the process of questioning, he reveals a logical fault in his motivation. Really, the primary reason for going to college should be to acquire wisdom, knowledge and information. In other words, to get an education!</span></p>
<p><span>Now try the process yourself, using this example:</span></p>
<p><em><span>Why do I want to get married?</span></em></p>
<p><span>Don't accept pat answers. Keep asking "Why, why why?" Be frank. It's yourself. Ask any question you like. Be patient and persistent. Eventually you'll get an answer.</span></p>
<p><span>When you thoroughly analyze an issue, then you can make wise decisions with confidence.</span></p>
<p><span>Identify where you lack confidence. What makes you nervous? What situations inhibit you from being yourself? Why can't you make decisions? Is it that you don't know how to make decisions? Or that you doubt your decisions after they're made? Or you just don't feel like making decisions?</span></p>
<p><span>Enjoy making decisions. Deal with the world you live in. That's loving the dynamics of life.</span></p>
<p><strong><span>ISOLATE YOUR BLOCKS</span></strong></p>
<p><span>Anytime you find it difficult to achieve a goal, figure out what's holding you back.</span></p>
<p><span>Everyone has problems. Being aware of these problems is the key to getting in touch with yourself. Because as long as you don't face problems, they fester and bug you from behind.</span></p>
<p><span>Write your "blocks" on a piece of paper. That's a good step in the right direction. By isolating specific obstacles, you turn them into concrete challenges that require solutions.</span></p>
<p><span>Ask yourself:</span></p>
<ul type=disc>
    <li><span>Am I lazy? Why?</span></li>
    <li><span>Am I disorganized? Why?</span></li>
    <li><span>Do I get angry? When?</span></li>
    <li><span>Why do ever I get defensive? About what?</span></li>
    <li><span>What makes me jealous?</span></li>
    <li><span>What makes me arrogant?</span></li>
    <li><span>Do I have trouble making decisions? Why?</span></li>
    <li><span>Do I lack self-discipline?</span></li>
    <li><span>Do I lack self-confidence?</span></li>
    <li><span>Why don't I take more initiative?</span></li>
</ul>
<p><span>Negative character traits are the roots of our problems. Make a list of your negative traits, and identify when they affect you the most. Then analyze what triggers these reactions in you. Finally, formulate an effective counter-approach.</span></p>
<p><span>Working through this takes time. But do you have anything better to be doing right now?</span></p>
<p><strong><span>READ YOUR EMOTIONS</span></strong></p>
<p><span>Get in touch with your emotional state. Take a reading of how you feel. Happy? Angry? Tense? Sad? Emotions are a measuring stick for what's going on below the surface. It's like taking your temperature. If you're sick, you need to be aware so you can fix the problem.</span></p>
<p><span>Find out why you're upset. Who or what is pressuring you? Is it an internal or an external problem? Identify it.</span></p>
<p><span>Let's say you are irritated. Why?</span></p>
<p><em><span>Because the boss chewed me out.<br>So why am I irritated?<br>Because I resent him.<br>So what? Why does that bother me?<br>Because I feel I am no good.<br>I'm no good? He's nuts!</span></em></p>
<p><span>Get out of yourself and track it down. If you don't, it's just irritation. And the next thing you know, you'll go home and yell at your kids.</span></p>
<p><span>Once you've identified what causes negative feelings, adjust yourself to minimize the impact. Either avoid these situations, or prepare yourself to handle them when they arise.</span></p>
<p><span>Further, root out negative motivations that corrupt your behavior. Let's say that you give charity. Why? One motivation is to help humanity. Another is the pleasure of being constructive. A third is the desire to do the right thing. These are all positive motivations. A negative motivation for giving charity is: "I want people to admire me." That's corruptive.</span></p>
<p><span>The next time you give charity, do so anonymously. Eliminate the wrong reasons. They are destructive.</span></p>
<p><span>The same goes with the positive emotions. Be aware of how your emotional state affects decisions. For example, don't buy a new stereo when you're in a euphoric mood. Wait. Think it over. You are susceptible.</span></p>
<p><span>Pinpoint what makes you happy. You can have more joy on a daily basis by formulating some practical applications. You got up in the morning, it's a gorgeous day and you feet great. You're energized. Now take that feeling and teach yourself how to get up on the right side -- every day!</span></p>
<p><span>Another example: You did a good job and got the boss's compliment. Now focus: Do you need the boss to tell you did a good job? No! Create your own pleasure out of doing a good job.</span></p>
<p><strong><span>GET IN TOUCH WITH YOUR TWO SIDES</span></strong></p>
<p><span>Everyone has an urge for greatness. We want self respect, power, fame. We want to accomplish, to be strong, to do the right thing, to even save the world.</span></p>
<p><span>Yet at the same time, we have a counter-urge to run away from responsibility, to get into bed and crawl under the covers.</span></p>
<p><span>Someone may say, "Life is beautiful," but he doesn't feel it. His emotions hold him back and he walks around going, "Ugh, life is a burden."</span></p>
<p><span>Recognize the volcano of conflict within you: What you truly "want," versus what you "feel" like. This is the conflict between body and soul.</span></p>
<p><span>Once you appreciate the dichotomy, you can identify at any moment whether your body or soul is talking. This makes it possible to live with sanity and choose the right thing.</span></p>
<p><span>The next step is to make peace between your two sides. The easiest way is to squash your drive to be great. But life is not about taking the easy way out. Just because you feel uncomfortable about an idea doesn't mean it's wrong for you. It's hard to break habits, and growth can be frightening.</span></p>
<p><span>For example, would you rather be happy or rich? Okay, you'd rather be happy. Now imagine this exchange:</span></p>
<p><span>"Come on, I'll teach you how to be happy. All it requires is effort and change."</span></p>
<p><span>"Oh, I'd love to, but I can't right now. It's impossible. I've got a flight to catch."</span></p>
<p><span>"Really? I'll pay you $10,000 a week to work on happiness."</span></p>
<p><span>"Sure! Where do I sign up?"</span></p>
<p><span>"Oh, but I thought you can't right now..."</span></p>
<p><span>We conceal our problems with rationalization: "I'll wreck my mind thinking about what life is about! Nobody really knows what life is about. It's not going to work. Nothing can be done about it anyway. I don't really care. It's not worth the time!"</span></p>
<p><span>The Sages say that a person only makes a mistake when overcome by a moment of insanity. So realize that you are fighting "insanity." It is not logical. You've got to be on guard. Because if you get off track, you'll pay for it down the road.</span></p>
<p><span>So ... do you want to change? What have you got against it? Feel the antipathy of the body. We are so darn lazy. The body just wants to sleep. "Aaaah ... I don't want to change. I'm happy enough. I'm comfortable in my niche of misery." Are you rich enough? No! So are you happy enough?</span></p>
<p><span>You see the importance of tracking that down? You have to identify the animal you are fighting. "The dread of change."</span></p>
<p><span>If you're alert, you see the enemy. You can fight it. You may lose a struggle with the body, but at least you have your confidence. "I know what I am doing."</span></p>
<p><strong><span>COAX THE BODY</span></strong></p>
<p><span>Get in touch with your spiritual core. Know what is driving you. Don't let free will be a subconscious thing. You want greatness. But the body says that's too much effort.</span></p>
<p><span>To try to convince the body, try to identify the tangible benefit. "Why is it necessary? What will it do for me?" You have to bring it home to emotional realization. "What do I lose?" What do I gain?" Only then will the idea have power. And you'll get out there and do it.</span></p>
<p><span>Here's the secret formula: Identify with your intellect, and coax your heart along. For example, if you're emotionally convinced of the benefit of getting into shape, then even when you break out in a cold sweat and your heart is doing palpitations, you will keep going. Because you have decided, "I want this," you know it is important.</span></p>
<p><span>To avoid negative backlash, your emotions have to feel comfortable with the changes you make. Learn to relax and reassure the body. Cajole the body and say, "It won't be so bad. Remember the last time you made an effort, how great you felt!" Be encouraging and reward yourself for success.</span></p>
<p><span>Don't say it doesn't work. You haven't made the effort. Don't give up on your intuition and perception. Just realize you haven't yet brought it home to actualization.</span></p>
<p><span>Consider how the basic human drives affect you: security, self-respect, honor, passions, social pressure, and possessions. Pay particularly close attention to how you accept responsibility. Let's say that you made a mistake. You want to apologize in a full and forthright manner. Yet you feel like forgetting the whole thing, hiding, running away and saying "it's not my fault."</span></p>
<p><span>This is the volcano. We want to be tough, dedicated and powerful -- yet we feel like being marshmallows. Choosing the path of the soul doesn't come naturally. It takes a lot of time and hard work.</span></p>
<p><strong><span>KNOW WHAT YOU KNOW</span></strong></p>
<p><span>Don't think that just because you understand something, you are living with it. It is possible to believe one way, and yet act another. It happens to us all the time. You can believe it's important to eat healthy food, yet gorge yourself on French fries and chocolate cake.</span></p>
<p><span>Our actions are determined by our level of clarity. If we understand an idea on just a superficial level, then we'll have difficulty sticking to it when the going gets tough.</span></p>
<p><span>Next time you go to a funeral, watch carefully. When they remove the body from the chapel, the mourners start to cry. Are they crying because they want to body to stay there?! No. All of a sudden there is a realization of death, that he won't be coming back. At the cemetery, they lower the casket into the ground and the mourners cry again. It's the emotional realization that death is final now.</span></p>
<p><span>Until you align your feelings with reality, you are in dreamland. Growth begins in the mind, but your heart has to buy into everything your mind discovers. Only then will you integrate these ideas into day-to-day life.</span></p>
<p><span>A lot of people believe in God. There are very few people who live with God. Does that make sense? You have to assimilate something that you've accepted as true. It has to become part of you.</span></p>
<p><strong><span>FIVE-FINGER CLARITY</span></strong></p>
<p><span>You've got to know yourself cold, just like you know your hand has five fingers. How do you know you are on the right path? How do you know you're not making a mistake right now?</span></p>
<p><span>To develop this clarity, articulate the important principles that guide your life. For example, in Judaism we say that love is an obligation. Is this reasonable? Work the issue through with yourself:</span></p>
<p><em><span>"Ridiculous. You can't obligate me to love."<br>"But if I have children, will I love them?"<br>"Of course I'm going to love my kids!"<br>"How do I know? I don't know what kind of kids I'm going to have. Maybe they'll be brats and I won't love them."<br>"I will. I'm obligated to love my children."</span></em></p>
<p><span>Do you see the contradiction? On an intuitive level, you know that love is an obligation. But the concept is not so clear that you can articulate it.</span></p>
<p><span>Take your time. Sort out the basic aspects of living. Ask yourself important questions about life's global and spiritual issues.</span></p>
<p><span>--?What is the meaning of existence?<br>--?What's good about living?<br>--?How do I feel about humanity?<br>--?What is the afterlife?<br>--?How do I understand good versus evil?<br>--?Do I have free will? How do I activate it?<br>--?What makes me sad? Is it okay to be sad?<br>--?How do I feel about God?<br>--?Am I proud to be a Jew?<br>--?How do I understand the Holocaust?</span></p>
<p><span>Some of these topics may be unpleasant to think about. If so, why is it unpleasant? Track it down.</span></p>
<p><span>Don't just use slogans to parrot things that you heard. Know why you are doing what you are doing. Otherwise, it's just society talking. You may have adopted part of society without analyzing its validity. Check it out.</span></p>
<p><span>Work through all the issues until you have "five-finger clarity." A human being who knows what he wants will get there. By hook or by crook. It's like a homing mechanism on a missile. If you program it right, you will get there.</span></p>
<p><strong><span>WHY IS "KNOWING YOURSELF" AN INGREDIENT IN WISDOM?</span></strong></p>
<ul type=disc>
    <li><span>You can know truth if you look honestly into yourself. </span></li>
    <li><span>Emotions are powerful forces of greatness. Know them. Harness them. </span></li>
    <li><span>Identify your problems. It's the beginning of solving them. </span></li>
    <li><span>If you don't get it straight now, you're bound to make some bad mistakes. </span></li>
    <li><span>Don't be afraid of finding out who you really are. </span></li>
    <li><span>Use your free will as a conscious tool for better living. </span></li>
    <li><span>If you're angry or upset, track it down. What's the root? </span></li>
    <li><span>If you're acting illogically, at least acknowledge that to yourself! </span></li>
    <li><span>The key to sanity is letting truth into the body. </span></li>
    <li><span>You can't afford to wait too long to get to know yourself. Because you are the most fascinating person you'll ever meet.</span></li>
</ul>
<p>&nbsp;</p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/22222.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-18 16:22 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/18/22222.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>动态语言</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/16/22034.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Mon, 16 Apr 2007 10:15:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/16/22034.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/22034.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/16/22034.html#Feedback</comments><slash:comments>1</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/22034.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/22034.html</trackback:ping><description><![CDATA[<p><span>所谓动态语言，也叫脚本语言，就是说<strong>一种在执行期间才去发现数据类型的程序设计语言，主要创建一些需要</strong></span><strong><span>经常更新的动态系统。</span></strong><span>常见的有</span><span>Python, Lua, Perl, PHP<span>等。</span><strong></strong></span></p>
<p><span>以往，我们所使用的语言，比如</span><span>C<span>，</span>C++<span>等等，都称为静态语言。什么是静态语言呢？就是说，在使用数据之前，我们必须首先定义数据类型，这些数据类型包括</span>int, float, double<span>等等。就相当于在使用它们之前，首先要为它们分配好内存空间；而动态语言就刚刚是相反的，它是在得到数据类型之后，再为它分配内存空间。</span></span></p>
<p><span>&#8220;脚本语言除了接近口语化的设计外，另加上简化后的语法。（除了内建的命令外，通常仅需简单的逻辑判断与数值计算即可胜任）因此用脚本语言制作游戏，不再是非程序员不可的工作（除了系统本身的修订），企画人员也可以很快地进入状态。另外，如果将来需要将游戏移植到其他平台时，比起程序与资料的盘根错节的设计，利用脚本语言来开发的游戏，只需要修改系统本身，脚本语言部分本身毋须更动，相形之下出现问题的机会与范围就缩小了很多。</span></p>
<img src ="http://www.cppblog.com/jacky2019/aggbug/22034.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-16 18:15 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/16/22034.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>名言</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/13/21808.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Fri, 13 Apr 2007 08:42:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/13/21808.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/21808.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/13/21808.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/21808.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/21808.html</trackback:ping><description><![CDATA[C++名人Alexandrescu说，十几岁的少年天才到处都有，三十多岁的优秀设计师凤毛麟角，掌握一种力量是容易的，学会恰当地使用这种力量却难得多，这就是聪明与智慧之间的差别。文武之道，有张有弛，知道什么是该做的，什么是不该做的，这是人生最大的学问。
<img src ="http://www.cppblog.com/jacky2019/aggbug/21808.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-13 16:42 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/13/21808.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>Intel.VTune.Performance.Analyzer.v8.0.014.ISO下载</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/13/21797.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Fri, 13 Apr 2007 08:07:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/13/21797.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/21797.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/13/21797.html#Feedback</comments><slash:comments>2</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/21797.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/21797.html</trackback:ping><description><![CDATA[<a href="http://down.gogobox.com.tw/kisser/7ir3">http://down.gogobox.com.tw/kisser/7ir3</a><br>下载的时候要安装一个ActiveX插件的。<br>
<img src ="http://www.cppblog.com/jacky2019/aggbug/21797.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-13 16:07 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/13/21797.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>Finding crash information using the MAP file 2</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/10/21584.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Tue, 10 Apr 2007 03:27:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/10/21584.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/21584.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/10/21584.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/21584.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/21584.html</trackback:ping><description><![CDATA[很奇怪，cppblog居然不能发表评论了，哭。。。所以就在发一篇了哦，呵呵。<br><a id=_4f0e0710e665_HomePageDays_DaysList_ctl01_DayItem_DayList_ctl03_TitleUrl href="http://www.cppblog.com/jacky2019/archive/2007/04/09/21527.html"><u><font color=#810081>Finding crash information using the MAP file</font></u></a><br><a href="http://www.cppblog.com/jacky2019/archive/2007/04/09/21527.html">http://www.cppblog.com/jacky2019/archive/2007/04/09/21527.html</a><br><span>上文中的关于小说明的第</span><span>2<span>点是有问题的</span></span>
<p><span>偶又试了一次，发现</span><span>lines number<span>可能是有问题的，而且也不是我的那种算法。不过函数名肯定是对的<br><br>在vckbase中又发现一篇好文，所以就又贴过来了。不过可惜的是，还是没能解决dll的地址映射问题，无法track到dll内的信息，可惜了。<br><br></p>
<p align=center><strong>对&#8220;仅通过崩溃地址找出源代码的出错行&#8221;一文的补充与改进</strong><br><br>作者：<a href="&#109;&#97;&#105;&#108;&#116;&#111;&#58;&#116;&#120;&#113;&#99;&#52;&#64;&#115;&#111;&#104;&#117;&#46;&#99;&#111;&#109;"><font color=#0000ff><u>上海伟功通信 roc</u></font></a></p>
<p><a href="http://www.vckbase.com/code/downcode.asp?id=2691"><u><font color=#0000ff>下载源代码</font></u></a><br><br>　　读了老罗的&#8220;<a href="http://www.vckbase.com/document/viewdoc/?id=908" target=_blank><u><font color=#810081>仅通过崩溃地址找出源代码的出错行</font></u></a>&#8221;(下称"罗文")一文后，感觉该文还是可以学到不少东西的。不过文中尚存在有些说法不妥，以及有些操作太繁琐的地方 。为此，本人在学习了此文后，在多次实验实践基础上，把该文中的一些内容进行补充与改进，希望对大家调试程序，尤其是release版本的程序有帮助 。欢迎各位朋友批评指正。<br><br><br><strong>一、该方法适用的范围</strong><br>　　在windows程序中造成程序崩溃的原因很多，而文中所述的方法仅适用与:由一条语句当即引起的程序崩溃。如原文中举的除数为零的崩溃例子。而笔者在实际工作中碰到更多的情况是:指针指向一非法地址 ，然后对指针的内容进行了，读或写的操作。例如：<br></p>
<pre>void Crash1()
{
char * p =(char*)100;
*p=100;
}<br></pre>
<p>　　这些原因造成的崩溃，无论是debug版本，还是release版本的程序，使用该方法都可找到造成崩溃的函数或子程序中的语句行，具体方法的下面还会补充说明。 另外，实践中另一种常见的造成程序崩溃的原因:函数或子程序中局部变量数组越界付值，造成函数或子程序返回地址遭覆盖，从而造成函数或子程序返回时崩溃。例如:<br></p>
<pre>#include <string.h>
void Crash2();
int main(int argc,char* argv[])
{
Crash2();
return 0;
}
void Crash2()
{
char p[1];
strcpy(p,"0123456789");
}<br></pre>
<p>在vc中编译运行此程序的release版本，会跳出如下的出错提示框。 <br><br><img height=139 src="http://www.vckbase.com/document/journal/vckbase42/images/crashimg1.jpg" width=434 border=0><br>图一 上面例子运行结果<br><br>　　这里显示的崩溃地址为:0x34333231。这种由前面语句造成的崩溃根源，在后续程序中方才显现出来的情况，显然用该文所述的方法就无能为力了。不过在此例中多少还有些蛛丝马迹可寻找到崩溃的原因:函数Crash2中的局部数组p只有一个字节大小 ，显然拷贝"0123456789"这个字符串会把超出长度的字符串拷贝到数组p的后面，即*(p+1)=''1''，*(p+2)=''2''，*(p+3)=''3''，*(p+4)=4。。。。。。而字符''1''的ASC码的值为0x31，''2''为0x32，''3''为0x33，''4''为0x34。。。。。，由于intel的cpu中int型数据是低字节保存在低地址中 ，所以保存字符串''1234''的内存，显示为一个4字节的int型数时就是0x34333231。显然拷贝"0123456789"这个字符串时，"1234"这几个字符把函数Crash2的返回地址给覆盖 ，从而造成程序崩溃。对于类似的这种造成程序崩溃的错误朋友们还有其他方法排错的话，欢迎一起交流讨论。<br><br><br><strong>二、设置编译产生map文件的方法</strong><br>　　该文中产生map文件的方法是手工添加编译参数来产生map文件。其实在vc6的IDE中有产生map文件的配置选项的。操作如下:先点击菜单"Project"-&gt;"Settings。。。"，弹出的属性页中选中"Link"页 ，确保在"category"中选中"General"，最后选中"Generate mapfile"的可选项。若要在在map文件中显示Line numbers的信息的话 ，还需在project options 中加入/mapinfo:lines 。Line numbers信息对于"罗文"所用的方法来定位出错源代码行很重要 ，但笔者后面会介绍更加好的方法来定位出错代码行，那种方法不需要Line numbers信息。 <br><br><img height=446 src="http://www.vckbase.com/document/journal/vckbase42/images/crashimg3.JPG" width=677 border=0><br>图二 设置产生MAP文件 <br><br><br><strong>三、定位崩溃语句位置的方法</strong><br>　　"罗文"所述的定位方法中，找到产生崩溃的函数位置的方法是正确的，即在map文件列出的每个函数的起始地址中，最近的且不大于崩溃地址的地址即为包含崩溃语句的函数的地址 。但之后的再进一步的定位出错语句行的方法不是最妥当，因为那种方法前提是，假设基地址的值是 0x00400000 ，以及一般的 PE 文件的代码段都是从 0x1000偏移开始的 。虽然这种情况很普遍，但在vc中还是可以基地址设置为其他数，比如设置为0x00500000，这时仍旧套用<br></p>
<pre> 崩溃行偏移 = 崩溃地址 - 0x00400000 - 0x1000 </pre>
<p>的公式显然无法找到崩溃行偏移。 其实上述公式若改为<br></p>
<pre>崩溃行偏移 = 崩溃地址 - 崩溃函数绝对地址 + 函数相对偏移</pre>
<p>即可通用了。仍以"罗文"中的例子为例:"罗文"中提到的在其崩溃程序的对应map文件中，崩溃函数的编译结果为<br></p>
<pre>0001:00000020 ?Crash@@YAXXZ 00401020 f CrashDemo。obj </pre>
<p>对与上述结果，在使用我的公式时 ，"崩溃函数绝对地址"指00401020， 函数相对偏移指 00000020， 当崩溃地址= 0x0040104a时， 则 崩溃行偏移 = 崩溃地址 - 崩溃函数起始地址+ 函数相对偏移 = 0x0040104a - 0x00401020 + 0x00000020= 0x4a，结果与"罗文"计算结果相同 。但这个公式更通用。<br><br><br><strong>四、更好的定位崩溃语句位置的方法。</strong><br>　　其实除了依靠map文件中的Line numbers信息最终定位出错语句行外，在vc6中我们还可以通过编译程序产生的对应的汇编语句，二进制码，以及对应c/c++语句为一体的"cod"文件来定位出错语句行 。先介绍一下产生这种包含了三种信息的"cod"文件的设置方法:先点击菜单"Project"-&gt;"Settings。。。"，弹出的属性页中选中"C/C++"页 ，然后在"Category"中选则"Listing Files"，再在"Listing file type"的组合框中选择"Assembly，Machine code， and source"。接下去再通过一个具体的例子来说明这种方法的具体操作。 <br><br><img height=446 src="http://www.vckbase.com/document/journal/vckbase42/images/crashimg4.JPG" width=677 border=0><br>图三 设置产生"cod"文件 <br><br>准备步骤1)产生崩溃的程序如下:<br></p>
<pre>01 //****************************************************************
02 //文件名称：crash。cpp
03 //作用:    演示通过崩溃地址找出源代码的出错行新方法
04 //作者：   伟功通信 roc
05 //日期：   2005-5-16
06//****************************************************************
07 void Crash1();
08 int main(int argc,char* argv[])
09 {
10 Crash1();
11 return 0;
12 }
13
14 void Crash1()
15 {
16  char * p =(char*)100;
17  *p=100;
18 }
</pre>
<p>准备步骤2)按本文所述设置产生map文件(不需要产生Line numbers信息)。<br>准备步骤3)按本文所述设置产生cod文件。<br>准备步骤4)编译。这里以debug版本为例(若是release版本需要将编译选项改为不进行任何优化的选项，否则上述代码会因为优化时看作废代码而不被编译，从而看不到崩溃的结果)，编译后产生一个"exe"文件 ，一个"map"文件，一个"cod"文件。 <br>运行此程序，产生如下如下崩溃提示: <br><br><img height=139 src="http://www.vckbase.com/document/journal/vckbase42/images/crashimg2.JPG" width=434 border=0><br>图四 上面例子运行结果 <br><br>排错步骤1)定位崩溃函数。可以查询map文件获得。我的机器编译产生的map文件的部分如下:<br></p>
<pre> Crash
Timestamp is 42881a01 (Mon May 16 11:56:49 2005)
Preferred load address is 00400000
Start Length Name Class
0001:00000000 0000ddf1H .text CODE
0001:0000ddf1 0001000fH .textbss CODE
0002:00000000 00001346H .rdata DATA
0002:00001346 00000000H .edata DATA
0003:00000000 00000104H .CRT$XCA DATA
0003:00000104 00000104H .CRT$XCZ DATA
0003:00000208 00000104H .CRT$XIA DATA
0003:0000030c 00000109H .CRT$XIC DATA
0003:00000418 00000104H .CRT$XIZ DATA
0003:0000051c 00000104H .CRT$XPA DATA
0003:00000620 00000104H .CRT$XPX DATA
0003:00000724 00000104H .CRT$XPZ DATA
0003:00000828 00000104H .CRT$XTA DATA
0003:0000092c 00000104H .CRT$XTZ DATA
0003:00000a30 00000b93H .data DATA
0003:000015c4 00001974H .bss DATA
0004:00000000 00000014H .idata$2 DATA
0004:00000014 00000014H .idata$3 DATA
0004:00000028 00000110H .idata$4 DATA
0004:00000138 00000110H .idata$5 DATA
0004:00000248 000004afH .idata$6 DATA
Address Publics by Value Rva+Base Lib:Object
0001:00000020 _main 00401020 f Crash.obj
0001:00000060 ?Crash1@@YAXXZ 00401060 f Crash.obj
0001:000000a0 __chkesp 004010a0 f LIBCD:chkesp.obj
0001:000000e0 _mainCRTStartup 004010e0 f LIBCD:crt0.obj
0001:00000210 __amsg_exit 00401210 f LIBCD:crt0.obj
0001:00000270 __CrtDbgBreak 00401270 f LIBCD:dbgrpt.obj
...
</pre>
<p>对于崩溃地址0x00401082而言，小于此地址中最接近的地址(Rva+Base中的地址)为00401060，其对应的函数名为?Crash1@@YAXXZ，由于所有以问号开头的函数名称都是 C++ 修饰的名称 ，"@@YAXXZ"则为区别重载函数而加的后缀，所以?Crash1@@YAXXZ就是我们的源程序中，Crash1() 这个函数。<br>排错步骤2)定位出错行。打开编译生成的"cod"文件，我机器上生成的文件内容如下:<br></p>
<pre> TITLE E:\Crash\Crash。cpp
.386P
include listing.inc
if @Version gt 510
.model FLAT
else
_TEXT SEGMENT PARA USE32 PUBLIC ''CODE''
_TEXT ENDS
_DATA SEGMENT DWORD USE32 PUBLIC ''DATA''
_DATA ENDS
CONST SEGMENT DWORD USE32 PUBLIC ''CONST''
CONST ENDS
_BSS SEGMENT DWORD USE32 PUBLIC ''BSS''
_BSS ENDS
$$SYMBOLS SEGMENT BYTE USE32 ''DEBSYM''
$$SYMBOLS ENDS
$$TYPES SEGMENT BYTE USE32 ''DEBTYP''
$$TYPES ENDS
_TLS SEGMENT DWORD USE32 PUBLIC ''TLS''
_TLS ENDS
; COMDAT _main
_TEXT SEGMENT PARA USE32 PUBLIC ''CODE''
_TEXT ENDS
; COMDAT ?Crash1@@YAXXZ
_TEXT SEGMENT PARA USE32 PUBLIC ''CODE''
_TEXT ENDS
FLAT GROUP _DATA， CONST, _BSS
ASSUME CS: FLAT, DS: FLAT, SS: FLAT
endif
PUBLIC ?Crash1@@YAXXZ     ; Crash1
PUBLIC _main
EXTRN __chkesp:NEAR
; COMDAT _main
_TEXT SEGMENT
_main PROC NEAR     ; COMDAT
; 9    : {
00000 55   push  ebp
00001 8b ec   mov  ebp， esp
00003 83 ec 40  sub  esp, 64   ; 00000040H
00006 53   push  ebx
00007 56   push  esi
00008 57   push  edi
00009 8d 7d c0  lea  edi, DWORD PTR [ebp-64]
0000c b9 10 00 00 00  mov  ecx， 16   ; 00000010H
00011 b8 cc cc cc cc  mov  eax， -858993460  ; ccccccccH
00016 f3 ab   rep stosd
; 10   :  Crash1();
00018 e8 00 00 00 00  call  ?Crash1@@YAXXZ  ; Crash1
; 11   :  return 0;
0001d 33 c0   xor  eax， eax
; 12   : }
0001f 5f   pop  edi
00020 5e   pop  esi
00021 5b   pop  ebx
00022 83 c4 40  add  esp, 64   ; 00000040H
00025 3b ec   cmp  ebp, esp
00027 e8 00 00 00 00  call  __chkesp
0002c 8b e5   mov  esp, ebp
0002e 5d   pop  ebp
0002f c3   ret  0
_main ENDP
_TEXT ENDS
; COMDAT ?Crash1@@YAXXZ
_TEXT SEGMENT
_p$ = -4
?Crash1@@YAXXZ PROC NEAR    ; Crash1, COMDAT
; 15   : {
00000 55   push  ebp
00001 8b ec   mov  ebp, esp
00003 83 ec 44  sub  esp, 68   ; 00000044H
00006 53   push  ebx
00007 56   push  esi
00008 57   push  edi
00009 8d 7d bc  lea  edi, DWORD PTR [ebp-68]
0000c b9 11 00 00 00  mov  ecx, 17   ; 00000011H
00011 b8 cc cc cc cc  mov  eax, -858993460  ; ccccccccH
00016 f3 ab   rep stosd
; 16   :  char * p =(char*)100;
00018 c7 45 fc 64 00
00 00   mov  DWORD PTR _p$[ebp], 100 ; 00000064H
; 17   :  *p=100;
0001f 8b 45 fc  mov  eax, DWORD PTR _p$[ebp]
00022 c6 00 64  mov  BYTE PTR [eax], 100 ; 00000064H
; 18   : }
00025 5f   pop  edi
00026 5e   pop  esi
00027 5b   pop  ebx
00028 8b e5   mov  esp, ebp
0002a 5d   pop  ebp
0002b c3   ret  0
?Crash1@@YAXXZ ENDP     ; Crash1
_TEXT ENDS
END
</pre>
<p>其中<br></p>
<pre>?Crash1@@YAXXZ PROC NEAR    ; Crash1, COMDAT</pre>
<p>为Crash1汇编代码的起始行。产生崩溃的代码便在其后的某个位置。接下去的一行为: <br></p>
<pre>; 15   : {</pre>
<p>冒号后的"{"表示源文件中的语句，冒号前的"15"表示该语句在源文件中的行数。 这之后显示该语句汇编后的偏移地址，二进制码，汇编代码。如 <br></p>
<pre>00000 55   push  ebp</pre>
<p>其中"0000"表示相对于函数开始地址后的偏移，"55"为编译后的机器代码，" push ebp"为汇编代码。从"cod"文件中我们可以看出，一条(c/c++)语句通常需要编译成数条汇编语句 。此外有些汇编语句太长则会分两行显示如: <br></p>
<pre>00018 c7 45 fc 64 00
00 00   mov  DWORD PTR _p$[ebp], 100 ; 00000064H</pre>
<p>其中"0018"表示相对偏移，在debug版本中，这个数据为相对于函数起始地址的偏移(此时每个函数第一条语句相对偏移为0000)；release版本中为相对于代码段第一条语句的偏移(即代码段第一条语句相对偏移为0000，而以后的每个函数第一条语句相对偏移就不为0000了)。"c7 45 fc 64 00 00 00 "为编译后的机器代码 ，"mov DWORD PTR _p$[ebp]， 100"为汇编代码， 汇编语言中";"后的内容为注释，所以";00000064H"，是个注释这里用来说明100转换成16进制时为"00000064H"。<br>接下去，我们开始来定位产生崩溃的语句。<br>第一步，计算崩溃地址相对于崩溃函数的偏移，在本例中已经知道了崩溃语句的地址(0x00401082)，和对应函数的起始地址(0x00401060)，所以崩溃地址相对函数起始地址的偏移就很容易计算了: <br></p>
<pre>  崩溃偏移地址 = 崩溃语句地址 - 崩溃函数的起始地址 = 0x00401082 - 0x00401060 = 0x22。</pre>
<p>第二步，计算出错的汇编语句在cod文件中的相对偏移。我们可以看到函数Crash1()在cod文件中的相对偏移地址为0000，则 <br></p>
<pre>崩溃语句在cod文件中的相对偏移 =  崩溃函数在cod文件中相对偏移 + 崩溃偏移地址 = 0x0000 + 0x22 = 0x22</pre>
<p>第三步，我们看Crash1函数偏移0x22除的代码是什么?结果如下 <br></p>
<pre> 00022 c6 00 64  mov  BYTE PTR [eax], 100 ; 00000064H</pre>
<p>这句汇编语句表示将100这个数保存到寄存器eax所指的内存单元中去，保存空间大小为1个字节(byte)。程序正是执行这条命令时产生了崩溃，显然这里eax中的为一个非法地址 ，所以程序崩溃了!<br>第四步，再查看该汇编语句在其前面几行的其对应的源代码，结果如下: <br></p>
<pre>; 17   :  *p=100;</pre>
<p>其中17表示该语句位于源文件中第17行，而&#8220;*p=100;&#8221;这正是源文件中产生崩溃的语句。<br>至此我们仅从崩溃地址就查找出了造成崩溃的源代码语句和该语句所在源文件中的确切位置，甚至查找到了造成崩溃的编译后的确切汇编代码!<br>怎么样，是不是感觉更爽啊?<br><br><br><strong>五、小节</strong><br><br>1、新方法同样要注意可以适用的范围，即程序由一条语句当即引起的崩溃。另外我不知道除了VC6外，是否还有其他的编译器能够产生类似的"cod"文件。<br>2、我们可以通过比较 新方法产生的debug和releae版本的"cod"文件，查找那些仅release版本(或debug版本)有另一个版本没有的bug(或其他性状)。例如"罗文"中所举的那个用例 ，只要打开release版本的"cod"文件，就明白了为啥debug版本会产生崩溃而release版本却没有:原来release版本中产生崩溃的语句其实根本都没有编译 。同样本例中的release版本要看到崩溃的效果，需要将编译选项改为为不优化的配置。</p><img src ="http://www.cppblog.com/jacky2019/aggbug/21584.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-10 11:27 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/10/21584.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>DeinoMPI</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/10/21578.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Tue, 10 Apr 2007 02:57:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/10/21578.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/21578.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/10/21578.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/21578.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/21578.html</trackback:ping><description><![CDATA[<p>DeinoMPI is an implementation of MPI-2 for Microsoft Windows.&nbsp;</p>
<p>DeinoMPI<span>是微软</span>windows<span>下</span>MPI-2<span>的一个实现。</span></p>
<p><span>注：</span>MPI-2<span>是</span>Message Passing Interface, <a href="http://www.mpi-forum.org/">http://www.mpi-forum.org/</a><span>。</span>What is MPI? MPI is a library of functions and macros which is intended for use in programs that exploit the existence of multiple processors by message-passing. <span>也就是说，</span>MPI<span>是一套规范，定义了一些函数和宏，它被用来在程序中通过消息传递来发挥多处理器的处理能力。</span></p>
<p>DeinoMPI is an implementation of the MPI-2 standard for parallel computing or more generally it is system level middle-ware for high performance parallel computing on Microsoft Windows systems. It supports Win32 and Win64 machines.</p>
<p>DeinoMPI<span>是</span>MPI-2<span>的一个实现，</span>MPI-2<span>是</span>for<span>并行计算的。更通用地说，</span>DeinoMPI<span>是</span>windows<span>系统上的一个高性能并行计算的系统</span>level<span>中间件。它有两个版本，分别是</span>for win32<span>和</span>for win64<span>的。</span></p>
<p>1) DeinoMPI provides the libraries necessary for software developers to write parallel applications that conform to the <a href="http://www.mpi-forum.org/">MPI-2</a> standard for parallel computing.&nbsp; Software developers who wish to develop new parallel applications or wish to add parallel capabilities to existing software would benefit from using DeinoMPI.&nbsp; The libraries provided support a wide range of C, C++ and Fortran compilers.</p>
<p>2) DeinoMPI also provides a process manager used to start processes on multiple machines in a cluster remotely.&nbsp; Microsoft Windows does not provide native capability to start user applications on remote machines.&nbsp; DeinoMPI provides a secure mechanism to do just that.&nbsp; The primary purpose of the DeinoMPI process manager is to set up the environment and launch processes used in MPI jobs but it is not restricted to this functionality.&nbsp; It can launch any application remotely on behalf of the user.</p>
<p>The following users will benefit from DeinoMPI: Businesses that develop software and wish to add parallel capabilities to increase the performance of their software.&nbsp; Research institutions that have parallel software codes that they want to run on their Windows machines.&nbsp; Universities or other educational institutions that teach courses on parallel computing.&nbsp; University or research labs that run computationally intensive software and wish to use parallel versions of their software to better utilize their computer resources.&nbsp; Students or hobbyists that wish to write parallel programs.</p>
<p>&nbsp;</p><img src ="http://www.cppblog.com/jacky2019/aggbug/21578.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-10 10:57 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/10/21578.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>可变参数研究</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/10/21577.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Tue, 10 Apr 2007 02:38:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/10/21577.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/21577.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/10/21577.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/21577.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/21577.html</trackback:ping><description><![CDATA[<p><span>一．</span><span> </span><span>何谓可变参数</span><br>int printf( const char* format, ...); <br><span>这是使用过</span>C<span>语言的人所再熟悉不过的</span>printf<span>函数原型，它的参数中就有固定参数</span>format<span>和可变参数（用</span>&#8221;&#8230;&#8221;<span>表示）</span>. <span>而我们又可以用各种方式来调用</span>printf,<span>如</span>:<br>printf("%d",value); <br>printf("%s",str); <br>printf("the number is %d ,string is:%s", value, str);<br><span>二</span><span>.</span><span>实现原理</span><br>C <span>语言用宏来处理这些可变参数。这些宏看起来很复杂，其实原理挺简单，就是根据参数入栈的特点从最靠近第一个可变参数的固定参数开始，依次获取每个可变参数</span><span> </span><span>的地址。下面我们来分析这些宏。在</span>VC<span>中的</span>stdarg.h<span>头文件中，针对不同平台有不同的宏定义，我们选取</span>X86<span>平台下的宏定义：</span><br>typedef char *va_list; <br>/*<span>把</span>va_list<span>被定义成</span>char*<span>，这是因为在我们目前所用的</span>PC<span>机上，字符指针类型可以用来存储内存单元地址。而在有的机器上</span>va_list<span>是被定义成</span>void*<span>的</span>*/<br>#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) &amp; ~(sizeof(int) - 1) )<br>/*_INTSIZEOF (n)<span>宏是为了考虑那些内存地址需要对齐的系统，从宏的名字来应该是跟</span>sizeof(int)<span>对齐。一般的</span>sizeof(int)=4<span>，也就是参数在内存</span><span> </span><span>中的地址都为</span>4<span>的倍数。比如，如果</span>sizeof(n)<span>在</span>1<span>－</span>4<span>之间，那么</span>_INTSIZEOF(n)<span>＝</span>4<span>；如果</span>sizeof(n)<span>在</span>5<span>－</span>8<span>之间，那么</span> _INTSIZEOF(n)=8<span>。</span>*/<br>#define va_start(ap,v)( ap = (va_list)&amp;v + _INTSIZEOF(v) )<br>/*va_start <span>的定义为</span> &amp;v+_INTSIZEOF(v) ,<span>这里</span>&amp;v<span>是最后一个固定参数的起始地址，再加上其实际占用大小后，就得到了第一个可变参数的起始内存地址。所以我们运行</span>va_start (ap, v)<span>以后</span>,ap<span>指向第一个可变参数在的内存地址</span>*/<br>#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )<br>/*<span>这个宏做了两个事情，</span><br><span>①用用户输入的类型名对参数地址进行强制类型转换，得到用户所需要的值</span><br><span>②计算出本参数的实际大小，将指针调到本参数的结尾，也就是下一个参数的首地址，以便后续处理。</span>*/<br><span>　　</span>#define va_end(ap) ( ap = (va_list)0 ) <br>/*x86 <span>平台定义为</span>ap=(char*)0;<span>使</span>ap<span>不再</span><span> </span><span>指向堆栈</span>,<span>而是跟</span>NULL<span>一样</span>.<span>有些直接定义为</span>((void*)0),<span>这样编译器不会为</span>va_end<span>产生代码</span>,<span>例如</span>gcc<span>在</span>linux<span>的</span>x86<span>平台就是这</span><span> </span><span>样定义的</span>. <span>在这里大家要注意一个问题</span>:<span>由于参数的地址用于</span>va_start<span>宏</span>,<span>所以参数不能声明为寄存器变量或作为函数或数组类型</span>. */</p>
<p><span>以下再用图来表示</span>:</p>
<p><span>在</span>VC<span>等绝大多数</span>C<span>编译器中，默认情况下，参数进栈的顺序是由右向左的，因此，参数进栈以后的内存模型如下图所示：最后一个固定参数的地址位于第一个可变参数之下，并且是连续存储的。</span><br>|——————————————————————————|<br>|<span>最后一个可变参数</span> | -&gt;<span>高内存地址处</span><br>|——————————————————————————|<br>...................<br>|——————————————————————————|<br>|<span>第</span>N<span>个可变参数</span> | -&gt;va_arg(arg_ptr,int)<span>后</span>arg_ptr<span>所指的地方</span>,<br>| | <span>即第</span>N<span>个可变参数的地址。</span><br>|——————————————— | <br>&#8230;&#8230;&#8230;&#8230;&#8230;&#8230;&#8230;&#8230;&#8230;&#8230;.<br>|——————————————————————————|<br>|<span>第一个可变参数</span> | -&gt;va_start(arg_ptr,start)<span>后</span>arg_ptr<span>所指的地方</span><br>| | <span>即第一个可变参数的地址</span><br>|——————————————— | <br>|———————————————————————— ——|<br>| |<br>|<span>最后一个固定参数</span> | -&gt; start<span>的起始地址</span><br>|—————————————— —| .................<br>|—————————————————————————— |<br>| |<br>|——————————————— |-&gt; <span>低内存地址处</span></p>
<p><span>三</span><span>.printf</span><span>研究</span></p>
<p><span>下面是一个简单的</span>printf<span>函数的实现，参考了中的</span>156<span>页的例子，读者可以结合书上的代码与本文参照。</span><br>#include "stdio.h"<br>#include "stdlib.h"<br>void myprintf(char* fmt, ...) //<span>一个简单的类似于</span>printf<span>的实现，</span>//<span>参数必须都是</span>int <span>类型</span><br>{ <br>char* pArg=NULL; //<span>等价于原来的</span>va_list <br>char c;</p>
<p>pArg = (char*) &amp;fmt; //<span>注意不要写成</span>p = fmt !!<span>因为这里要对</span>//<span>参数取址，而不是取值</span><br>pArg += sizeof(fmt); //<span>等价于原来的</span>va_start </p>
<p>do<br>{<br>c =*fmt;<br>if (c != '%')<br>{<br>putchar(c); //<span>照原样输出字符</span><br>}<br>else<br>{<br>//<span>按格式字符输出数据</span><br>switch(*++fmt) <br>{<br>case 'd':<br>printf("%d",*((int*)pArg)); <br>break;<br>case 'x':<br>printf("%#x",*((int*)pArg));<br>break;<br>default:<br>break;<br>} <br>pArg += sizeof(int); //<span>等价于原来的</span>va_arg<br>}<br>++fmt;<br>}while (*fmt != '\0'); <br>pArg = NULL; //<span>等价于</span>va_end<br>return; <br>}<br>int main(int argc, char* argv[])<br>{<br>int i = 1234;<br>int j = 5678;</p>
<p>myprintf("the first test:i=%d",i,j); <br>myprintf("the secend test:i=%d; %x;j=%d;",i,0xabcd,j); <br>system("pause");<br>return 0;<br>}<br><span>在</span>intel+win2k+vc6<span>的机器执行结果如下：</span><br>the first test:i=1234<br>the secend test:i=1234; 0xabcd;j=5678;</p>
<p><span>四</span><span>.</span><span>应用</span><span><br></span><span>求最大值</span>:<br>#include //<span>不定数目参数需要的宏</span><br>int max(int n,int num,...)<br>{<br>va_list x;//<span>说明变量</span>x<br>va_start(x,num);//x<span>被初始化为指向</span>num<span>后的第一个参数</span><br>int m=num;<br>for(int i=1;i {<br>//<span>将变量</span>x<span>所指向的</span>int<span>类型的值赋给</span>y,<span>同时使</span>x<span>指向下一个参数</span><br>int y=va_arg(x,int);<br>if(y&gt;m)m=y;<br>}<br>va_end(x);//<span>清除变量</span>x<br>return m;<br>}<br>main()<br>{<br>printf("%d,%d",max(3,5,56),max(6,0,4,32,45,533));<br>}</p>
<p>&nbsp;</p>
<p><span>本文转载自网上，本来是要注明出处的，结果别人也都是转载的，呵呵。不过此文讲的很不错，很清楚，特别是把可变参数实现的那几个宏！偶也是冲着这几个宏去的。</span></p>
<p><span>#define _INT</span>SIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) &amp; ~(sizeof(int) - 1) )<br><span>这个宏，一开始我就是想不明白！不知道是老了，还是笨了，或者是生锈了。想了好一会还是没搞明白，不过看了一下本文的分析，一下子就明白了，那是相当的恍然大悟啊。</span></p>
<p><span>#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )</span></p>
<p><span>这个宏，一开始也是没有明白。多看了好几遍才发现了奥秘所在啊，为什么要加一个，然后再减一个呢？因为第一个加直接加到</span><span>ap</span><span>上去了，而后一个减只是减了一下括号内的值，也就是当前值了。</span></p>
<p><span>宏真是厉害啊！或者说它应用真广！这不禁让我想起以前看过的宏，怎么判断是</span><span>win</span><span>还是</span><span>linux</span><span>平台的，怎么判断是</span><span>32</span><span>位的还是</span><span>64</span><span>位的。宏是一门学问啊。</span></p>
<p><span>本文的另一大优点是，有非常简单的例子，看了就懂。恩。看了保管你就会用了。不过这年头指针也是个好东西啊，需要什么，传个指针就是传了一切想要的东西啊，只要让指针指向你需要的东西，可以是任意多的参数（不过这样子的话，具体到哪个参数结束就要我们自己来定了，不像这里，所有的参数都已经压栈了，编译器可以帮我们决定具体有多少个参数，到什么时候结束）。</span></p>
<p><span>Have fun.</span></p><img src ="http://www.cppblog.com/jacky2019/aggbug/21577.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-10 10:38 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/10/21577.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>Rational Rose 2003下载地址</title><link>http://www.cppblog.com/jacky2019/archive/2007/04/10/21573.html</link><dc:creator>kk</dc:creator><author>kk</author><pubDate>Tue, 10 Apr 2007 01:22:00 GMT</pubDate><guid>http://www.cppblog.com/jacky2019/archive/2007/04/10/21573.html</guid><wfw:comment>http://www.cppblog.com/jacky2019/comments/21573.html</wfw:comment><comments>http://www.cppblog.com/jacky2019/archive/2007/04/10/21573.html#Feedback</comments><slash:comments>16</slash:comments><wfw:commentRss>http://www.cppblog.com/jacky2019/comments/commentRss/21573.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/jacky2019/services/trackbacks/21573.html</trackback:ping><description><![CDATA[<a href="http://www.21php.com/Rational">http://www.21php.com/Rational</a> Rose 2003.rar<br><br>昨晚下的，速度100K左右
<img src ="http://www.cppblog.com/jacky2019/aggbug/21573.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/jacky2019/" target="_blank">kk</a> 2007-04-10 09:22 <a href="http://www.cppblog.com/jacky2019/archive/2007/04/10/21573.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>