编程论坛

 找回密码
 立即注册
广告联系qq1031180668广告位
查看: 1833|回复: 9

c语言枚举进程以及遇到的编码问题

[复制链接]

963

主题

1684

帖子

3257

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
3257

最佳新人活跃会员热心会员推广达人宣传达人灌水之王突出贡献优秀版主荣誉管理论坛元老

发表于 2016-4-27 15:54:30 | 显示全部楼层 |阅读模式
首先了解枚举进程需要用到的相关的api:CreateToolhelp32Snapshot 获取进程快照
[C] syntaxhighlighter_viewsource syntaxhighlighter_copycode
函数原型:
HANDLE WINAPI CreateToolhelp32Snapshot(
DWORD dwFlags, //用来指定“快照”中需要返回的对象,可以是TH32CS_SNAPPROCESS等
DWORD th32ProcessID //一个进程ID号,用来指定要获取哪一个进程的快照,当获取系统进程列表或获取 当前进程快照时可以设为0
);
//调用成功,返回快照的句柄,调用失败,返回INVALID_HANDLE_VALUE


Process32First 获取第一个进程信息
[C] syntaxhighlighter_viewsource syntaxhighlighter_copycode
 Process32First(
    HANDLE hSnapshot,//CreateToolhelp32Snapshot 返回的句柄
    LPPROCESSENTRY32 lppe//保存进程信息的结构
);

PROCESSENTRY32 结构如下:
[C] syntaxhighlighter_viewsource syntaxhighlighter_copycode
typedef struct tagPROCESSENTRY32 {
    DWORD dwSize; // 结构大小;
    DWORD cntUsage; // 此进程的引用计数;
    DWORD th32ProcessID; // 进程ID;
    DWORD th32DefaultHeapID; // 进程默认堆ID;
    DWORD th32ModuleID; // 进程模块ID;
    DWORD cntThreads; // 此进程开启的线程计数;
    DWORD th32ParentProcessID;// 父进程ID;
    LONG pcPriClassBase; // 线程优先权;
    DWORD dwFlags; // 保留;
    WCHAR szExeFile[MAX_PATH]; // 进程全名;
} PROCESSENTRY32;


Process32Next 来获得下一个进程信息
参数同Process32First  。


纯c语言实现代码:
[C] syntaxhighlighter_viewsource syntaxhighlighter_copycode
#include <Windows.h>
#include <TlHelp32.h>
#include <stdio.h>
int main()
{
        PROCESSENTRY32 processEntry = { 0 };
        processEntry.dwSize = sizeof(PROCESSENTRY32);
        HANDLE hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0);
        if (hProcessSnap == INVALID_HANDLE_VALUE) return -1;

        
        BOOL bRet = Process32First(hProcessSnap,&processEntry);
        while (bRet)
        {
                printf("ProcessID:%d  %s\n",processEntry.th32ProcessID,processEntry.szExeFile);
                bRet = Process32Next(hProcessSnap,&processEntry);
        }
        CloseHandle(hProcessSnap);
        system("pause");
        return 0;
}


vs2013 执行结果:

c语言枚举进程以及遇到的编码问题

c语言枚举进程以及遇到的编码问题

这时候发现一个很重要的问题,和预料的不同啊,进程名称显示乱码!代码并没有什么指针泄露,也不是printf安全不安全的问题,习惯性的丢到vc6里面,把代码修改为兼容vc6的标准(就是变量定义放在前面):
[C] syntaxhighlighter_viewsource syntaxhighlighter_copycode
#include <Windows.h>
#include <TlHelp32.h>
#include <stdio.h>
int main()
{
        HANDLE hProcessSnap;
        PROCESSENTRY32 processEntry = { 0 };
        BOOL bRet ;
        processEntry.dwSize = sizeof(PROCESSENTRY32);

        hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0);
        if (hProcessSnap == INVALID_HANDLE_VALUE) return -1;

        
        bRet= Process32First(hProcessSnap,&processEntry);
        while (bRet)
        {

                printf("ProcessID:%d  %s\n",processEntry.th32ProcessID,processEntry.szExeFile);
                bRet = Process32Next(hProcessSnap,&processEntry);
        }
        CloseHandle(hProcessSnap);
        system("pause");
        return 0;
}


运行结果:

