|
一、问题的提出
% e L6 a( f. d0 B3 e, I, f$ d9 FWindows 已由原来的16位 windows 3.x 升级为现今我们使用的32位windows 95/97/98
1 M7 M: |: R) E6 m$ |: I以其 Windows NT,用户不仅在使用上应逐步适应,对于程序开发人员来说在编程技术9 Z' A) H7 z4 _& w
上也应紧跟操作系统的技术发展,就如同 在 Linux 操作系统下, X-Window 编程就显
5 t7 C) o C) O- \2 B* Z {得很重要一样。作为一个完整成熟的 Windows 程序,需要保存程序所有的环境变量和
0 J8 f& r5 K/ _8 s8 k# Y私有信息。诸如用户的偏好,文件装入的列表、退出时用户使用的窗口位置 .存盘历史6 t0 i9 O* w5 X7 s3 z5 m
纪录等。过去在 windows 3.x时代 ,常用 Win16函数 Get/RegWrite ProfileString
1 U- @. v. Z4 R' i将有关程序的信息写入 *.ini 文件,但现在该项技术由 Win32注册表所替代。. V4 L ~9 {, p
可以这样说,注册表是当今32位 Windows 操作系统的灵魂,一切信息都在其中,也就# t4 K: q+ B- Z" w6 G
是为什么Windows98在Windows95的基础上升级可以不重装软件等等
, `# T. C% v0 R的如此方便的应用,其原理就是根据了原注册表中的信息来完成各种方便的处理,所以* \" O0 ?& Q* c9 D% ^ w
Windows注册表对应用程序的重要性就显而易见了。
) C; j0 {' ]2 e/ V4 ~2 O原来的 Win16程序存储私有信息是在一个平面文件INI中,这样做有很多弊端,例如该9 o7 R. T* J/ v, C% q1 b
INI文件没有任何安全机制,用户可以直接在INI文件中修改各种参数和程序入口,这样
6 R$ F) A" U# R就可能造成不可估计的严重后果,还有该文件只能支持和文本数据不能存入二进制数据( L2 Q1 Z( L( q$ e1 C: N0 {! r8 W; L
等各种不利因素,所以微软的工程师也认识到这一点,于是注册数据库就诞生了,注册
' ~: J" X0 s, K2 ~: N8 W4 M数据库就是为了解决在 Windows 3.x 的一些关于 OLE 的此类问题而创建的,现在
/ W/ C7 _; f' U* X: @1 iWin32 应用程序的注册数据库通过微软带给我们的新的 Win32 API 得到了显著的改& w: c0 N. m$ m3 } v3 h
善。使用访问注册表的 Win32 函数比起使用管理 INI 文件的 Win16函数要灵活的多,
. C4 p2 O" [9 B! G+ O. ?这意味着在功能上将大大增强,但是,另一方面,如果你还未用过,就会对处理注册表
# S/ z9 F8 ]' A! U# V( v的Win32 API 的新规则感到困惑或不知所措。本文就是本着这一目的,逐步让你懂得并
8 w3 G: W: a6 b$ U2 L! A- Y掌握怎样用 Win32API 函数来处理32位 Windows 程序注册表的方法。 / I- L) F5 w% T' M& _
. H9 o7 l1 d T' I
二 . 技术的实现原理 : R3 m( k& G) J! J& V
为了在以后自己编写的程序中更多的体现模块化思想以及使编程变得更加简单,应尽可, S0 D v4 Q+ J6 n `) ?
能的建立自己实现各种功能的类,以类作为实现应用程序各种功能的单位。 在此,可
' m2 Z2 V; V, I0 X! r+ O6 S# ]以创建一个包括注册表许多常用功能而接口简单的类库,下面将建立 CMyRegKey类,对% n) } s- X6 W
应用程序处理注册表的具体细节进行封装,从而在外部通过这个功能类方便地实现进行
) j: O7 d+ x' p8 y访问注册表信息的各种操作,在外部调用其成员函数即可。以后,你就可以在每一个应
' ~: ^! y: ?" B/ k' B1 I" F7 m用程序中包含此类并用其外部接口进行编程了。
1 Y, S5 k' ~7 C j8 ?8 `0 K三 . 实现代码与步骤
5 E l8 I2 Q% T' Z' i1 D7 S7 d
1. 建立功能类的头文件:
" k# ]) s, k& m' x) d R创建一个新的头文件 MyRegKey.h ,在其中加入以下的代码。
3 z* e( R# V5 R% f5 H( c
/ ?$ P8 \# x' I, ~6 u& G v#include "winreg.h" ; H3 {3 Q% g, _( ]& J/ @) P
// 包含头文件 winreg.h , 因注册表Win32 API 函数在其内定义 % U8 D7 _! ]: c! O9 A5 Q. L, W- k
// 建立 CMyRegKey 类:* \7 ?) Z. F+ i; l; X4 K7 P
class CMyRegKey
$ M' ^5 _% T6 v5 p6 z) F# i2 \{ " p7 G5 |7 g( u; R- O H/ h
* \2 M5 b! C% U* \$ ]. `// Construction5 s g+ a% e+ \6 h; ~: y* p
public:, c, g8 ^5 w3 m/ |
CMyRegKey();9 U8 @6 ~; W8 L8 A( A
virtual ~CMyRegKey ();
3 u7 S- h0 e% `( H& ^- M$ j- M// Attributes. _. o& e/ m4 N5 Z+ s: x/ b6 T
public:
9 k) {+ e, X" J- U S6 O( q// 定义打开和关闭注册表的成员函数:
4 f$ B+ Y- M: k1 _6 p1 I* f& PLONG RegRegOpen(HKEY hKeyRoot,LPCTSTR pszPath);/ q! p* |/ S! T$ W6 o
void RegRegClose();
9 ^+ v+ Z& {& B- Z0 w// 利用函数重载实现对注册表键值(串值,二进制值,DWORD值 ) 的读和写:
9 l, f9 L; Y" z8 F7 WLONG RegRead (LPCTSTR pszKey,DWORD& dwVal);
1 f& | u z- }3 l+ ^LONG RegRead (LPCTSTR pszKey,CString& sVal);
: H8 w8 X U4 s7 t6 mLONG RegRead (LPCTSTR pszKey,BYTE *pData,DWORD& dwLength);
. p" |: e5 L+ A9 P/ u8 k% vLONG RegWrite (LPCTSTR pszKey,DWORD dwVal);
$ @& h! d% {0 B: d% K3 h! kLONG RegWrite (LPCTSTR pszKey,LPCTSTR pszVal);# B3 n" L# b) C; v# w
LONG RegWrite (LPCTSTR pszKey,const BYTE *pData,DWORD dwLength);
$ f. N! |9 c& Q& O# ~" wprotected:
; V+ w+ W3 I {2 I) d) y* \HKEY m_hKey;% g/ f4 \; I$ X. d8 ~$ y3 f
CString m_sPath;7 I! d% y' R8 Z
};
# M2 l# `) @. D( S. I9 v2. 建立功能类的Cpp文件定义 CMyRegKey类:
$ P1 [% R `' K+ W8 R0 a8 y9 ?创建一个新文件 MyRegKey.cpp ,代码如下:
# H* M) U( X+ i% J9 w9 ~
/ G' l- p0 ^ }+ V: E#include "MyRegKey.h" : B* y7 }6 F8 i' Z9 E
////////////////////////////////////////////////////////////////////////////
; g/ I8 P( @5 }/ x8 Q) U3 A' M, U$ F/5 ]. [. e/ V- S& k/ W0 r
// CMyRegKey
# {% u7 {+ k, Z////////////////////////////////////////////////////////////////////////////
- x% e0 ~0 I3 f) A/ f/ + Q$ y7 ?, P/ I6 ?) K
CMyRegKey:: CMyRegKey()
& \$ J9 T4 S/ h) X$ E8 Q2 K{% X; e8 n# {, S5 F* Y, v
m_hKey = NULL;" b( Z r* h7 x+ V, V
} ( C& J( o& D7 r/ z+ F2 ~9 u! Y+ B
CMyRegKey:: ~CMyRegKey()
' R- r7 z c7 |{
' ?8 {" x& V5 ?- S* V1 LRegClose();7 |% L' k3 l0 c, k9 ^
}
% q- B0 q$ {$ ~$ v+ x- {( Q// 定义打开注册表的函数,RegOpen 函数带有两个参数:指定要访问注册表根结点的. k) I& [8 G0 w" G5 {+ A
HKEY, 以及注 - s5 F6 [" L0 Y. d
// 册表中信息的全路径。如果给入的路径不存在,则需创建一个新路径。从
/ S8 C) J' Q! h8 g' ]! d* MRegCreateKeyEx API函数返// 回的 HKEY 作为 m_hKey 存储。 & `3 S% ^4 F; E; Y) X0 P
LONG CMyRegKey::RegOpen(HKEY hKeyRoot,LPCTSTR pszPath) 1 ?5 P: H# ?7 q
{
) z1 P% }" Q9 M1 JDWORD dw;3 _5 m& m' \$ z; q- M% f2 S9 H& M
m_sPath = pszPath; 1 }& O2 U c7 e$ D9 I3 Y
return3 c* g) g- d& H, Z( q& _' ^
RegCreateKeyEx(hKeyRoot,pszPath,0L,NULL,REG_OPTION_VOLATILE,KEY_ALL_ACCESS,N9 i: Q0 E: U B. x
ULL,* D5 j4 _- W- p( T
&m_hKey,&dw);2 f1 s/ F4 K3 K$ Y2 j+ u3 r1 X* Z
} 1 l4 I3 m! y9 @4 S- Q, O
void CMyRegKey::RegClose(); x0 F( u+ n# D1 ]% O7 }, A
{5 B$ v) a3 w8 i7 E) i- x: K7 u
if(m_hKey)
% J6 l+ k9 O! v/ K{
4 m* j# @! o, R" `: lRegCloseKey (m_hKey);
* |* X0 B& F: M% cm_hKey = NULL; ! i$ S# R6 r) P) T2 t
}
D# Q/ f1 j' _) F# {) q8 p} + j* f( M, M2 c- I1 @0 x
LONG CMyRegKey::RegWrite(LPCTSTR pszKey,DWORD dwVal)& [5 d# u! L7 Z' a4 u1 T* z
{ ( H% Y8 d/ U: e* J+ L+ h
ASSERT(m_hKey); " u& Z+ S7 o7 W3 P
ASSERT(pszKey);
; s5 w% @8 d! A+ ?3 Z8 v* XASSERT(pData&&dwLength>0);
! K4 q+ t6 Q& wASSERT(AfxIsValidAddress(pData,dwLength,FALSE));
$ S/ ^4 k# B, X. G4 ^; ]- Yreturn RegSetValueEx(m_hKey,pszKey,0L,REG_DWORD,(CONST BYTE$ G/ {6 M3 ? A3 w$ }9 ^, A
*)&dwVal,sizeof(DWORD));2 X+ p8 {1 x# f* i1 O
}
9 R) x2 Z0 o+ [( o6 K. O2 M5 GLONG CMyRegKey::RegWrite(LPCTSTR pszKey,LPCTSTR pszData)
S4 D; z: d9 M, l7 j{ 6 z4 R! u# @5 e0 P& M7 i3 P" j
ASSERT(m_hKey);% D. l, s I/ u: V5 f/ J1 U/ ]
ASSERT(pszKey);
/ |& D3 v; F% I5 T% aASSERT(pszData); 7 P1 { J4 F; u& ^6 K- b7 h# P
ASSERT(pData&&dwLength>0);
- o7 Z) d$ B4 NASSERT(AfxIsValidAddress(pszData,strlen(pszData),FALSE));
9 h4 W7 n. p. i$ M( yreturn RegSetValueEx(m_hKey,pszKey,0L,REG_SZ,(CONST BYTE *)pszData,strlen7 j% H( V* m# U+ o: C' o- V
(pszData)+1);
: A& i" [$ [) b" C}
% u1 L% h( G1 n9 ?) t/ @LONG CMyRegKey::RegWrite(LPCTSTR pszKey,const BYTE *pData,DWORD dwLength)) `! K/ f' D! i* z
{ 3 w! \. N2 }4 _6 f
ASSERT(m_hKey);) @- h0 j Z: R" B: M `9 y5 Y
ASSERT(pszKey);
/ k. E" q; F8 IASSERT(AfxIsValidAddress (pData,dwLength,FALSE)); % D* n O* N! |1 t2 q
ASSERT(pData&&dwLength>0);
/ [* |$ ^6 Q, o) ^# A8 q" r- `ASSERT(AfxIsValidAddress(pData,dwLength,FALSE)); 7 u- |$ Y3 u1 l2 o# _& l0 }* i
return RegSetValueEx(m_hKey,pszKey,0L,REG_BINARY,pData,dwLength);
1 h. w) @# b7 k# E# {9 v; p$ D}
- B( }5 f. w: |& q+ ^) sLONG CMyRegKey::RegRead (LPCTSTR pszKey,DWORD& dwVal)" f/ I, ~, R4 t, }, q1 Q. t
{ ' c- X& Y0 `4 b* p5 A
ASSERT(m_hKey);' z( j& n; i; B! W/ k3 e
ASSERT(pszKey); % Q0 ` f* o8 `# i; P0 Q4 t; `# ]
DWORD dwType;
' P6 X3 G _& tDWORD dwSize = sizeof (DWORD);
1 h# o2 H+ Q; N- }9 lDWORD dwDest;
( r/ c7 ]3 n1 |0 m# b1 j+ \! XLONG LRet = RegQueryValueEx(m_hKey,(LPSTR)pszKey,NULL,&dwType,(BYTE *)
# j( T$ J' S/ m# i: u&dwDest,&dwSize); ! \+ l; t0 Q/ s* e, w
if(LRet==ERROR_SUCCESS)
+ ]3 R5 C& U" P$ B3 f. pdwVal = dwDest;( H$ D8 V6 l5 w" ~' N8 ~5 \
return LRet;
. f3 A) q& p6 i" E7 }! V}
+ j3 L/ t! v5 P- q5 ELONG CMyRegKey::RegRead (LPCTSTR pszKey,CString& sVal)
9 c% k/ a1 y0 U/ u{
5 L8 e' {. |5 S" B. M) z. cASSERT(m_hKey);2 [* t0 M+ m& I f) T: w
ASSERT(pszKey);
t6 V( g) } w) _, E9 I! dDWORD dwType;
+ z3 y% o6 `, p& u( U% J1 }DWORD dwSize = 200;
8 X6 B* Z% ~' Y7 s) achar string[200]; : P/ [1 D: k6 }% j/ W
LONG IReturn = RegQueryValueEx(m_hKey,(LPSTR)pszKey,NULL,&dwType,(BYTE *)5 ?1 u% X% c: H7 o! j% ~
string,&dwSize); ' z; a( e4 [' E% l
if(IReturn==ERROR_SUCCESS) & S" J' A T: ?( p: v
sVal = string;
% C3 m* N# [" O0 T$ y7 Vreturn IReturn;
) g' a1 _( P3 ~} 0 @1 Z$ W6 O& q* Q" U5 `% p, I9 `* ]
LONG CMyRegKey::RegRead (LPCTSTR pszKey,BYTE * pData,DWORD& dwLen)) m. v% T* j8 {" ^' K
{ " t7 a$ q; Z8 X
ASSERT(m_hKey);" X* ^" b4 \3 _3 `
ASSERT(pszKey);
8 h- j- s% y+ o8 u& LDWORD dwType;% w$ {6 m! N4 C' ~" l
return RegQueryValueEx(m_hKey,(LPSTR)pszKey,NULL,&dwType,pData,&dwLen); ' Q" Y% z4 Q- G! U# ~ w( l5 R5 X
}
0 |. j* F: f2 I在用户需要使用时只需在你的 Project 中的 SorceFile 和 HeadFile 中分别加入
9 }: a9 y- s0 t% `8 f- ?5 KMyRegKey.cpp 以及 MyRegKey.h 程序文件。
. Y0 f* e& }" x; D四 . 使用外部接口示例
2 h* m" e _4 O1 P
7 r' e8 c* {: N1 ^' ]7 _; F8 c在VC中建立一个基于对话框(Dialog Base) 的应用程序,在对话框上放上几个Edit
$ K/ R% P4 R% D( E% u( r* v' Ucontrol 的控件,如同示例小程序 RegTech 框(见图一), 程序执行时,首先读出注
' {5 f" P4 Y! c1 \) X5 d m% x册表信息分别显示在三个编辑栏中,为了演示写入操作,你可以在注册用户栏中重新输
1 m$ D9 Z% K$ f$ m, V/ B9 Z& ]入用户名,按更改完成写入,重新运行程序,查看写入是否成功。在RegTech 框中安置
0 [4 v' u O0 ?( o了三个编辑栏,ID 为 IDC_INSTALL,IDC_USERID,IDC_VERSION,
4 a: d- L4 Y9 x用ClassWizard 的Member Variable 分别加上对象参数:m_Install, m_UserID 和
6 D' H& v+ Y2 f6 a: b, lm_Version.4 Z s, |7 s q! D! |- b* }
用参数来传递注册表键值。 4 ?9 z% |& D) o
在初始化对话框时就应打开注册表并读取所需的信息,这三项存放路径为) g( q, P8 W+ v2 F2 |0 a2 G
HKEY_LOCAL_MACHINE \SOFTWARE \Microsoft \Windows \CurrentVersion下,
1 F2 N- O# A" T+ U3 P0 n$ o分别读出windows 版本号(放置于Version键值中),注册用户名(放置于
" f& O+ D; k" j, HRegisteredOwner键值中),Windows 安装目录(放置于SystemRoot 键值中 ),更多
/ Y7 H1 P8 p: J& P: H/ B的信息请使用Windows 目录下的 RegEdit.exe程序. 7 f$ H8 F$ B( c. X L, e# S1 m6 X! c
需用到注册表类的原程序文件中加上 #inxlude " MyRegKey.h" 即可。9 G; N |* t* n1 X
在文件 RegTechDlg.cpp 中初始化对话框的地方加上以下代码打开路径并读取键值: . R- Y4 X* O* n0 B+ B
BOOL CRegtechDlg::OnInitDialog()
" J2 m8 }+ z; u2 T( r# W/ S{ 2 X, M y6 _2 j/ J
CDialog::OnInitDialog(); 1 o( V* {9 i. q7 ~' @; b
......................# l7 y6 H" V$ ^3 Y3 P7 I
...................... 3 F/ X/ k( i4 N, ]' [( H
// TOD Add extra initialization here ( S& \# d2 Y4 p4 }2 g! s
. b0 x1 X, h7 {. s) c! w
file://###########################################
' U* _9 M! z6 X5 u: Y+ l l# [% K8 w8 E// 打开注册表路径 . . E6 A* S4 L7 d* n$ O1 h) _1 V& E
CMyRegKey regKey1,regKey2,regKey3;
4 n; Z; w. I/ J# eregKey1.RegOpen(HKEY_LOCAL_MACHINE,- l1 @; q7 P" D( q# B4 f
_T("Software\\Microsoft\\Windows\\CurrentVersion"));! l: q. { ^4 y D* y' I
regKey2.RegOpen(HKEY_LOCAL_MACHINE,
( `# ~. S! d% `& k' _) _5 L3 {_T("Software\\Microsoft\\Windows\\CurrentVersion"));
- i, T+ M# }# d: z( j9 f2 DregKey3.RegOpen(HKEY_LOCAL_MACHINE,7 v2 j* `+ }# b+ @$ v
_T("Software\\Microsoft\\Windows\\CurrentVersion")); 2 n! S) E# F6 E/ O
' p0 e: Q) W* V
// 读取键值并分别传递到编辑控件参数中去:2 @6 Q0 x8 O4 g) A F2 e
regKey1.RegRead(_T("Version"),m_Version);
7 S% C3 B2 V2 Q/ k( cregKey2.RegRead(_T("RegisteredOwner"),m_UserID);
5 ~ W! g( Z- Q- z9 O' e/ PregKey3.RegRead(_T("SystemRoot"),m_Install);* j& r, R' y# T/ |( K
UpdateData(FALSE); : q# m- S3 M, H+ e" \# Y9 N) Q
file://############################################ . m" y$ I# G1 c
return TRUE; // return TRUE unless you set the focus to a control; k5 \# K* X% [+ K: p
} % Y4 _" @7 m1 D3 c8 C
在重新输入了注册用户名后,按下更改按钮(按钮控件ID_ChangeID),将完成注册表
2 z) i8 w* p' U0 x. `) h的写入操作:
8 V& m/ l# z: h* @( k4 Kvoid CRegtechDlg::OnChangeID()! A: V, y! I; _9 O' G1 E) j' C
{ 1 [7 ^! J0 q% n" B' l9 `. Q/ G0 Q
/ `8 N+ y, v0 P/ }+ T! V6 t
file://########################
! S- i6 ~. H8 |% A; _CMyRegKey regKey;
3 \- p" ]0 _9 S9 y
6 {; s3 G' C2 GregKey.RegOpen(HKEY_LOCAL_MACHINE,( j+ y; L5 |# o
_T("SOFTWARE\\Microsoft\\Windows\\CurrentVersion"));
! [9 ~; q5 t% X( E( L& K" _# S// 传递从编辑拦输入的值并写入打开路径的指定键值名中:
' n6 R; ?: d& s1 r. X' D2 TUpdateData(TRUE);
9 w6 [: a$ c, K! H/ M! r+ TregKey.RegWrite(_T("RegisteredOwner"),m_UserID);
& i* D0 b5 @. z; ~9 ]; [" g% w. ]MessageBox("更改注册用户名完毕!","Message",MB_ICONINFORMATION|MB_OK); , H6 w, w, p' u7 S
9 @8 F$ }3 w! k2 D3 M3 h7 w///#########################
9 {5 w0 i; U$ K* r0 r, b8 b
: H, W U. m8 i// TOD Add your control notification handler code here
9 y; M" w& @ I) A d1 ~} " `1 J& d% K. g6 R& T3 ?
注册表类在实际应用程序编制中使用频率极高,比如程序员自定义的文件的图标选择等
, R. R; }% e! Z/ o: l的很多地方都要用到,所以一个成熟优秀的应用程序应尽可能的适应环境及发挥操作系8 j* E3 @: _8 ?$ J) \' Q
统的各种优势,在此,使用win32 API对Windows灵魂--注册表存取技术的操作已完
1 ]- ]! r$ M6 l- K* q4 M) n2 [( b,你就可以利用它来完善自己的应用程序了。 |