2008年2月13日星期三

Total Commander 7.02a CCF中英绿色美化版本
推荐的原因:最经典的软件之一,如果你不会使用,请到xbeta兄的新浪博客 善用佳软栏目学习:

http://blog.sina.com.cn/s/alist_1188742767_2_1.html

下载在下面附件!

预览 (7.0):

s1.jpg (375.44 KB)
2007-10-25 18:59

======================================================

1. 常见问题
配置以英文/XP-2K3/XGA/Windows Classic Theme为标准. 最佳搭配WinRAR/DAEMON/集成解码包. 其余搜索论坛.

a. Vista用户
1. Office插件
此插件貌似只能使用绝对路径.由于Vista没有自带转换器,若TC的安装目录不是C:\Program Files\Total Commander,则必须手动修改Office.ini中MSWRD832.CNV的所在目录.
2. 部分WFX不支持Vista.
b. 创建自解压档:[url=]?[/url][url=]压缩时将*.RAR/ZIP/7Z改为*.EXE. ZIP格式可以在TC中将*.ZIP 直接改名为*.EXE?[/url]
c. 别名:[url=]?[/url][url=]按左/右方向键定位到命令行后,输入别名即可打开相应程序. 如→CDO打开光驱,CDC关闭.?[/url]
d. 整合WinRAR[url=]?[/url][url=]在TOOLS下建立WinRAR目录,拷入WinRAR.exe/Default.SFX/Rarreg.key三个文件, 打开USER.ini文件,将%PROGRAMFILES%\WINRAR\WINRAR.EXE替换为%COMMANDER_PATH%\TOOLS\WINRAR\WINRAR.EXE.?[/url]
e. (批量)转换图像格式[url=]?[/url][url=]选中一/多张图片,按Alt+F5, 右侧下拉框中选择imagine, 将文件后缀由imagine改为想要的格式, 如JPG, GIF, PIC等, 确定.不建议转换多帧GIF. ?[/url]
f. CHM[url=]?[/url][url=]CHMDir是WCX&WDX, WCX时用于制作CHM或提取CHM资源, WDX时用于自定义列/鼠标悬停/批量改名/搜索等, 此插件配合TC的按目录压缩功能, 适合打包Firefox插件Scrapbook储存的网页并利用其WDX功能进行批量改名.
WCX用法:

提取资源: 选中某CHM后按Ctrl+PgDn.
制作CHM: 选中文件夹或文件后,按Alt+F5,下拉选中CHM,确定.

IEView是WLX, 使用IE核心查看CHM, 按F3. 第一次打开时可能会跳出下载的提示,选择打开, 方框打勾, 确定. 按4则用Archview查看CHM压缩情况.?[/url]
g. 报毒
使用自解压文件时会自动删除目标目录中的所有文件.手动解压亦可.

2. 插件/工具
此版本包括WCX, WDX, WFX, WLX 共47个, ADDON/工具若干. 全部汉化.

WCX - 12个:
7zip (7z), CheckSum (sha), CHMDir (chm/its), DirCopy (dircopy), ICLRead (icl), Imagine (imagine), ISO (iso&etc), MHTUnpack (mht/msg), Mover (mover), TreeCopyPlus (treecopy), Wipe (wipe), InstallExplorer (exe/msi)
---Alt+F5打包,Ctrl+PgDn提取资源. 部分如 MHTUnp & ISO & InstallExplorer 只可Ctrl+PgDn, 不适用Alt+F5

Dircopy:复制本目录结构, 不复制文件
Imagine.wcx: Ctrl+PgDn提取图片资源, Alt+F5转换图片格式.
InstallExplorer: 支持Wise/Vise Installer, Inno Setup, Gentee Installer, InstallShield, NullSoft Installer, SetupFactory, Eschalon Installer, 和 Windows Installer (MSI).
Mover: 移动归类. 以文件后缀/日期等整理本目录下选定的 (打包)文件, 子目录不受影响.
TreeCopyPlus: 复制归类. 用于整理搜索结果, 保留各文件原来的目录结构.
Wipe: 安全删除文件.

WDX - 15个:
AnyTag, CDocProp, CRCTag32, DirSizeCalc, ExeFormat, EXIF, FileX, ImgSize, Permissions, RarInfo, Shareinfo, TextLine, Torrent, UnicodeTest, WDX4I
---Shift+F1选择.