c语言枚举进程以及遇到的编码问题

c语言枚举进程以及遇到的编码问题

竟然没有出现乱码!!
机智的我这时候考虑到了编码问题,vc6的编码是Ansi,vs2013默认的Unicode,
那么vs修改项目文件属性,把字符集Unicode改为多字节字符集(以Ansi为基础的字符集):

c语言枚举进程以及遇到的编码问题

c语言枚举进程以及遇到的编码问题

再次在vs2013下运行:

c语言枚举进程以及遇到的编码问题

c语言枚举进程以及遇到的编码问题

这样就正常了,问题解决。

导致这个问题的原因是在Unicode编码下processEntry.szExeFile是WCHAR(宽字节)类型数组
在多字节字符集下是CHAR(窄字节)类型数组,如果希望不改变项目的字符集解决这一问题,可以通过编码转换的方式把Unicode编码下获取到的processEntry.szExeFile转换为Ansi编码,然后输出。
不改变项目字符集,成功枚举进程的代码:
[C] syntaxhighlighter_viewsource syntaxhighlighter_copycode
#include <Windows.h>
#include <TlHelp32.h>
#include <stdio.h>
int main()
{
        PROCESSENTRY32 processEntry = { 0 };
        processEntry.dwSize = sizeof(PROCESSENTRY32);
        HANDLE hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0);
        if (hProcessSnap == INVALID_HANDLE_VALUE) return -1;

        
        BOOL bRet = Process32First(hProcessSnap,&processEntry);
        while (bRet)
        {
                int  nLength = WideCharToMultiByte(CP_ACP, 0, processEntry.szExeFile, -1, NULL, 0, NULL, NULL);//获取字符长度
                char *str = (char *)malloc(sizeof(char)*nLength);
                WideCharToMultiByte(CP_ACP, 0, processEntry.szExeFile, -1, str, nLength, NULL, NULL);//编码转换-unicode转ansi
                printf("ProcessID:%d  %s\n",processEntry.th32ProcessID,str);
                bRet = Process32Next(hProcessSnap,&processEntry);
        }
        CloseHandle(hProcessSnap);
        system("pause");
        return 0;
}


运行结果同上。


c语言枚举进程以及遇到的编码问题

c语言枚举进程以及遇到的编码问题



不积跬步,无以至千里

5

主题

28

帖子

76

积分

注册会员

Rank: 2

积分
76
发表于 2016-4-28 11:14:52 来自手机 | 显示全部楼层
腻害!!
回复

使用道具 举报

0

主题

20

帖子

70

积分

注册会员

Rank: 2

积分
70
发表于 2016-5-8 01:18:39 | 显示全部楼层
编程经验很丰富啊

1

主题

60

帖子

74

积分

注册会员

Rank: 2

积分
74
发表于 2017-3-5 21:32:09 | 显示全部楼层
感谢分享
回复

使用道具 举报

1

主题

60

帖子

74

积分

注册会员

Rank: 2

积分
74
发表于 2017-3-5 21:33:08 | 显示全部楼层
感谢分享
回复

使用道具 举报

1

主题

60

帖子

74

积分

注册会员

Rank: 2

积分
74
发表于 2017-3-5 21:33:25 | 显示全部楼层
感谢分享
回复

使用道具 举报

1

主题

60

帖子

74

积分

注册会员

Rank: 2

积分
74
发表于 2017-3-5 21:33:44 | 显示全部楼层
感谢分享
回复

使用道具 举报

0

主题

18

帖子

40

积分

新手上路

Rank: 1

积分
40
发表于 2017-5-8 16:29:32 | 显示全部楼层
666666666666666666

0

主题

37

帖子

69

积分

注册会员

Rank: 2

积分
69
发表于 2017-10-31 14:57:42 | 显示全部楼层
感谢大大们的分享

0

主题

37

帖子

69

积分

注册会员

Rank: 2

积分
69
发表于 2017-10-31 14:59:00 | 显示全部楼层
感谢大大们的分享
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|Archiver|小黑屋|sitemap|编程论坛 - 一个单纯的编程学习交流论坛 ( 豫ICP备15032706号 )

GMT+8, 2018-8-16 14:42 , Processed in 1.168283 second(s), 30 queries .

Powered by Discuz! X3.4

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表