摘 要 介绍了Windows磁盘清理工具二次开发的扩展接口,对其COM接口加以分解,并运用ATL库具体实现了清理“*.tmp”临时文件的功能。
关键词 磁盘清理工具、ATL库、COM接口。
引言
Windows磁盘清理工具(Disk CleanUp)是一个实用快捷并拥有简单易用界面的系统清理软件,更值得系统开发管理人员注意的是,此系统清理软件是建立在以COM技术为基础发展的,支持第三方插件,并且可以根据需要自制定义功能二次开发的平台。在这里,我们对于Windows磁盘清理工具的开发接口做深入地研究,在此基础上举例示范添加一个查找“*.tmp”临时文件的功能。
技术讨论
微软的COM技术广泛地运用在Windows的模块化设计中,致使支持二次开发。关于COM技术基础与应用,可参考。在此,我们只为Windows磁盘清理工具,简称清理工具的扩展接口加以分解。清理工具首次出现在Windows 98操作系统中,并在后来推出的Windows版本中予以改进,添加了新的功能。比如说,在NTFS的文件系统下,自动压缩不经常访问的文件。这些新功能通过COM模块实现,在清理工具中作为插件调用。早期的版本是通过IEmptyVolumeCache接口调用,在Windows 2000以后的版本中,还加入了IEmptyVolumeCache2接口,加入了较小的更新。
IEmptyVolumeCache接口由五个函数组成,根据呼叫的顺序,分别是:
virtual /* [local] */ HRESULT STDMETHODCALLTYPE initialize( /* [in] */ HKEY hkRegKey, /* [in] */ LPCWSTR pcwszVolume, /* [out] */ LPWSTR *ppwszDisplayName, /* [out] */ LPWSTR *ppwszDescription, /* [out] */ DWORD *pdwFlags) = 0;
virtual HRESULT STDMETHODCALLTYPE getspaceused( /* [out] */ DWORDLONG *pdwlSpaceUsed, /* [in] */ IEmptyVolumeCacheCallBack *picb) = 0;
virtual HRESULT STDMETHODCALLTYPE showproperties( /* [in] */ HWND hwnd) = 0;
virtual HRESULT STDMETHODCALLTYPE purge( /* [in] */ DWORDLONG dwlSpaceToFree, /* [in] */ IEmptyVolumeCacheCallBack *picb) = 0;
virtual HRESULT STDMETHODCALLTYPE deactivate( /* [out] */ DWORD *pdwFlags) = 0; |
清理工具在正常执行时,首先调用Initialize初始化插件,随后执行GetSpaceUsed来扫描可清除的文件大小。扫描完毕后,清理工具的主界面就出现了如图1所示,在此,我们加入了清理TMP文件的功能可以浏览不同的清理文件种类。列表中的每一个文件种类均由一个COM插件实现。除了阅览可清理文件大小以外,用户在可以点击一个可自定义的按钮,调用插件的ShowProperties功能,以显示更详细的资料。如用户选择OK,清理工具就调用Purge函数,清理扫描出来的文件。最后,Deactivate函数被调用,终止插件的应用。
运用于Windows 2000以后的清理工具的插件也应该支持IEmptyVolumeCache的接口。IEmptyVolumeCache只由一个函数组成:
virtual /* [local] */ HRESULT STDMETHODCALLTYPE initializeex(
/* [in] */ HKEY hkRegKey, /* [in] */ LPCWSTR pcwszVolume, /* [in] */ LPCWSTR pcwszKeyName, /* [out] */ LPWSTR *ppwszDisplayName, /* [out] */ LPWSTR *ppwszDescription, /* [out] */ LPWSTR *ppwszBtnText, /* [out] */ DWORD *pdwFlags) = 0; |
InitializeEx增加了更严格的本地化语言要求,加强了国际化的支持,并且允许自定义按钮的显示文字。pdwFlags变量用于在工具与插件间传递信息,支持下列旗标:
EVCF_OUTOFDISKSPACE EVCF_SETTINGSMODE EVCF_DONTSHOWIFZERO EVCF_ENABLEBYDEFAULT EVCF_ENABLEBYDEFAULT_AUTO EVCF_HASSETTINGS EVCF_REMOVEFROMLIST |
EVCF_OUTOFDISKSPACE与EVCF_SETTINGSMODE用于工具传递给插件的设定。EVCF_OUTOFDISKSPACE表示当前硬盘的空余空间非常有限,需要尽可能多地清理,即使是系统的性能会受到影响。EVCF_SETTINGSMODE表示可定期执行的无人控制模式。在此模式下,GetSpaceUsed,Purge,及ShowProperties都将不予调用,所有清理任务应予InitializeEx时执行。其它旗标用于插件传递给工具的不同运行模式。EVCF_DONTSHOWIFZERO表示在没有找到可删除文件时不显示此类型,EVCF_ENABLEBYDEFAULT表示此类型文件可以安全删除,EVCF_ENABLEBYDEFAULT_AUTO表示此类型文件可以非常安全的删除,EVCF_HASSETTINGS表示此插件支持ShowProperties功能,可以显示详细信息。EVCF_REMOVEFROMLIST表示是一次性清理任务,清理工具在执行后自动将插件关闭,以后不再执行。
图1 清理工具的主界面
实现方法
我们开发一个新的清理工具插件,扫描并清理*.TMP文件。COM的编程有多种方法,我们选择了ATL库。关于ATL库的运用。
我们在Visual Studio .Net 2003中生成新的ATL的DLL Server项目,并使用Add Class加入新的ATL Simple Object控件类CCleanSimpleHandler.在定义中,我们让CCleanSimpleHandler从IEmptyVolumeCache2继承。并且,我们添加了下列变量:
// 储存扫描出文件的大小
DWORDLONG m_dwlFileSize;
// 储存根目录
WCHAR m_strRootDir[MAX_PATH];
// 储存扫描出文件列表
std::vector<WCHAR *> m_lstFilesToDel; |
然后,我们一一实现IEmptyVolumeCache及IEmptyVolumeCache2接口的函数。在下面的代码列表中,没有包括严格的检查错误返回值。这是为了简短代码的长度,提高可读性。在实际应用中,检查错误返回值是不可少的。为了不同版本Windows兼容,我们在InitializeEx中调用Initialize.
HRESULT CCleanSimpleHandler::InitializeEx (HKEY hKey, LPCWSTR pcwszVolume, LPCWSTR pcwszKeyName, LPWSTR *ppwszDisplayName, LPWSTR *ppwszDescription, LPWSTR *ppwszBtnText, DWORD *pdwFlags) { HRESULT hr = Initialize (hKey, pcwszVolume, ppwszDisplayName, ppwszDescription,pdwFlags); *ppwszBtnText = (LPWSTR) CoTaskMemAlloc (64 * sizeof (WCHAR)); StrCpyW(*ppwszBtnText, L"View files"); return hr; }
HRESULT CCleanSimpleHandler::Initialize (HKEY hKey, LPCWSTR pcwszVolume, LPWSTR *ppwszDisplayName, LPWSTR *ppwszDescription, DWORD *pdwFlags) { StrCpyW(m_strRootDir, pcwszVolume); *ppwszDisplayName = (LPWSTR) CoTaskMemAlloc(256 * sizeof (WCHAR)); StrCpyW(*ppwszDisplayName, L"*.TMP files"); *ppwszDescription = (LPWSTR) CoTaskMemAlloc (256 * sizeof (WCHAR)); StrCpyW(*ppwszDescription, L"Temporary files - *.TMP"); *pdwFlags = EVCF_HASSETTINGS | EVCF_ENABLEBYDEFAULT; m_dwlFileSize = 0; return S_OK; } |
在GetSpaceUsed中,我们调用ScanDir来扫描*.TMP文件,储存在m_lstFilesToDel中。GetSpaceUsed的第二个参数是IEmptyVolumeCacheCallBack接口的指针,用于调用其ScanProgress函数以报告扫描的进展情况。ScanProgress函数定义是:
HRESULT ScanProgress(DWORDLONG dwlSpaceUsed, DWORD dwFlags, LPCWSTR pwszReserved);
其中dwFlags正常应设为零,在结束时改为EVCCBF_LASTNOTIFICATION.ScanProgress函数的返回值很重要,因为用户可以在任何时候中断在进行中的清理任务。如ScanProgress返回E_ABORT,GetSpaceUsed应最快终端扫描,函数返回。因此,我们在递归的目录扫描函数ScanDir中,加入了如中断立即退出的功能。
HRESULT CCleanSimpleHandler::GetSpaceUsed (DWORDLONG *pdwSpaceUsed, IEmptyVolumeCacheCallBack *picb) { m_dwlFileSize = 0; ScanDir(m_strRootDir, picb); picb->ScanProgress(m_dwlFileSize, EVCCBF_LASTNOTIFICATION ,NULL); *pdwSpaceUsed = m_dwlFileSize; return S_OK; }
bool CCleanSimpleHandler::ScanDir(WCHAR * szDir, IEmptyVolumeCacheCallBack *pcib) { WCHAR strPath[MAX_PATH]; WCHAR* pchPathFileName; bool cancelled = false; WIN32_FIND_DATAW fd; HANDLE hFind;
if (cancelled = FAILED(pcib->ScanProgress(m_dwlFileSize, NULL, NULL))) return false; StrCpyW(strPath,szDir); PathAppendW(strPath, L"*"); pchPathFileName = strPath+lstrlenW(strPath)-1; hFind = FindFirstFileW(strPath, &fd); if (hFind == INVALID_HANDLE_VALUE) // E.g. Due to security issues return true; do { StrCpyW(pchPathFileName, fd.cFileName); if ((fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { if (fd.cFileName[0] != '.') { if (cancelled = !ScanDir(strPath, pcib)) break; } } else { WCHAR* pchExt = PathFindExtensionW(strPath); if ( StrCmpIW(pchExt, L".tmp") == 0 ) { m_dwlFileSize += ((DWORDLONG)fd.nFileSizeHigh)*4294967295+ fd.nFileSizeLow; WCHAR* filename = (WCHAR *)CoTaskMemAlloc((lstrlenW(strPath)+1)* sizeof(WCHAR)); StrCpyW(filename, strPath); m_lstFilesToDel.push_back(filename); } }
} while (FindNextFileW(hFind, &fd) != NULL); FindClose(hFind); return !cancelled; } |
其他的函数很简单。Purge函数将扫描出的文件列表m_lstFilesToDel中的文件一一删除。ShowProperties中,我们显示扫描出来的文件。最后,Deactivate将分配的内存释放。
HRESULT CCleanSimpleHandler::Purge (DWORDLONG dwSpaceToFree, IEmptyVolumeCacheCallBack *picb) { for (unsigned int i=0; i < m_lstFilesToDel.size(); ++i) DeleteFileW(m_lstFilesToDel[i]); return S_OK; }
HRESULT CCleanSimpleHandler::ShowProperties (HWND hWnd) { for (unsigned int i=0; i < m_lstFilesToDel.size(); ++i) if (MessageBoxW(hWnd, m_lstFilesToDel[i], L"View files", MB_OKCANCEL|MB_ICONINFORMATION)==IDCANCEL) break; return S_OK; }
HRESULT CCleanSimpleHandler::Deactivate (LPDWORD pdwFlags) { for (unsigned int i=0; i < m_lstFilesToDel.size(); ++i) CoTaskMemFree(m_lstFilesToDel[i]); m_lstFilesToDel.clear(); *pdwFlags = 0; return S_OK; |
结论和建议
通过实例分解,我们对Windows磁盘清理工具的基于COM技术的开发接口做了深入地研究。Windows外壳中有较多的开发接口,本文介绍的开发思想也可以运用在其它扩展插件中。