Font (WLX)/CHMDir (WCX) 同时作为WDX使用.
自定义列预设12项:
Audio, Checksum, CHM, Compiled, Directory, FileInfo, Font, Photo, Picture, Office, RAR, Torrent.
自定义提示信息:
覆盖操作提示 - 6类, 鼠标悬停提示 - 9类.

WFX - 7个:
ProcFS (查看/关闭进程), MSIE Cache (查看IE缓存), TC Services (查看/启/禁用系统服务), Uninstalle (删除程序), TConsole (命令行窗口), DevMan (设备管理器), EventNT (事件查看)
---Alt+R进入网上邻居后使用.

WLX -13个:
Archview, Excellence, FileInfo, Font, ICLView, HTMLView, IEView, Imagine, Mmedia, nfoviewer, Office, VisualDirSize.
---F3/Ctrl+Q 调用.

支持查看字体 (Font), 影音 (Mmedia), HTM?/MHT (HTMLView), CHM/PDF... (IEview), 压缩档/CHM/MHT/ISO... (ArchView), Exe/Dll... (FileInfo), Excel (Excellence), Word (Office), 图片 (Imagine), Nfo/Diz (nfoviewer), 图标 (ICLView), 屏幕保护 (SCRList), 文件/目录大小 (VisualDirSize, 只支持Ctrl+Q).
部分文件可同时被多个插件支持,如本版的WinRAR自解压程序被Archview/FileInfo/ICLView同时支持,F3查看时可按4切换插件.

---工具
Akelpad: 文本编辑器, 含以下16个插件

AutoComplete: 编辑CPP/HTML时自动完成, Ctrl+F1
AutoSave: 自动保存. F8
FullScreen: 全屏. F11
HexSel: 查看HEX值
Highlight: 语法高亮
LineBoard: 行号, F6开关, Shift+F6设置颜色.
Lines: 行处理
LinkExtract: 提取HTML中的超链接
Macros: 宏
MinimizeToTray: 最小化至系统栏. F9
RecentFiles: 自动删除列表中不存在的文件
SaveAllAs: 批量转换代码页.
SelAutoCopy: 选择后自动复制.
SpecialChar: 显示换行符等特殊字符. F7开关, Shift+F7设置颜色.
Toolbar: 工具栏
ContextMenu: F12调用. 7.02+附带Akelpad的重要插件. Lines, Macros, HexSel, SelAutoCopy和LinkExtract等均通过此插件调用.不再设置单独的快捷键. 部分插件亦可通过此插件开关/设置.

NIRCMD: 多用途命令行工具,预设10个外部命令和9个别名.
---ADDON
TotalEditor:
预设TC外部命令em_batchopen/em_batchmedia. 同时打开选定的文本或影音档. 若选定 (多重)目录,则其下所有子目录的此类文件一并打开.
NewFile: 新建文件 (夹).
Restarter: 重启TC
TCQSP: 快速查找
TC Log Viewer: 查看操作记录

3. 配置
大致说明

a. 颜色:11种. MultiMedia, Picture, Archive, Disk Image, Document, Executable, Temporary, Hidden, Checksum, Download, Database
b. 工具栏: 2个, DEFAULT (用于TC内部命令)和TOOLS (用于管理系统/绿色工具).
c.Hotlist列表: 以WXP的系统变量为准 (英).
d. 图标/帮助: X-Qute, 其它为Vicons.中文帮助文件使用本坛slownet兄作品.
e. 菜单: 整合自EE+张学思版 (6.5x)+大量改动. 快捷键大部参照张版 (6.5x).
f. 开始菜单:

加/卸载镜像: 需DAEMON支持.
快速删除目录: 用于文件数量极大的目录 (不使用回收站),慎用!!!
批量打开文本: 选定目录/文件后使用该命令, 则Akelpad会打开选定 (文件夹 (含子文件夹下))的所有文本.
批量打开影音: 同上, 调用Windows Media Player播放, 最好配合集成Codec包,如Final Codecs.
新建文件/夹: 给不知道Shift+F4的人用.
新建会话: 强行新开TC窗口
重启TC.
快速查找加强版.
查看操作记录.
备份至桌面: 打包User目录和WCX_FTP.ini至桌面.

======================================================
更新细节:
* Update on 1/10/07
更新:
TC -> 7.02a
WDX 2个: AnyTag -> 0.98(支持Vista), WDX4I -> 0.1b
Akelpad -> 3.40及全部插件
ADDON 1个: TCQSP -> 2.2
UNRAR.DLL -> 3.71.1.246 (WinRAR)
中文帮助HLP -> 7.02a Final.

