|
一、问题的提出 9 g2 J) ]8 p. e; H, P% u
Windows 已由原来的16位 windows 3.x 升级为现今我们使用的32位windows 95/97/98% Y# T! `4 \$ I- k2 U0 U3 t
以其 Windows NT,用户不仅在使用上应逐步适应,对于程序开发人员来说在编程技术
+ D9 Q) b3 n3 c上也应紧跟操作系统的技术发展,就如同 在 Linux 操作系统下, X-Window 编程就显- a2 E- p6 v' f% b$ x
得很重要一样。作为一个完整成熟的 Windows 程序,需要保存程序所有的环境变量和
+ \& I' J: Z8 ~( X6 o [ m9 V私有信息。诸如用户的偏好,文件装入的列表、退出时用户使用的窗口位置 .存盘历史
0 F+ z5 B8 J6 K! ^$ N纪录等。过去在 windows 3.x时代 ,常用 Win16函数 Get/RegWrite ProfileString9 ]6 ^! J9 Z+ A
将有关程序的信息写入 *.ini 文件,但现在该项技术由 Win32注册表所替代。
1 L6 x8 f: N! }; r+ g可以这样说,注册表是当今32位 Windows 操作系统的灵魂,一切信息都在其中,也就
2 s' K2 \0 V% }$ m是为什么Windows98在Windows95的基础上升级可以不重装软件等等( M4 E6 x- W5 L* `6 W+ }
的如此方便的应用,其原理就是根据了原注册表中的信息来完成各种方便的处理,所以( Q8 l8 a k% H: G8 ]& l" t
Windows注册表对应用程序的重要性就显而易见了。% J" s; g+ P0 A
原来的 Win16程序存储私有信息是在一个平面文件INI中,这样做有很多弊端,例如该# R/ s0 E, q' L- M( j$ T: M
INI文件没有任何安全机制,用户可以直接在INI文件中修改各种参数和程序入口,这样
7 d+ n: N/ T% C0 z! T N# Q! i9 V( d就可能造成不可估计的严重后果,还有该文件只能支持和文本数据不能存入二进制数据! {, ?( \: ~5 j9 e+ A6 s! S
等各种不利因素,所以微软的工程师也认识到这一点,于是注册数据库就诞生了,注册
* X1 F# D$ @+ K& C$ Q M$ ]数据库就是为了解决在 Windows 3.x 的一些关于 OLE 的此类问题而创建的,现在
2 _' ~3 J Q0 n; c/ [Win32 应用程序的注册数据库通过微软带给我们的新的 Win32 API 得到了显著的改
9 s4 X5 B) }: m7 ^善。使用访问注册表的 Win32 函数比起使用管理 INI 文件的 Win16函数要灵活的多,
8 Z" q3 V$ P+ H5 j! F$ C0 T这意味着在功能上将大大增强,但是,另一方面,如果你还未用过,就会对处理注册表% t" F/ W. H8 D4 g
的Win32 API 的新规则感到困惑或不知所措。本文就是本着这一目的,逐步让你懂得并
( P! j' o5 ~6 I6 j5 m4 T" p' b掌握怎样用 Win32API 函数来处理32位 Windows 程序注册表的方法。
/ e4 ?4 ^% m( K6 L4 }* m! T* k9 r. H" b! i$ q: S5 z9 t& b, g" C; b
二 . 技术的实现原理 * U% B3 A' ~$ e) B& T8 n# i
为了在以后自己编写的程序中更多的体现模块化思想以及使编程变得更加简单,应尽可
B' c0 s+ i& [! P. } g6 V* I6 N: w能的建立自己实现各种功能的类,以类作为实现应用程序各种功能的单位。 在此,可
" r, H) f1 g+ Z以创建一个包括注册表许多常用功能而接口简单的类库,下面将建立 CMyRegKey类,对
+ I! ]! E+ M/ a3 y% n5 w8 T" ^应用程序处理注册表的具体细节进行封装,从而在外部通过这个功能类方便地实现进行7 @) r. \0 Z0 r2 l
访问注册表信息的各种操作,在外部调用其成员函数即可。以后,你就可以在每一个应- D# f3 t- s! V9 O0 L
用程序中包含此类并用其外部接口进行编程了。 ! q, Y, D0 j, E* m. B. b4 U: e
三 . 实现代码与步骤
% W3 i+ F' Z2 J M4 ?( E% A4 l( }) p8 S9 G' ?) c I( H
1. 建立功能类的头文件:
4 c* D2 i+ X* X! X( ~创建一个新的头文件 MyRegKey.h ,在其中加入以下的代码。 6 l1 ^" L" _" v+ u/ \2 }1 i# P
. M7 k7 P; t) m. J1 d$ n- p7 G
#include "winreg.h"
. b2 b9 u7 [. z* V9 |// 包含头文件 winreg.h , 因注册表Win32 API 函数在其内定义
: H" p& b* J! j1 Y// 建立 CMyRegKey 类:
# W) Z( m, P+ Q5 Xclass CMyRegKey1 v) t4 c4 B& g; Z2 e6 x
{
: j9 H+ d0 F8 t4 u- {7 Z: j; ]# X1 U9 j# c9 i7 _3 T/ [
// Construction* O& h; n6 \4 n, ~ J% y
public:! A) `- L+ K# h
CMyRegKey();" w, R0 X4 h& q1 a1 e/ n
virtual ~CMyRegKey (); ! |. a8 g+ q8 K* [$ G
// Attributes* S5 J' S% i, g5 X, V: I
public: 5 H/ T ^$ }- m! L; c
// 定义打开和关闭注册表的成员函数:
- p# P" ` f, \9 w/ w/ u% L, |LONG RegRegOpen(HKEY hKeyRoot,LPCTSTR pszPath);8 X0 \2 z2 v5 c9 z) J) a& B
void RegRegClose();
8 @/ g% i: M, `# O// 利用函数重载实现对注册表键值(串值,二进制值,DWORD值 ) 的读和写:
4 M8 Z; t$ T( U# N; e: OLONG RegRead (LPCTSTR pszKey,DWORD& dwVal);" Q5 j* j, c: Y6 o$ R' \$ y$ ^
LONG RegRead (LPCTSTR pszKey,CString& sVal);* p/ R. b( w1 E1 b1 I$ m
LONG RegRead (LPCTSTR pszKey,BYTE *pData,DWORD& dwLength);
8 ~" M$ X( V$ p- ZLONG RegWrite (LPCTSTR pszKey,DWORD dwVal);7 R( O+ d6 E2 h4 Z8 V# k
LONG RegWrite (LPCTSTR pszKey,LPCTSTR pszVal);1 O. W _7 t( F/ f* r
LONG RegWrite (LPCTSTR pszKey,const BYTE *pData,DWORD dwLength); 7 |7 h* x$ ?2 e
protected:4 u% P! g5 U8 W$ u+ b- e
HKEY m_hKey;
( }3 L7 u# _+ tCString m_sPath;
9 o0 W0 V' O6 P, p+ d" b}; - T. ^0 r& _: M
2. 建立功能类的Cpp文件定义 CMyRegKey类: , Q+ o I* I: ?4 c
创建一个新文件 MyRegKey.cpp ,代码如下:
2 \8 w0 V9 v d! J* x% w: j, b' |& U6 j) k/ r) C6 i
#include "MyRegKey.h" 6 {# B& A. z8 V1 A9 T& `
///////////////////////////////////////////////////////////////////////////// E) ]3 R/ ]2 Y: k" \
/
7 }4 S' U" ~: t+ B// CMyRegKey
. t) g( z8 ^+ {, f////////////////////////////////////////////////////////////////////////////3 O1 T" j! o4 c1 }: A
/
9 G9 K8 A$ R8 Q6 }! ]' Q' zCMyRegKey:: CMyRegKey()- c1 m F% i2 z+ ?$ b
{/ f) q4 x3 k* L
m_hKey = NULL;
2 z1 X( s' u8 L) }& |} ; |4 R2 c# _# s0 y( l5 d
CMyRegKey:: ~CMyRegKey()5 X0 s& K4 ^3 C+ N* [0 L
{& X1 h( Q9 V$ \; P
RegClose();) S, h. g* M9 C. Q7 m
}
- t" ]! C+ A1 [$ @$ J8 V8 ~// 定义打开注册表的函数,RegOpen 函数带有两个参数:指定要访问注册表根结点的- L n( G. y4 l/ x( Y; V
HKEY, 以及注
( B, o6 k4 F. U9 U, ?+ w: E// 册表中信息的全路径。如果给入的路径不存在,则需创建一个新路径。从
% k0 Z+ Y& g0 KRegCreateKeyEx API函数返// 回的 HKEY 作为 m_hKey 存储。
+ s% e& W, f2 { z' B' Z' C4 WLONG CMyRegKey::RegOpen(HKEY hKeyRoot,LPCTSTR pszPath) y+ h1 ~7 f t& s
{
, L. Z5 G1 w+ Q0 E/ j% gDWORD dw;2 I% N# d! E% G% P* z4 ], ^( @
m_sPath = pszPath;
; e+ b: k2 P7 H. jreturn
5 s+ u2 ^% Y$ R- bRegCreateKeyEx(hKeyRoot,pszPath,0L,NULL,REG_OPTION_VOLATILE,KEY_ALL_ACCESS,N
7 T' r, d" h P; e' ~+ U. @. iULL,8 G$ _5 e$ l G. ~* Q# ^' U9 V$ y* Z
&m_hKey,&dw);* j+ k+ ~$ d* s; S4 W3 K
} # P& G- k" ~3 ]7 w& }
void CMyRegKey::RegClose()4 ]' c8 `* P7 E" ?% b
{5 A \6 q- P1 }: n# L- r5 c3 o
if(m_hKey)
" k$ ^6 q2 S$ e{4 ]7 i1 O& X$ \; E. [3 ^
RegCloseKey (m_hKey);, f) t% J5 @1 B' p9 a
m_hKey = NULL;
/ X; ^# x! M2 l* V* K7 c}+ I d: g) p# H0 u8 Y7 t5 P
}
4 G H7 j+ {6 |# e5 CLONG CMyRegKey::RegWrite(LPCTSTR pszKey,DWORD dwVal)
X, l$ d5 a) y* ]8 n o{
$ K% q* J& o2 X5 J0 J4 \ASSERT(m_hKey); ) r4 @' I$ W, X
ASSERT(pszKey);
/ F7 {4 c" ^5 D& y: r! |9 W# MASSERT(pData&&dwLength>0);
5 r& L8 E x( c3 C, i3 R" O3 l0 dASSERT(AfxIsValidAddress(pData,dwLength,FALSE)); : ]% O) T! l" N: ^
return RegSetValueEx(m_hKey,pszKey,0L,REG_DWORD,(CONST BYTE
; }. r9 S8 |" @*)&dwVal,sizeof(DWORD));& j* A2 C+ l; Q/ I2 h' w' @
} , o1 K) I: K5 Y" X i; x6 y3 j
LONG CMyRegKey::RegWrite(LPCTSTR pszKey,LPCTSTR pszData)
. C9 [8 q! l* S1 k5 f6 O- s7 T{
! f! m( W! l0 J; D& d" y% KASSERT(m_hKey);& p- j1 |7 M5 k- p
ASSERT(pszKey);
% }% b2 _8 P: L d( ]% jASSERT(pszData); 1 m& v3 @& w, S# R5 N! t; K" t
ASSERT(pData&&dwLength>0);
7 I/ m9 C) x* C7 ~& s- SASSERT(AfxIsValidAddress(pszData,strlen(pszData),FALSE));
5 c. j# \* Z: N3 Kreturn RegSetValueEx(m_hKey,pszKey,0L,REG_SZ,(CONST BYTE *)pszData,strlen2 S" @* P: u' g2 J) L: x3 b& W% ?7 A; F
(pszData)+1);
, W1 ]8 H5 p8 Q( I& a}
. O' h% ?/ \3 Y( f2 y% BLONG CMyRegKey::RegWrite(LPCTSTR pszKey,const BYTE *pData,DWORD dwLength)
9 n$ g5 `$ u9 H& G% Y8 l{ `* U, g4 q0 K; [8 R1 B1 F+ M" C
ASSERT(m_hKey);
' u. l6 ^/ |8 eASSERT(pszKey);
& d/ Z+ c; L; X+ `, D* U: ]ASSERT(AfxIsValidAddress (pData,dwLength,FALSE)); # z. D) x# y3 b, i7 R
ASSERT(pData&&dwLength>0);
8 m( F3 P9 W8 [' C+ WASSERT(AfxIsValidAddress(pData,dwLength,FALSE));
! Y5 p, f0 L5 ~- O9 n. ireturn RegSetValueEx(m_hKey,pszKey,0L,REG_BINARY,pData,dwLength); 6 P& y5 M9 V, U) d
}
2 Z# R# a( o9 oLONG CMyRegKey::RegRead (LPCTSTR pszKey,DWORD& dwVal)3 h. ~" Z# c& k" g5 q+ m' t
{ 3 H. @) A% w' `. G
ASSERT(m_hKey);
7 A a! M2 \* x) wASSERT(pszKey); % T/ ?4 m/ c4 D2 m% s( B3 L/ Q
DWORD dwType;
2 N; i. d6 a( N3 h" g2 ^DWORD dwSize = sizeof (DWORD);
, k* _" l4 l6 s0 s0 R3 Z YDWORD dwDest;
1 h9 L l: G' ZLONG LRet = RegQueryValueEx(m_hKey,(LPSTR)pszKey,NULL,&dwType,(BYTE *)
c+ M& o U0 j( s1 C, F' X&dwDest,&dwSize); : V/ l% g0 v" s8 s9 }& c. i6 U
if(LRet==ERROR_SUCCESS)+ n0 e: `7 j: {
dwVal = dwDest;
9 D M' P0 J6 p& \ ?return LRet; - X& w9 p3 w) b" X9 ]
}
; j* C; P6 Z- f: P" h% [7 ]1 d+ K9 aLONG CMyRegKey::RegRead (LPCTSTR pszKey,CString& sVal), z+ L7 P$ e# q8 x$ h# {0 B+ _* C
{
) y4 W4 p3 V1 _ASSERT(m_hKey);
% C' {' u- i: i, YASSERT(pszKey);
0 V* |: Q' c3 y7 vDWORD dwType;
. s, t/ x, v8 L' h. C, ^DWORD dwSize = 200;. ]" q$ ?6 M4 Y: \. H7 Q1 @* A
char string[200]; ( P; c3 t2 X% c' j8 f
LONG IReturn = RegQueryValueEx(m_hKey,(LPSTR)pszKey,NULL,&dwType,(BYTE *). J7 M# h# O( W
string,&dwSize);
& `+ d5 l) a: T# S* m5 ^+ j. pif(IReturn==ERROR_SUCCESS) 6 Z a$ U8 K5 d2 P
sVal = string;
@" q/ s( B: A, C1 a3 ]8 `return IReturn; $ R, U( s$ X! n* c9 I: k- ^
} ) Z( y5 O: B2 c3 I, N, ^% j( `
LONG CMyRegKey::RegRead (LPCTSTR pszKey,BYTE * pData,DWORD& dwLen)
6 R, D6 l+ R) b: r" {{
! r, \" n8 x+ N( o. ]$ RASSERT(m_hKey);
* R# ]' p" D( L8 ^9 u( RASSERT(pszKey);
, j/ |: ?" C: V; p0 ?/ PDWORD dwType;
$ h& w8 o3 q# d- jreturn RegQueryValueEx(m_hKey,(LPSTR)pszKey,NULL,&dwType,pData,&dwLen); + c% L5 m- x% m6 [7 a
}
/ Z) |! ?/ f5 Q4 P8 x在用户需要使用时只需在你的 Project 中的 SorceFile 和 HeadFile 中分别加入
/ S6 b6 D* d9 U4 d, RMyRegKey.cpp 以及 MyRegKey.h 程序文件。 : v" F; s$ a% x0 d# _" M% e; G
四 . 使用外部接口示例 + H8 F, I T. x
/ \$ Y' w: ?" J
在VC中建立一个基于对话框(Dialog Base) 的应用程序,在对话框上放上几个Edit: Q' s. @# f* C. ^
control 的控件,如同示例小程序 RegTech 框(见图一), 程序执行时,首先读出注
6 K* H* Q2 i2 ?& G9 v册表信息分别显示在三个编辑栏中,为了演示写入操作,你可以在注册用户栏中重新输; C c, t0 v6 j! O4 {7 y
入用户名,按更改完成写入,重新运行程序,查看写入是否成功。在RegTech 框中安置
- c4 U5 o! S/ p- P了三个编辑栏,ID 为 IDC_INSTALL,IDC_USERID,IDC_VERSION,+ v5 c! u) R/ U3 X f8 i7 I3 _
用ClassWizard 的Member Variable 分别加上对象参数:m_Install, m_UserID 和
( J/ @9 J8 J; h4 [m_Version.
: Y' w% I Z# T" z( R/ h用参数来传递注册表键值。
2 b4 y3 e4 l8 A& X* f; ]/ Z在初始化对话框时就应打开注册表并读取所需的信息,这三项存放路径为3 B, X9 U' h+ E7 \6 c
HKEY_LOCAL_MACHINE \SOFTWARE \Microsoft \Windows \CurrentVersion下,$ Y0 M$ |- M) A* E: z
分别读出windows 版本号(放置于Version键值中),注册用户名(放置于
) r K y$ G! r. B6 `/ u# ARegisteredOwner键值中),Windows 安装目录(放置于SystemRoot 键值中 ),更多
# y, i/ ~. |2 p* D8 k e& D: L8 T8 q的信息请使用Windows 目录下的 RegEdit.exe程序.
" s0 K! E# t# y' O: j需用到注册表类的原程序文件中加上 #inxlude " MyRegKey.h" 即可。( ^% z6 w# H# M! ~
在文件 RegTechDlg.cpp 中初始化对话框的地方加上以下代码打开路径并读取键值:
' I5 C9 c: I/ K& V0 t1 d" zBOOL CRegtechDlg::OnInitDialog()8 L* i- ]& h5 `2 B/ }, u1 F- x
{ 4 D$ @' V2 I0 O# y- B+ @$ C7 L( n7 w
CDialog::OnInitDialog();
( X3 T# z/ I& a6 S' p9 r* K6 I......................8 f2 ]( g3 p% Z7 b. [. K7 B
......................
7 z8 u; b4 k% @; O1 T$ s; o// TOD Add extra initialization here 5 y3 z7 p: m d* q5 r. I
5 T; |' F) z+ ]$ S& x- \8 Dfile://########################################### 7 @9 X7 R* o0 _* i+ Q8 s5 M
// 打开注册表路径 .
4 @- i* e% X- y& B$ x$ V3 CCMyRegKey regKey1,regKey2,regKey3; : ~! i0 k( I0 s; b( v7 [" B
regKey1.RegOpen(HKEY_LOCAL_MACHINE,! X0 e' Q) D4 E' s' Y$ W+ r: J
_T("Software\\Microsoft\\Windows\\CurrentVersion"));
1 P: _9 y; M. S0 v4 AregKey2.RegOpen(HKEY_LOCAL_MACHINE,) R, s2 |; H# \. ?
_T("Software\\Microsoft\\Windows\\CurrentVersion"));
- U1 j: E# d/ I+ E [& Y2 |7 dregKey3.RegOpen(HKEY_LOCAL_MACHINE,
9 i. r' p7 d& {7 l+ n/ h_T("Software\\Microsoft\\Windows\\CurrentVersion")); 1 _/ S2 ?; |. D
4 x; e9 d( P0 n1 N
// 读取键值并分别传递到编辑控件参数中去:
' L8 O6 D7 F% p9 N7 c* XregKey1.RegRead(_T("Version"),m_Version);% q# |2 ~1 o. o6 Q) g; u
regKey2.RegRead(_T("RegisteredOwner"),m_UserID);% I. ]2 R6 {4 y- @8 Y* k. w
regKey3.RegRead(_T("SystemRoot"),m_Install);6 b( t1 m# k. [9 b
UpdateData(FALSE); ! o0 N8 w/ U. f
file://############################################ / }8 O" B7 J4 b& |/ r
return TRUE; // return TRUE unless you set the focus to a control# C* F; @ Q$ R$ z, t/ e8 F
}
' N2 m( q# @" P9 k. t" G/ e在重新输入了注册用户名后,按下更改按钮(按钮控件ID_ChangeID),将完成注册表# y; L% ~- B- A- r. w) @
的写入操作: . D& c8 l* z8 W
void CRegtechDlg::OnChangeID()" m0 w/ L, y, M* C/ W
{ 8 N1 h& W* z) R1 w) y( F, c5 B- V
( N! i: _7 S$ Q1 L) V( ~* s: `6 q
file://########################
5 W' B0 v( _4 i, f( ?/ YCMyRegKey regKey; 5 D% _* A8 W/ T8 y) s
/ C$ x" E4 L% X
regKey.RegOpen(HKEY_LOCAL_MACHINE,% R! v. `. H" H$ R
_T("SOFTWARE\\Microsoft\\Windows\\CurrentVersion"));
/ O$ f: k! s2 ^' m( I7 n# I0 q: g// 传递从编辑拦输入的值并写入打开路径的指定键值名中:
" V9 `( u. E) lUpdateData(TRUE);
2 d* _, ?" D) P# `4 a( hregKey.RegWrite(_T("RegisteredOwner"),m_UserID);
6 m7 c4 D4 x% Z5 \' E$ }& C- FMessageBox("更改注册用户名完毕!","Message",MB_ICONINFORMATION|MB_OK); & E6 r6 b- l4 Y! M# {' S" n1 }9 k3 m
5 }* `( x) A( L8 K* U
///######################### 1 ^0 q: \7 `9 h* X- X% g5 }
0 Y; ?! I% z3 r3 `- `2 C0 w Z, b// TOD Add your control notification handler code here
2 W8 G7 E& }' L& ^; H! V}
: n6 m- a+ ^0 i4 P( ?注册表类在实际应用程序编制中使用频率极高,比如程序员自定义的文件的图标选择等+ m# k" i' j: S1 ?2 m/ }7 w& {
的很多地方都要用到,所以一个成熟优秀的应用程序应尽可能的适应环境及发挥操作系/ _& s+ V4 U0 r0 g2 @
统的各种优势,在此,使用win32 API对Windows灵魂--注册表存取技术的操作已完- q& Z7 R" B2 y4 ~
,你就可以利用它来完善自己的应用程序了。 |