增加:
ADDON 1个: TC Log Viewer(查看操作记录)及相关外部命令.
Akelpad插件 1个: FullScreen
Office插件所需的转换器.(For Vista)

调整:
部分插件设置
更换MHTUnp.wcx为MHTUnPack.wcx

* Update on 7/9/07[url=]?[/url][url=]更新:
TC -> 7.02
WDX 2个: FileX -> 1.7; CRC32Tag -> 0.21Beta
WLX 2个: Imagine -> 1.0.0.0 Beta 5, HTMLView -> 1.2
AkelPad -> 3.38 及大部分插件.
NirCMD -> 2.00, 兼容Vista
UNRAR.DLL -> 3.70.100.211 (WinRAR)
中文帮助HLP -> 7.01 Final.

增加:
WDX 5个: WDX4i, Permissions, ExeFormat, UnicodeTest, TextLine
WFX 2个: MSIE Cache (查看IE缓存), EventNT (事件查看)
WLX 2个: HTMLView (这个和IEView都有Bug,只能再次补上), SCRList (查看SCR)
ADDON 1个: TCQSP (快速查找加强版)
颜色方案 2个: Download, Database
自定义列 1个: Compiled
鼠标悬停提示 2个: Compiled及Ini/Inf
TinnyTCRestarter及相关外部命令/开始菜单项
快捷键 2个: Shift+Ctrl+F9/0
别名7个 (NirCMD相关)
AkelPad插件 3个: Macros, LinkExtract, ContextMenu, AutoSave
FileInfo插件UnUpack.dll (PE Explorer)
常用设置备份功能.

调整:
菜单/中文语言包/开始菜单/INC/插件设置/自定义列...
WDX <- DirSizeCalc: 降回1.21, 2.0B2无法按目录大小排序, 除非设置两列大小.
拆分WINCMD.INI/USER.INI, 升级保留文件大都并入USER目录, 其中USER.INI定义常用选项, HISTORY.INI保存操作记录.
汉化FileInfo (2.09), EventNT (1.3), 进一步汉化ICLView和部分WDX显示信息.
大量微调.

删除:
WFX 1个: Versions
RAR的覆盖提示设置
AkelPad插件: URLMenu?[/url]
* Update on 22/6/07[url=]?[/url][url=]修正版本.
更新:
TC -> 7.01
AkelPad -> 3.32 及部分插件.
UnRAR.dll (WinRAR)
中文帮助HLP -> 7.0 Final.

增加:
快捷键: Shift+Ctrl+Home
AkelPad插件: RecentFiles

删除:
外部命令/别名三个,RS/IS/CS.

调整:
菜单/INC描述, Ctrl+Home/End
修正英文版使用汉化MMedia及菜单中的Unicode字符.?[/url]
* Update on 8/6/07[url=]?[/url][url=]更新:
TC -> 7.0 Final
AkelPad -> 3.31, 及全部插件.
WLX 1个: Imagine -> 1.0.0.0 Beta 4
WDX 2个: FileX ->1.6, DirSizeCalc -> 2.0 Beta 2

增加:
WDX 1个: AnyTag (尺寸较大,但AudioInfo有些问题)
Addon 1个: NewFile (用于新建文件 (夹))
AkelPad插件5个: HexSel, AutoComplete, Lines, SaveAllAs, SpecialChar.
NIRCMD及相关外部命令10个/别名2个
快捷键: Shift+Alt+123456789, Shift+Ctrl+C, Shift+Ctrl+I
TotalEditor外部命令1个: BatchMedia.

删除:
CDOC及相关外部命令em_CDOC/别名CDOC
WLX 1个: HTMLView
WDX 1个: AudioInfo (速度慢/MP3只读取ID3V1/停止开发)及相应配置

调整:
所有分辨率共用设置,存入User.ini
菜单, INC, 语言包
WCX后缀: gif -> imagine, tcpy -> treecopy, move -> mover
Devcon.exe并入TC目录,用于在\\设备管理器中删除设备.
[Rename]和[Splitperfile]段并入USER.ini.
快捷键: Shift+Ctrl+123456, Ctrl+123456
默认显示属性, 记录日志.
插件参数.?[/url]
* Update on 17/3/07[url=]?[/url][url=]更新:
TC -> 7.0 PB4
ADDON 1个: TotalEditor -> 0.20
WCX 2个: ISO -> 1.74b1, 7Zip -> 0.55
WDX 1个: CRC32Tag -> 0.19B
TOOLS 1个: Akelpad -> 3.25

增加:
WLX 1个: HTMLView, 主要针对IEView和MMedia的冲突.
Akelpad插件2个: LineBoard, HighLight
程序别名18个
覆盖操作及鼠标悬停自定义提示信息, 7类/5类
外部命令20个.
自定义列2个: CHM, FileInfo
新增内部命令的菜单项及相应快捷键

调整:
NoClose并入TC所在目录.
菜单, 语言包, INC, 工具栏, 其它参数修正
快捷键:Ctrl+012349, Shift+Ctrl+012349,Alt+X/Z,Ctrl+Home/End

取消:
TotalEditor对压缩包的设置?[/url]
* Update on 24/12/6 (2007大礼包专用版)[url=]?[/url][url=]更新:
WCX 1个: 7zip -> 0.54
WLX 2个: FileInfo -> 2.09, Imagine -> 1.0.0.0 b3
WDX 1个: CRC32Tag -> 0.18B
UnRAR.dll (From WinRAR) & UnUPX.dll (From PE Explorer).
图标库: ICL->DLL, 支持菜单图标.

增加:
WFX 4个: Uninstaller (卸载程序) / TConsole (命令行窗口) / DevMan (设备管理器) / Versions (在线更新)
WCX 2个: MHTUnp (提取MHT资源) / Imagine (制作/提取GIF资源)
ADDON 1个: TotalEditor/及自定义命令em_batchopen, 已设置批量打开文本文件/压缩档 (开始菜单中).
自定义列1个: Font
菜单图标
缩略图自定义项

调整:
默认编辑器: AkelPad (含插件MinimizeToTray/SelAutoCopy/Toolbar/URLMenu).
快捷键: 细分Shift+F6和F2, Alt+L&U / 左&两侧文件夹树.
MSI: Enter运行,Ctrl+PgDn提取资源.
WLX次序,Archview增加7Zip支持,Fileinfo支持查看UPX/Aspack压缩细节.
Hotlist.
允许预览WMF (=3).
缩略图缓存目录设为..\CACHE.
开始菜单.
汉化部分插件, 如ProcFS/Services/TConsole/ICLView...
大量微调.

*****以下段值从此版开始另存入%commander_path%\user.ini文件,今后升级本美/绿版本可备份此文件保留大部分个人设置*****
[Commandlinehistory]
[DirMenu]
[DriveHints]
[MkDirHistory]
[RenameTemplates]
[RenameSearchFind]
[RenameSearchReplace]
[SearchName]
[SearchIn]
[SearchText]
[Selection]
[left]
[right]
[lefttabs]
[righttabs]
[RightHistory]
[LeftHistory]
[user]
[800x600 (8x16)]
[1152x864 (8x16)]
[1280x1024 (8x16)]
[1400x1050 (8x16)]
[1440x900 (8x16)]
[1600x1200 (8x16)]
[1680x1050 (8x16)]?[/url]
* Update on 24/11/06[url=]?[/url][url=]升级:
TC ->7.0 PB2
SkimEdit ->3.07
WCX 2个: CHMDir -> 0.40c, ICLRead -> 1.3
WLX 4个: FileInfo ->2.08, Font -> 0.09, ICLView -> 3.10.2006, Imagine -> 0.97 Final

增加:
WCX 4个: DirCopy (复制目录结构), ICLRead (制作/提取ICL资源) & Mover (文件移动归类), InstallExplorer (提取MSI资源)
WDX 1个: ShareInfo, CRCTag32
WFX 2个: ProcFS (任务管理器), TC Services (系统服务)
颜色方案1个: Checksum
自定义列1个: Checksum

修正:
菜单: TC7新增命令,冗余项
WLX: Font (中文版), 描述错误,
SFXHEAD.SFX: 替换为WinRAR的Zip.SFX.
图标库兼容Vista.

快捷键:
Alt+U: 改为开/关一列文件夹树 (左侧窗口)
Shift+Ctrl+F8: 设为开/关文件夹树
Shift+Ctrl+Y: 设为查看TC内部命令

调整:
菜单, 颜色, INC, Button Bar, blahblahblah...?[/url]
* Update on 7/8/6[url=]?[/url][url=]升级:
部分插件

修正:
菜单, 工具栏, 描述错误, WDX配置, 其它微调.

替换:
中文版帮助文件换回HLP格式, CHM对压缩免疫,:- (.
NoClose.pif 换成 NoClose Replacer
WLX 1个 - IEView (不同作者)

增加:
WCX 2个 - Wipe (Secure Delete) & TreeCopyPlus
WDX 4个 - AudioInfo & CDocProp & RarInfo & Torrent
WLX 3个 - ArchView & nfoviewer & Mmedia (调用WMP, 配合集成codecs为佳, e.g. ffdshow, MyMPC)
颜色方案 4个 - Document, Executable, Temporary, Hidden
自定义列 4个 - Office, RAR, Torrent, Photo
插件中文 - Excluding ICLView.

删除:
绿色软件 2个 - Remove Hotfix Backups & RegCompact
WDX 1个 - AnyTag (过大, 替换为AudioInfo)
图标库 Buttonbar.icl 中未用图标?[/url]
* Update on 3/8/06[url=]?[/url][url=]修正:英文版中的几个绝对路径, 菜单冗余项
增加:中文版.?[/url]

串行通讯注意事项

作者:佚名 文章来源:本站原创 点击数:1234 更新时间:2005-6-6

1 选择通讯方式 -- 同步还是非同步
正如在《Serial communications in MicrosoftWin32》等文章中提到的,同步(NonOverLapped)方式是比较简单的一种方式,编写起来代码的长度要明显少于异步(OverLapped)方式,我开始用同步方式编写了整个子程序,在Windows98 下工作正常,但后来在Windows2000下测试,发现接收正常,但一发送数据,程序就会停在那里,原因应该在于同步方式下如果有一个通讯 Api在操作中,另一个会阻塞直到上一个操作完成,所以当读数据的线程停留在 WaitCommEvent 的时候,WriteFile就停在那里。我又测试了我手上所有有关串行通讯的例子程序,发现所有使用同步方式的程序在 Windows 2000下全部工作不正常,对这个问题我一直找不到解决的办法,后来在 Iczelion 站点上发现一篇文章提到 NT 下对串行通讯的处理和 9x有些不同,根本不要指望在 NT 或 Windows 2000 下用同步方式同时收发数据,我只好又用异步方式把整个通讯子程序重新写了一遍。所以对于这个问题的建议是:如果程序只打算工作在 Win9x 下,为了简单起见,可以用同步方式写程序,如果程序打算在 NT下也可以工作的话,就必须用异步方式写。

2 Win32 通讯 API Bug 之一 --- CommConfigDialog

CommConfigDialog 是弹出系统内置串口设置对话框的 API,我们在设备管理器中设置串口参数的对话框就是这个,使用这个API 时不用先打开端口,它并不针对一个已打开的端口,而是仅仅是把 DCB 的内容填写到对话框中,当按了 OK 后把输入的结果存回到
DCB 数据结构中,至于什么时候把结果设置到串口上,那就是你自己要做的事情了。
CommCinfigDialog 的定义如下:
BOOL CommConfigDialog(
LPTSTR lpszName, // pointer to device name string
HWND hWnd, // handle to window
LPCOMMCONFIG lpCC // pointer to comm. configuration structure
);

但在使用中发现,对话框有时能出来,有时出不来,最后总结的经验是问题出在 COMMCONFIG 结构的 dwSize字段上,COMMCONFIG 的定义如下:
typedef struct _COMM_CONFIG {
DWORD dwSize;
WORD wVersion;
WORD wReserved;
DCB dcb;
DWORD dwProviderSubType;
DWORD dwProviderOffset;
DWORD dwProviderSize;
WCHAR wcProviderData[1];
} COMMCONFIG, *LPCOMMCONFIG;

在参数中,wVersion 要填 100h,dwProviderSubType 要填 1,但 dwSize 就不能填 sizeof
COMMCONFIG 了,我发现好象一定要把 dwSize 设置为比 sizeof COMMCONFIG
对话框才能出来,所以我用的代码中定义了一个足够大的缓冲区作为结构的地址:
_CommConfigDialog proc
local @stCC[256]:BYTE
pushad
invoke RtlZeroMemory,addr @stCC,sizeof @stCC
mov (COMMCONFIG ptr @stCC).dwSize,256
mov (COMMCONFIG ptr @stCC).wVersion,100h
mov (COMMCONFIG ptr @stCC).dwProviderSubType,1
invoke CommConfigDialog,addr [esi].szPortName,[esi].hWnd,addr @stCC
popad
ret
_CommConfigDialog endp

3 Win32 通讯 API Bug 之二--- BuildCommDCB

BuildCommDCB 的功能是把一个字符串如 com1:9600,n,8,1 这样的转换到具体的数据填写到 DCB中,但使用中也存在问题,我发现我用它转换象 com1:9600,e,7,1 之类的带校验位的字符串,它总是无法把这个 e给我转换过去,设置好串口一看,成了 9600,n,7,1,而上面提到的 CommConfigDialog返回的结果用来设置串口却是正确的,经过比较,发现问题出在 DCB.fbits.fParity 这个 bit 上,只有把这个 bit 置1,校验位才是有效的,而 BuildCommDCB 恰恰是漏了这个 bit,所有如果你要使用 BuildCommDCB,别忘了补充把DCB.fbits.fParity 设置回去,我用的代码是:

_BuildCommDCB proc _lpszPara,_lpstDCB
pushad
mov esi,_lpstDCB
assume esi:ptr DCB
invoke RtlZeroMemory,esi,sizeof DCB
invoke BuildCommDCB,_lpszPara,esi
;********************************************************************

; 根据校验位补充设置 DCB 中的 DCB.fbits.fParity 字段
;********************************************************************
mov dword ptr [esi].fbits,0010b
cld
@@:
lodsb
or al,al
jz @F
cmp al,'='
jz _BCD_Check
cmp al,','
jnz @B
_BCD_Check:
lodsb
or al,al
jz @F
or al,20h
cmp al,'n'

jnz @B

;********************************************************************
; 扫描到 =n 或 ,n 则取消校验位
;********************************************************************

mov esi,_lpstDCB
and dword ptr [esi].fbits,not 0010b
@@:
popad
ret
_BuildCommDCB endp

4 Win32 通讯编程的一般流程

由于同步方式相对比较简单,在这里讲述的是异步方式的流程,在其他的很多文章里提到了 Windows 通讯 API 有二十多个,它们是:
BuildCommDCB
BuildCommDCBAndTimeouts
ClearCommBreak
ClearCommError
CommConfigDialog
EscapeCommFunction
GetCommConfig
GetCommMask
GetCommModemStatus
GetCommProperties
GetCommState
GetCommTimeouts
GetDefaultCommConfig
PurgeComm
SetCommBreak
SetCommConfig
SetCommMask
SetCommState
SetCommTimeouts
SetDefaultCommConfig
SetupComm
TransmitCommChar
WaitCommEvent
我刚看到这些 API 的时候,都不知道如何使用它们,但并不是所有这些 API 都是必须用的,比如说你要检测当前串口的设置可以只用
SetCommState 而不用 GetCommProperties 和
GetCommConfig,虽然它们返回的信息可能更多。同样,如果有些值你想用缺省的,比如缓冲区的大小和超时的时间等等,那么
SetupComm 和 BuildCommDCBAndTimeouts、SetCommTimeouts
也可以不用,TransmitCommChar 是马上在发送序列中优先插入发送一个字符用的,平时也很少用到,下面讲的是必须用到的 API和使用步骤
建立 Event -- 用 CreateEventinvoke CreateEvent,NULL,TRUE,FALSE,NULL
用异步方式操作串口必须要定义 OVERLAPPED 结构,其中的 hEvent 必须自己建立,你要定义两个 OVERLAPPED结构,一个用于读一个用于写,当然也必须建立两个 Event,把它们放入 OVERLAPPED.hEvent

打开串口 -- 用 CreateFile
invoke CreateFile,addr szPortName,GENERIC_READ or
GENERIC_WRITE,0,NULL,OPEN_EXISTING,FILE_FLAG_OVERLAPPED,NULL
注意用异步方式必须指定 FILE_FLAG_OVERLAPPED,而文件方式必须 OPEN_EXISTING,读写必须是
GENERIC_READ or GENERIC_WRITE

设置串口参数 -- 用 SetCommState
invoke SetCommState,hCom,addr dcbx
hCom 是前面打开成功后返回的句柄,dcbx 是数据结构
DCB,里面包括了通讯的具体参数,至于这个参数的建立,你可以自己填写,也可以用前面提到的 BuildCommDCB 或
CommConfigDialog 填写

建立读数据的线程
到这里,就可以开始读数据了,一般我们是在主线程中写数据,因为写是我们可以控制的,而读的时候我们不知道数据什么时候会到,所以要建立一个线程专门用来读数据,在这个线程中,我们循环地用
ReadFile 读串口,同时用 WaitCommEvent 检测线路状态。

如果要检测通讯状态,如 CTS 信号,RingIn 等等 -- 用
SetCommMask、WaitCommEvent、ClearCommError、GetCommModemStatus

invoke SetCommMask,hCom,EV_BREAK or EV_CTS or EV_DSR or EV_ERR or
EV_RING or EV_RLSD or EV_RXCHAR or EV_RXFLAG or EV_TXEMPTY
SetCommMask 指定 WaitCommEvent 要等待的事件名称,具体的参数请查手册


invoke WaitCommEvent,hCom,addr dwEvent,NULL
WaitCommEvent 等待一直到 SetCommMask 指定事件之一发生


invoke ClearCommError,hCom,addr dwError,addr stComStat
在 WaitCommEvent 以后,要用 ClearCommError 清除事件的 Flag,以便进行下一轮
WaitCommEvent,同时这个 API 可以获得更详细的事件信息


invoke GetCommModemStatus,hCom,addr dwModemStatus
同样,GetCommModemStatus 是用来获得串口线路状态的,如 CTS、RING 等等,当 WaitCommEvent
返回时,只是指出了如 CTS 等等状态有变化,但具体是变成 On 还是 Off 了还要靠这个 API 去取得更详细的信息

读数据 -- 用 ReadFile


invoke ReadFile,hCom,addr szBuffer,sizeof szBuffer,addr
dwBytesRead,addr stReadState
最后一个参数是开头定义的 OVERLAPPED 结构的地址,指定了它就表示是用异步方式的读方式,这个 API 会马上返回,接下去要用

invoke GetOverlappedResult,hCom,addr stReadState,addr
dwBytesRead,FALSE
将其余的数据读完

结束时关闭端口 -- 停止 WaitCommEvent 的等待以及关闭端口 CloseHandle

平时程序会停留在 WaitCommEvent 的等待中,当要终止线程的时候,必须是程序从 WaitCommEvent
中退出来,这时候要用

按照 Win32 手册上的说明,参数为 NULL 的 SetCommMask 会使另一个线程中的 WaitCommEvent
马上返回,然后就是用 CloseHandle 关闭端口
invoke CloseHandle,hCom
5 Win32 通讯 API Bug 之二--- SetCommMask 和 WaitCommEvent

严格的说这不应该是 Bug,而是偶然的情况,我发现有些时候我的读线程无法结束,跟踪发现是停在了 WaitCommEvent
上,这说明有时候 invoke SetCommMask,hCom,NULL 并不能使 WaitCommEvent
退出,我最后使用的办法是: 在 SetCommMask 以后再执行 invoke
SetEvent,stReadState.hEvent,把读的 OVERLAPPED 结构中的 Event 置位,让
WaitCommEvent 认为有 Event 发生,它就会马上返回,也许这并不是普遍的情况,但如果你的程序也是停在了
WaitCommEvent 的地方,不妨一试。

6 如何编写读线程中的循环

按照《Serial communications in Microsoft Win32》一文中的例程,读循环可以用:
#define READ_TIMEOUT 500 // milliseconds
DWORD dwRes;
DWORD dwRead;
BOOL fWaitingOnRead = FALSE;
OVERLAPPED osReader = {0};
// Create the overlapped event. Must be closed before exiting
// to avoid a handle leak.

osReader.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (osReader.hEvent == NULL)
// Error creating overlapped event; abort.
if (!fWaitingOnRead) {
// Issue read operation.
if (!ReadFile(hComm, lpBuf, READ_BUF_SIZE, &dwRead, &osReader)) {
if (GetLastError() != ERROR_IO_PENDING) // read not delayed?
// Error in communications; report it.
else
fWaitingOnRead = TRUE;
}

else {
// read completed immediately
HandleASuccessfulRead(lpBuf, dwRead);
}

}

if (fWaitingOnRead) {
dwRes = WaitForSingleObject(osReader.hEvent, READ_TIMEOUT);
switch(dwRes)
{

// Read completed.
case WAIT_OBJECT_0:
if (!GetOverlappedResult(hComm, &osReader, &dwRead, FALSE))
// Error in communications; report it.
else
// Read completed successfully.
HandleASuccessfulRead(lpBuf, dwRead);

// Reset flag so that another opertion can be issued.
fWaitingOnRead = FALSE;
break;
case WAIT_TIMEOUT:

// Operation isn't complete yet. fWaitingOnRead flag isn't
// changed since I'll loop back around, and I don't want
// to issue another read until the first one finishes.
//
// This is a good time to do some background work.

break;
default:

// Error in the WaitForSingleObject; abort.
// This indicates a problem with the OVERLAPPED structure's
// event handle.

break;


}

这一段程序在 98 下正常,但非常不幸的是在 Win2000 下,ReadFile 总是返回读正确,并不返回
ERROR_IO_PENDING,使下面的 WaitForSingleObject 的循环形同虚设,要命的是,ReadFile
返回读正确却每次只读一个字节,结果程序工作得很奇怪,即使缓冲区中有很多的字符,程序也每次只能读一个字符,要等到发送字符或做其他的操作使线路状态改变了,才能读下一个字符,我不知道这个奇怪的现象是如何发生的,反正我解决的办法是在
ReadFile 前加 WaitCommEvent,真正等到 EV_RXCHAR 以后才去
ReadFile,到最后,我用的循环是这样的,虽然没有一篇文章中的例子是这样的,但它却同时在 windows9x 和
windows2000 下工作得很好:

.while dwFlag & IF_CONNECT

;********************************************************************
; 检测其它的通信事件
; 如果检测到且定义了 lpProcessEvent 则调用 lpProcessEvent
;********************************************************************

invoke WaitCommEvent,hCom,addr @dwEvent,NULL ;addr stReadState
push eax
invoke ClearCommError,hCom,addr @dwError,addr @stComStat
pop eax
.if eax == 0
invoke GetLastError
.if eax == ERROR_IO_PENDING
or dwFlag,IF_WAITING
.endif
.else

;这里是线路状态的处理
.endif

;********************************************************************
; 如果没有在等待异步读的过程中,则读端口
;********************************************************************

.if ! (dwFlag & IF_WAITING)
mov @dwBytesRead,0
invoke ReadFile,hCom,addr @szBuffer,sizeof @szBuffer,\
addr @dwBytesRead,addr stReadState
.if eax == FALSE
or dwFlag,IF_WAITING
invoke GetLastError
.if eax != ERROR_IO_PENDING
;这里是错误处理
.endif
.else

and dwFlag,not IF_WAITING
mov eax,@dwBytesRead
.if eax != 0
;这里是接收到的数据处理
.endif
.endif
.endif

;********************************************************************
; 如果在异步读端口中,则等待一段
;********************************************************************
.if dwFlag & IF_WAITING
invoke WaitForSingleObject,stReadState.hEvent,200
.if eax == WAIT_OBJECT_0
and dwFlag,not IF_WAITING
invoke GetOverlappedResult,hCom,addr stReadState,\
addr @dwBytesRead,FALSE
.if eax != 0
mov eax,@dwBytesRead
.if eax != 0

;这里是接收到的数据处理
.endif
.else
;这里是错误处理
invoke ClearCommError,hCom,addr @dwError,addr @stComStat
.endif
.else
;这里是错误处理
.endif
.endif
.endw

7 流控制的问题
在流控制方式为"无"和"软件控制"的情况下,基本上没有什么问题,但在"硬件控制"下,win32 手册中说明
RTS_CONTROL_HANDSHAKE 控制方式的含义是:
Enables RTS handshaking. The driver raises the RTS line when the
"type-ahead" (input) buffer is less than one-half full and lowers
the RTS line when the buffer is more than three-quarters full. If
handshaking is enabled, it is an error for the application to adjust
the line by using the EscapeCommFunction function.
也就是说,当缓冲区快满的时候 RTS 会自动 OFF 通知对方暂停发送,当缓冲区重新空出来的时候, RTS 会自动 ON,但我发现当
RTS 变 OFF 以后即使你已经清空了缓冲区, RTS 也不会自动的
ON,造成对方停在那里不发送了,所以,如果要用硬件流控制的话,还要在接收后最好加上检测缓冲区大小的判断,具体是使用
ClearCommError 后返回的 COMSTAT.cbInQue,当缓冲区已经空出来的时候,要使用 invoke
EscapeCommFunction,hCom,SETRTS 重新将 RTS 设置为 ON。


来自http://www.gjwtech.com