[置顶] 20160209.CCPP体系详解(0019天)

程序片段(01):01.字符串.c
内容概要:字符串

#include <stdio.h>
#include <stdlib.h>
#include <locale.h>
#include <string.h>

//00.语音合成与语音识别:
// 语音合成:文字-->语音
// 语音识别:语音-->文字
// 注:任何语音操作都应当使用宽字符!
//01.宽窄字符问题1:
// 1.窄字符采用单字节存储,宽字符采用双字节存储!
// 宽窄字符的"本质区别"在于存储数据的盒子尺寸不一样! 
// 2.宽窄字符当中使用sizeof关键字的情况区分:
// 1.sizeof求取的是数据类型所占用的尺寸而不是实际数据本身所占用的尺寸
// 2.但是操作"实际数据本身"的时候都是一样的
// 注:如果项目采用Unicode字符集,那所有非ASCII字符集范围内
// 的字符都必须采用宽字符进行存储(也即是除开英文之外的字符)
// 再加上本地化的设置才能正确的表示文字信息
int main01(void)
{
    char ch = 'A';//窄字符-->1个字节:不可以存储单个汉字儿
    wchar_t wch = L'A';//宽字符-->2个字节:可以存储单个汉字儿
    printf("%d ,%d \n", sizeof(ch), sizeof(wch));
    printf("%d ,%d \n", ch, wch);//内容一致-->ASCII码值一样

    system("pause");
}

//02.宽窄字符问题2:
// 1.宽窄字符只是存储数据的盒子不一样:
// 如果存储内容一样,那么二进制数据本质一样
// 2.Unicode字符集+(语言环境设置+宽字符使用):
// 才能使得除开标准ASCII编码集的字符之外的字符能够正确显示
// 3.只要在Unicode字符集的情况之下,所有内容都采用宽字符
// 就能保证字符的正确存储与使用
// 注:宽字符也是字符,只不过表象比窄字符多了一个宽字符标识符"L"
// 但还是需要使用单引号('')进行宽字符标识
int main02(void)
{                    
    char ch = 'A';
    putchar(ch);

    //wchar_t wch = L'A';
    //putwchar(wch);

    setlocale(LC_ALL, "zh-CN");
    wchar_t wch = L'我';
    putwchar(wch);//采用字符集+设定语言环境+使用宽字符+使用宽字符操作函数-->字符正确存储并使用

    system("pause");
}

//03.宽窄字符问题3:
// 1.VS默认配置情况下,所有字符都采用窄字符进行存储:
// 因为窄字符对于ASCII编码表当中的字符都能够进行兼容
// 2.代码区的内容不可以进行修改!(编译不报错,运行直接挂掉!)
// 找到代码区+突破权限
// 3.代码区当中的两块儿重要内存(常量池和符号表):
// 常量池:只需要拷贝指针,直接进行访问(类似于函数传参,针对于数组无副本机制)
// 提高内存操作效率
// 符号表:但需要拷贝数据,生成于寄存器(必须通过寄存器进行数据的传递操作)
// 注:只要是代码区的内容,就不可以直接进行修改,无论是常量池内容还是符号表内容
// 但是通过Detours这款工具就可以突破任何Windows程序的代码区"访问权限",并且
// 可以对该代码区的内容进行直接修改"修改权限"
int main03(void)
{
    char * pStr = "notepad";
    printf("%d \n", sizeof(pStr));
    printf("%d \n", sizeof("notepad"));
    printf("%p \n", pStr);//该地址属于(内存-->代码区)的地址

    //*pStr = 'A';//由于该指针所指向的内存区块儿位于代码区,因此只可读不可写[因为这里访问的字符串位于代码区常量池]
    //"notepad"这个字符串属于代码区的常量池当中,该常量池也被称作为只读常量池
    //"notepad"字符串都存储于代码区常量池,非字符串常量需要拷贝数据到寄存器才能对内存进行间接操作
    //字符串传递的是地址

    system("pause");
}

//04.宽窄字符问题4:
// 1.针对于英文,使用宽窄字符的效果都一样,不同的只是存储字符的盒子尺寸不一样
// 宽窄字符的针对对象:非英文|非标准ASCII码表包含字符
// 2.无论是针对于窄字符字符串还是宽字符字符串,该字符串的结尾标识符都是字符'\0'
// 注:只不过区分窄字符('\0')和宽字符(L'\0')所占用的存储空间尺寸不一样!
// (1).strlen和sizeof();的区别(一个是求取有效字符数;一个是求取实际"占用"字节数)
// 区分:"占用"和"实际"两个关键词儿的区别
// (2).sizeof();取值运算符(星号:"*")加字符指针(区分宽窄字符指针),
// 所求字节数为单个相应字符(宽窄)所占用的字节数
// 3.宽窄字符的打印:
// wprintf+L"格式字符串"+格式控制符
// 注:严格区分"格式字符串"和"格式控制符"之间的区别
// 4.字符指针所指向的数据有多重解析方式:
// 字符串层面的意义解析方式(字符串函数)+字符层面的解析范式(sizeof关键字)
// 注:切忌注意!
int main04(void)
{
    setlocale(LC_ALL, "zh-CN");
    wchar_t * pWchar = L"你好天朝";//代码区常量池
    printf("%d, %d \n", sizeof(pWchar), sizeof(*pWchar));//特殊点!->两个字节
    printf("%d \n", sizeof(L"你好天朝"));//实际占用内存尺寸!
    //printf("%p \n", pWchar);
    wprintf(L"%p \n", pWchar);

    system("pause");
}

//05.通过字符指针访问字符指针所指向的内存实体,只能访问到单个字符的效果!
// 数组:一旦前置手动初始化,那么后置一定是默认初始化,数据自动清为零
// 无论前置初始化的内容是什么,后置自动初始化的数据都为0
// 注:初始化与赋值的概念不同
// 取值运算符+字符指针-->只能访问到单个字符(区分宽窄字符!)
int main05(void)
{
    char * pStr = "notepad";//窄字符字符串,pStr是指针变量,存储的是(代码区常量池)中的常量字符串的首地址
    //pStr可以存储不同的指针,属于变量指针
    *pStr = 'A';//随时注意指针变量所指向的内存空间的访问权限(代码区只可读,不可写,除非借助Detours)
    char str[100] = "calc";
    //str = 1;//str作为数组名是常量指针
    //拷贝字符串到数组,等同于对数组的前置初始化,后置默认初始化,因此后续数据全部清零
    printf("%p \n", str);
    //system(str);
    *str = 'X';//通过(取值运算符+字符指针)只能访问到单个字符(区分宽窄字符)

    system("pause");
}

//06.宽窄字符数组注意事项:
// 1.数组名一定不可以直接进行修改!-->常量指针
// 2.C语言判断两个字符串是否相等的方式strcmp(str1, str2);函数
// 注:sizeof();和strlen();所求取的长度意义不同
// sizeof();求取的是真实的内存尺寸!
// strlen();和wstrlen();只是求取"有效"字符个数!
int main06(void)
{
    char str1[100] = "calc";
    char str2[100] = "calc1";
    //char * p;
    //str1 = str2;//比较的是常量指针的值
    if (0 == strcmp(str1, str2))
    {
        printf("相等! \n");
    }
    else
    {
        printf("不相等! \n");
    }
    wchar_t wStr[100] = L"您好天朝";
    printf("%d \n", sizeof(wStr));

    system("pause");
}

程序片段(02):speechrecognition.cpp
内容概要:语音识别

#include <windows.h>
#include <atlstr.h>
#include <sphelper.h>
#include <sapi.h>
#include<comutil.h>
#include<string.h>//01.所需头文件

#pragma comment(lib,"sapi.lib")
#pragma comment(lib, "comsupp.lib") //02.链接静态库

#define GID_CMD_GR 333333
#define WM_RECOEVENT WM_USER+1

 LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM);//03.所需COM组件:通过COM组件实现语音(识别|合成)功能

 char    szAppName[] = "TsinghuaYincheng";
 BOOL b_initSR;
 BOOL b_Cmd_Grammar;
 CComPtr<ISpRecoContext>m_cpRecoCtxt;  //语音识别程序接口
 CComPtr<ISpRecoGrammar>m_cpCmdGramma; //识别语法
 CComPtr<ISpRecognizer>m_cpRecoEngine; //语音识别引擎
 int speak(wchar_t *str);

 int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,PSTR szCmdLine,int iCmdShow)
 {
     HWND        hwnd;
     MSG         msg;
     WNDCLASS    wndclass;

     wndclass.cbClsExtra          =0;
     wndclass.cbWndExtra          =0;
     wndclass.hbrBackground       =(HBRUSH)GetStockObject(WHITE_BRUSH);
     wndclass.hCursor             =LoadCursor(NULL,IDC_ARROW);
     wndclass.hIcon               =LoadIcon(NULL,IDI_APPLICATION);
     wndclass.hInstance           =hInstance;
     wndclass.lpfnWndProc         =WndProc;
     wndclass.lpszClassName       =szAppName;
     wndclass.lpszMenuName        =NULL;
     wndclass.style               =CS_HREDRAW|CS_VREDRAW;

     if(!RegisterClass(&wndclass))
     {
         MessageBox(NULL,TEXT("This program requires Windows NT!"),szAppName,MB_ICONERROR);
         return 0;
     }
     speak(L"尹成是一个猥琐男,曾彬更加猥琐,Hello XiaoBin i love you!!!");//04.语音合成:将文件转化为语音(注:使用宽字符)
     //05.关于语音方面的编程概念:
     // 1.什么叫做语音识别?什么叫做语音合成?
     // 语音识别:将声音转化为文字
     // 语音合成:将文字转化为声音
     // 2.语音识别必须采用宽字符!
     // 配置:项目属性-->Unicode字符集-->宽字符操作(所有都使用宽字符)
    // speak(NULL);//表示不说话
     hwnd=CreateWindow(szAppName,
                       TEXT("清华-尹成语音识别教程"),
                       WS_OVERLAPPEDWINDOW,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       NULL,
                       NULL,
                       hInstance,
                       NULL);

     ShowWindow(hwnd,iCmdShow);
     UpdateWindow(hwnd);

     while(GetMessage(&msg,NULL,0,0))
     {
         TranslateMessage(&msg);
         DispatchMessage(&msg);
     }

     return msg.wParam;
 }

 LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
 {
     HDC           hdc;
     PAINTSTRUCT   ps;

     switch(message)
     {
     case WM_CREATE:
         {
             //初始化COM端口
             ::CoInitializeEx(NULL,COINIT_APARTMENTTHREADED);
             //创建识别引擎COM实例为共享型
             HRESULT hr=m_cpRecoEngine.CoCreateInstance(CLSID_SpSharedRecognizer);
             //创建识别上下文接口
             if(SUCCEEDED(hr))
             {
                 hr=m_cpRecoEngine->CreateRecoContext(&m_cpRecoCtxt);
             }
             else MessageBox(hwnd,TEXT("error1"),TEXT("error"),S_OK);
             //设置识别消息,使计算机时刻监听语音消息
             if(SUCCEEDED(hr))
             {
                 hr=m_cpRecoCtxt->SetNotifyWindowMessage(hwnd,WM_RECOEVENT,0,0);
             }
             else MessageBox(hwnd,TEXT("error2"),TEXT("error"),S_OK);
             //设置我们感兴趣的事件
             if(SUCCEEDED(hr))
             {
                 ULONGLONG ullMyEvents=SPFEI(SPEI_SOUND_START)|SPFEI(SPEI_RECOGNITION)|SPFEI(SPEI_SOUND_END);
                 hr=m_cpRecoCtxt->SetInterest(ullMyEvents,ullMyEvents);
             }
             else MessageBox(hwnd,TEXT("error3"),TEXT("error"),S_OK);
             //创建语法规则
             b_Cmd_Grammar=TRUE;
             if(FAILED(hr))
             {
                 MessageBox(hwnd,TEXT("error4"),TEXT("error"),S_OK);
             }
             hr=m_cpRecoCtxt->CreateGrammar(GID_CMD_GR,&m_cpCmdGramma);
             WCHAR wszXMLFile[20]=L"er.xml";
             MultiByteToWideChar(CP_ACP,0,(LPCSTR)"er.xml",-1,wszXMLFile,256);
             hr=m_cpCmdGramma->LoadCmdFromFile(wszXMLFile,SPLO_DYNAMIC);
             if(FAILED(hr))
             {
                 MessageBox(hwnd,TEXT("error5"),TEXT("error"),S_OK);
             }
             b_initSR=TRUE;
             //在开始识别时,激活语法进行识别
             hr=m_cpCmdGramma->SetRuleState(NULL,NULL,SPRS_ACTIVE);
             return 0;
         }
     case WM_RECOEVENT:
         {
             RECT rect;
             GetClientRect(hwnd,&rect);
             hdc=GetDC(hwnd);
             USES_CONVERSION;
             CSpEvent event;
             while(event.GetFrom(m_cpRecoCtxt)==S_OK)
             {
                 switch(event.eEventId)
                 {
                 case SPEI_RECOGNITION:
                     {
                         static const WCHAR wszUnrecognized[]=L"<Unrecognized>";
                         CSpDynamicString dstrText;
                         //取得识别结果
                         if(FAILED(event.RecoResult()->GetText(SP_GETWHOLEPHRASE,SP_GETWHOLEPHRASE,TRUE,&dstrText,NULL)))
                         {
                             dstrText=wszUnrecognized;
                         }
                         BSTR SRout;
                         dstrText.CopyToBSTR(&SRout);
                         char* lpszText2 = _com_util::ConvertBSTRToString(SRout);//06.获取语音识别之后所生成的字符串

                         if(b_Cmd_Grammar)
                         {
                             char * str[15] = {"我是学霸","清华土匪","天下无双","吴伟","曾彬",//07.查表法所对应的字符串表
                                 "记事本","计算器","关机","重启","取消","画图板",
                                 "何栋","林振华","聂千琳","赵学辉"};
                             int i = -1;
                             for (int j = 0; j < 15;j++)
                             {
                                 if (strcmp(str[j], lpszText2) == 0)
                                 {
                                     i = j;
                                     MessageBoxA(0, lpszText2, lpszText2, 0);//弹出对话框
                                     DrawText(hdc, TEXT(lpszText2), -1, &rect, DT_SINGLELINE | DT_CENTER | DT_VCENTER);//在窗体当中写一段儿文字
                                 }
                             }
                             switch (i)//08.根据查表法所查询到的结果,进行针对于结果的处理
                             {
                             case -1:
                                 break;
                             case 0:
                                 speak(L"男神是学霸");
                                 break;
                             case 1:
                                 speak(L"猥琐男尹成");
                                 break;
                             case 2:
                                 speak(L"曾彬猥琐的天下无创");
                                 break;
                             case 3:
                                 speak(L"男神帅的惊动了白宫");
                                 break;
                             case 4:
                                 speak(L"曾彬猥琐的的惊动了白宫");
                                 break;
                             case 5:
                                 system("start calc");//09.需要采用异步进行打开
                                 break;
                             case 6:
                                 system("start notepad");
                                 break;
                             case 7:
                                 system("shutdown -s -t 600");
                                 break;
                             case 8:
                                 system("shutdown -r -t 600");
                                 break;
                             case 9:
                                 system("shutdown -a");//10.取消(关机|重启)指令
                                 break;
                             case 10:
                                 system("start mspaint");
                                 break;
                             case 11:
                                 speak(L"何栋向三大金刚问好");//11.注意:通过语音识别的联系,可以提高语音识别的精确度!
                                 break;
                             case 12:
                                 speak(L"振华睡得很晚");//12.Debug模式相比Release模式更加的消耗资源
                                 break;
                             case 13:
                                 speak(L"聂千琳爱编码");//13.某些COM组件的调用需要在Debug模式下进行
                                 break;
                             case 14:
                                 speak(L"赵学辉爱编码");//14.经过语音识别训练之后,电脑的语音识别效果更佳!
                                 break;
                             default:
                                 break;
                             }
                         }    
                     }
                 }
             }
             return TRUE;
         }
     case WM_PAINT:
         hdc=BeginPaint(hwnd,&ps);
         EndPaint(hwnd,&ps);
         return 0;
     case WM_DESTROY:
         PostQuitMessage(0);
         return 0;
     }
     return DefWindowProc(hwnd,message,wParam,lParam);
 }

#pragma comment(lib, "ole32.lib") //CoInitialize CoCreateInstance需要调用ole32.dll 
int speak(wchar_t *str)
{
     ISpVoice * pVoice = NULL;
     ::CoInitialize(NULL);
     //获取ISpVoice接口: 
     long hr = CoCreateInstance(CLSID_SpVoice, NULL, CLSCTX_ALL, IID_ISpVoice, (void **)&pVoice);
     hr = pVoice->Speak(str, 0, NULL);
     pVoice->Release();
     pVoice = NULL;
     //千万不要忘记: 
     ::CoUninitialize();
     return TRUE;
 }

程序片段(03):speechrecognition.cpp
内容概要:语音识别

#include <windows.h>
#include <atlstr.h>
#include <sphelper.h>
#include <sapi.h>
#include<comutil.h>
#include<string.h>

#pragma comment(lib,"sapi.lib")
#pragma comment(lib, "comsupp.lib") 

#define GID_CMD_GR 333333
#define WM_RECOEVENT WM_USER+1

 LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM);

 char    szAppName[] = "TsinghuaYincheng";
 BOOL b_initSR;
 BOOL b_Cmd_Grammar;
 CComPtr<ISpRecoContext>m_cpRecoCtxt;  //语音识别程序接口
 CComPtr<ISpRecoGrammar>m_cpCmdGramma; //识别语法
 CComPtr<ISpRecognizer>m_cpRecoEngine; //语音识别引擎
 int speak(wchar_t *str);

 int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,PSTR szCmdLine,int iCmdShow)
 {
     HWND        hwnd;
     MSG         msg;
     WNDCLASS    wndclass;

     wndclass.cbClsExtra          =0;
     wndclass.cbWndExtra          =0;
     wndclass.hbrBackground       =(HBRUSH)GetStockObject(WHITE_BRUSH);
     wndclass.hCursor             =LoadCursor(NULL,IDC_ARROW);
     wndclass.hIcon               =LoadIcon(NULL,IDI_APPLICATION);
     wndclass.hInstance           =hInstance;
     wndclass.lpfnWndProc         =WndProc;
     wndclass.lpszClassName       =szAppName;
     wndclass.lpszMenuName        =NULL;
     wndclass.style               =CS_HREDRAW|CS_VREDRAW;

     if(!RegisterClass(&wndclass))
     {
         MessageBox(NULL,TEXT("This program requires Windows NT!"),szAppName,MB_ICONERROR);
         return 0;
     }
     speak(L"尹成是一个猥琐男");
    // speak(NULL);
     hwnd=CreateWindow(szAppName,
                       TEXT("清华-尹成语音识别教程"),
                       WS_OVERLAPPEDWINDOW,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       NULL,
                       NULL,
                       hInstance,
                       NULL);

     ShowWindow(hwnd,iCmdShow);
     UpdateWindow(hwnd);

     while(GetMessage(&msg,NULL,0,0))
     {
         TranslateMessage(&msg);
         DispatchMessage(&msg);
     }

     return msg.wParam;
 }

 LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
 {
     HDC           hdc;
     PAINTSTRUCT   ps;

     switch(message)
     {
     case WM_CREATE:
         {
             //初始化COM端口
             ::CoInitializeEx(NULL,COINIT_APARTMENTTHREADED);
             //创建识别引擎COM实例为共享型
             HRESULT hr=m_cpRecoEngine.CoCreateInstance(CLSID_SpSharedRecognizer);
             //创建识别上下文接口
             if(SUCCEEDED(hr))
             {
                 hr=m_cpRecoEngine->CreateRecoContext(&m_cpRecoCtxt);
             }
             else MessageBox(hwnd,TEXT("error1"),TEXT("error"),S_OK);
             //设置识别消息,使计算机时刻监听语音消息
             if(SUCCEEDED(hr))
             {
                 hr=m_cpRecoCtxt->SetNotifyWindowMessage(hwnd,WM_RECOEVENT,0,0);
             }
             else MessageBox(hwnd,TEXT("error2"),TEXT("error"),S_OK);
             //设置我们感兴趣的事件
             if(SUCCEEDED(hr))
             {
                 ULONGLONG ullMyEvents=SPFEI(SPEI_SOUND_START)|SPFEI(SPEI_RECOGNITION)|SPFEI(SPEI_SOUND_END);
                 hr=m_cpRecoCtxt->SetInterest(ullMyEvents,ullMyEvents);
             }
             else MessageBox(hwnd,TEXT("error3"),TEXT("error"),S_OK);
             //创建语法规则
             b_Cmd_Grammar=TRUE;
             if(FAILED(hr))
             {
                 MessageBox(hwnd,TEXT("error4"),TEXT("error"),S_OK);
             }
             hr=m_cpRecoCtxt->CreateGrammar(GID_CMD_GR,&m_cpCmdGramma);
             WCHAR wszXMLFile[20]=L"er.xml";
             MultiByteToWideChar(CP_ACP,0,(LPCSTR)"er.xml",-1,wszXMLFile,256);
             hr=m_cpCmdGramma->LoadCmdFromFile(wszXMLFile,SPLO_DYNAMIC);
             if(FAILED(hr))
             {
                 MessageBox(hwnd,TEXT("error5"),TEXT("error"),S_OK);
             }
             b_initSR=TRUE;
             //在开始识别时,激活语法进行识别
             hr=m_cpCmdGramma->SetRuleState(NULL,NULL,SPRS_ACTIVE);
             return 0;
         }
     case WM_RECOEVENT:
         {
             RECT rect;
             GetClientRect(hwnd,&rect);
             hdc=GetDC(hwnd);
             USES_CONVERSION;
             CSpEvent event;
             while(event.GetFrom(m_cpRecoCtxt)==S_OK)
             {
                 switch(event.eEventId)
                 {
                 case SPEI_RECOGNITION:
                     {
                         static const WCHAR wszUnrecognized[]=L"<Unrecognized>";
                         CSpDynamicString dstrText;
                         //取得识别结果
                         if(FAILED(event.RecoResult()->GetText(SP_GETWHOLEPHRASE,SP_GETWHOLEPHRASE,TRUE,&dstrText,NULL)))
                         {
                             dstrText=wszUnrecognized;
                         }
                         BSTR SRout;
                         dstrText.CopyToBSTR(&SRout);
                         char* lpszText2 = _com_util::ConvertBSTRToString(SRout);

                         if(b_Cmd_Grammar)
                         {
                             if (strcmp("跳跃",lpszText2)==0)
                             {    
                                 keybd_event(VK_SPACE, 0, 0, 0);//按下
                                 keybd_event(VK_SPACE, 0, 2, 0);//松开
                             }
                             if (strcmp("趴下", lpszText2) == 0)
                             {
                                 keybd_event('Y', 0, 0, 0);//按下
                                 keybd_event('Y', 0, 2, 0);//松开
                             }
                             if (strcmp("后前前", lpszText2) == 0)
                             {
                                 keybd_event('S', 0, 0, 0);//按下
                                 keybd_event('S', 0, 2, 0);//松开
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开

                                 mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);//01.鼠标左键按下才开始攻击鼠标位置的人物!
                                 mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
                                 ////MapVirtualKey映射
                                 //keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                                 //keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                             if (strcmp("左右前", lpszText2) == 0)
                             {
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('D', 0, 0, 0);//按下
                                 keybd_event('D', 0, 2, 0);//松开
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);
                                 mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
                             }
                             if (strcmp("火冒三丈", lpszText2) == 0)
                             {
                                 keybd_event(VK_OEM_102, 0, 0, 0);//按下//02.C语言函数模拟反斜杠("\")
                                 keybd_event(VK_OEM_102, 0, 2, 0);//松开 \ang
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('N', 0, 0, 0);//按下
                                 keybd_event('N', 0, 2, 0);//松开
                                 keybd_event('G', 0, 0, 0);//按下
                                 keybd_event('G', 0, 2, 0);//松开

                                 //MapVirtualKey映射
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN,0), 0, 0);//按下
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                             if (strcmp("旋风刀", lpszText2) == 0)
                             {
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('D', 0, 0, 0);//按下
                                 keybd_event('D', 0, 2, 0);//松开
                                 keybd_event('S', 0, 0, 0);//按下
                                 keybd_event('S', 0, 2, 0);//松开
                                 mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);
                                 mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
                             }
                             if (strcmp("大枪", lpszText2) == 0)
                             {
                                 keybd_event(VK_OEM_102, 0, 0, 0);//按下
                                 keybd_event(VK_OEM_102, 0, 2, 0);//松开 \ang
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 keybd_event('E', 0, 0, 0);//按下
                                 keybd_event('E', 0, 2, 0);//松开
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('P', 0, 0, 0);//按下
                                 keybd_event('P', 0, 2, 0);//松开
                                 keybd_event('O', 0, 0, 0);//按下
                                 keybd_event('O', 0, 2, 0);//松开
                                 keybd_event('N', 0, 0, 0);//按下
                                 keybd_event('N', 0, 2, 0);//松开

                                 keybd_event(VK_SPACE, 0, 0, 0);//按下
                                 keybd_event(VK_SPACE, 0, 2, 0);//松开

                                 keybd_event(VK_NUMPAD7, 0, 0, 0);//按下 //03.数字键所对应的虚拟映射键
                                 keybd_event(VK_NUMPAD7, 0, 2, 0);//松开

                                 //MapVirtualKey映射
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                             if (strcmp("大刀", lpszText2) == 0)
                             {
                                 keybd_event(VK_OEM_102, 0, 0, 0);//按下
                                 keybd_event(VK_OEM_102, 0, 2, 0);//松开 \ang
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 keybd_event('E', 0, 0, 0);//按下
                                 keybd_event('E', 0, 2, 0);//松开
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('P', 0, 0, 0);//按下
                                 keybd_event('P', 0, 2, 0);//松开
                                 keybd_event('O', 0, 0, 0);//按下
                                 keybd_event('O', 0, 2, 0);//松开
                                 keybd_event('N', 0, 0, 0);//按下
                                 keybd_event('N', 0, 2, 0);//松开

                                 keybd_event(VK_SPACE, 0, 0, 0);//按下
                                 keybd_event(VK_SPACE, 0, 2, 0);//松开

                                 keybd_event(VK_NUMPAD6, 0, 0, 0);//按下
                                 keybd_event(VK_NUMPAD6, 0, 2, 0);//松开

                                 //MapVirtualKey映射
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                             if (strcmp("刺蛇", lpszText2) == 0)
                             {
                                 keybd_event(VK_OEM_102, 0, 0, 0);//按下
                                 keybd_event(VK_OEM_102, 0, 2, 0);//松开 \ang
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 keybd_event('E', 0, 0, 0);//按下
                                 keybd_event('E', 0, 2, 0);//松开
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('P', 0, 0, 0);//按下
                                 keybd_event('P', 0, 2, 0);//松开
                                 keybd_event('O', 0, 0, 0);//按下
                                 keybd_event('O', 0, 2, 0);//松开
                                 keybd_event('N', 0, 0, 0);//按下
                                 keybd_event('N', 0, 2, 0);//松开

                                 keybd_event(VK_SPACE, 0, 0, 0);//按下
                                 keybd_event(VK_SPACE, 0, 2, 0);//松开

                                 keybd_event(VK_NUMPAD4, 0, 0, 0);//按下
                                 keybd_event(VK_NUMPAD4, 0, 2, 0);//松开

                                 //MapVirtualKey映射
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                             if (strcmp("大剑", lpszText2) == 0)
                             {
                                 keybd_event(VK_OEM_102, 0, 0, 0);//按下
                                 keybd_event(VK_OEM_102, 0, 2, 0);//松开 \ang
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 keybd_event('E', 0, 0, 0);//按下
                                 keybd_event('E', 0, 2, 0);//松开
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('P', 0, 0, 0);//按下
                                 keybd_event('P', 0, 2, 0);//松开
                                 keybd_event('O', 0, 0, 0);//按下
                                 keybd_event('O', 0, 2, 0);//松开
                                 keybd_event('N', 0, 0, 0);//按下
                                 keybd_event('N', 0, 2, 0);//松开

                                 keybd_event(VK_SPACE, 0, 0, 0);//按下
                                 keybd_event(VK_SPACE, 0, 2, 0);//松开

                                 keybd_event(VK_NUMPAD1, 0, 0, 0);//按下
                                 keybd_event(VK_NUMPAD1, 0, 2, 0);//松开
                                 keybd_event(VK_NUMPAD5, 0, 0, 0);//按下
                                 keybd_event(VK_NUMPAD5, 0, 2, 0);//松开

                                 //MapVirtualKey映射
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                             if (strcmp("大锤", lpszText2) == 0)
                             {
                                 keybd_event(VK_OEM_102, 0, 0, 0);//按下
                                 keybd_event(VK_OEM_102, 0, 2, 0);//松开 \ang
                                 keybd_event('W', 0, 0, 0);//按下
                                 keybd_event('W', 0, 2, 0);//松开
                                 keybd_event('E', 0, 0, 0);//按下
                                 keybd_event('E', 0, 2, 0);//松开
                                 keybd_event('A', 0, 0, 0);//按下
                                 keybd_event('A', 0, 2, 0);//松开
                                 keybd_event('P', 0, 0, 0);//按下
                                 keybd_event('P', 0, 2, 0);//松开
                                 keybd_event('O', 0, 0, 0);//按下
                                 keybd_event('O', 0, 2, 0);//松开
                                 keybd_event('N', 0, 0, 0);//按下
                                 keybd_event('N', 0, 2, 0);//松开

                                 keybd_event(VK_SPACE, 0, 0, 0);//按下
                                 keybd_event(VK_SPACE, 0, 2, 0);//松开

                                 keybd_event(VK_NUMPAD1, 0, 0, 0);//按下
                                 keybd_event(VK_NUMPAD1, 0, 2, 0);//松开
                                 keybd_event(VK_NUMPAD8, 0, 0, 0);//按下
                                 keybd_event(VK_NUMPAD8, 0, 2, 0);//松开

                                 //MapVirtualKey映射
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                                 keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
                             }
                         }    
                     }
                 }
             }
             return TRUE;
         }
     case WM_PAINT:
         hdc=BeginPaint(hwnd,&ps);
         EndPaint(hwnd,&ps);
         return 0;
     case WM_DESTROY:
         system("taskkill /f /im 语音识别.exe");
         PostQuitMessage(0);
         return 0;
     }
     return DefWindowProc(hwnd,message,wParam,lParam);
 }

#pragma comment(lib, "ole32.lib") //CoInitialize CoCreateInstance需要调用ole32.dll 
int speak(wchar_t *str)
{
     ISpVoice * pVoice = NULL;
     ::CoInitialize(NULL);
     //获取ISpVoice接口: 
     long hr = CoCreateInstance(CLSID_SpVoice, NULL, CLSCTX_ALL, IID_ISpVoice, (void **)&pVoice);
     hr = pVoice->Speak(str, 0, NULL);
     pVoice->Release();
     pVoice = NULL;
     //千万不要忘记: 
     ::CoUninitialize();
     return TRUE;
 }
//04.识别游戏资源文件的修改
//05.区分:游戏引擎和游戏脚本
//06.逻辑文件+剧情文件+资源文件
// 任意进行修改
//07.游戏文件通常采用zip格式进行压缩
// 注:解压缩游戏资源文件需要注意目录层级!
//08.大多数都是PAK文件

程序片段(04):speechrecognition.cpp
内容概要:语音识别

#include <windows.h>
#include <atlstr.h>
#include <sphelper.h>
#include <sapi.h>
#include<comutil.h>
#include<string.h>

#pragma comment(lib,"sapi.lib")
#pragma comment(lib, "comsupp.lib") 

#define GID_CMD_GR 333333
#define WM_RECOEVENT WM_USER+1

 LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM);

 char    szAppName[] = "TsinghuaYincheng";
 BOOL b_initSR;
 BOOL b_Cmd_Grammar;
 CComPtr<ISpRecoContext>m_cpRecoCtxt;  //语音识别程序接口
 CComPtr<ISpRecoGrammar>m_cpCmdGramma; //识别语法
 CComPtr<ISpRecognizer>m_cpRecoEngine; //语音识别引擎
 int speak(wchar_t *str);

 int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,PSTR szCmdLine,int iCmdShow)
 {
     HWND        hwnd;
     MSG         msg;
     WNDCLASS    wndclass;

     wndclass.cbClsExtra          =0;
     wndclass.cbWndExtra          =0;
     wndclass.hbrBackground       =(HBRUSH)GetStockObject(WHITE_BRUSH);
     wndclass.hCursor             =LoadCursor(NULL,IDC_ARROW);
     wndclass.hIcon               =LoadIcon(NULL,IDI_APPLICATION);
     wndclass.hInstance           =hInstance;
     wndclass.lpfnWndProc         =WndProc;
     wndclass.lpszClassName       =szAppName;
     wndclass.lpszMenuName        =NULL;
     wndclass.style               =CS_HREDRAW|CS_VREDRAW;

     if(!RegisterClass(&wndclass))
     {
         MessageBox(NULL,TEXT("This program requires Windows NT!"),szAppName,MB_ICONERROR);
         return 0;
     }
     speak(L"尹成是一个猥琐男");
    // speak(NULL);
     hwnd=CreateWindow(szAppName,
                       TEXT("清华-尹成语音识别教程"),
                       WS_OVERLAPPEDWINDOW,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       CW_USEDEFAULT,
                       NULL,
                       NULL,
                       hInstance,
                       NULL);

     ShowWindow(hwnd,iCmdShow);
     UpdateWindow(hwnd);

     while(GetMessage(&msg,NULL,0,0))
     {
         TranslateMessage(&msg);
         DispatchMessage(&msg);
     }

     return msg.wParam;
 }

 LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
 {
     HDC           hdc;
     PAINTSTRUCT   ps;

     switch(message)
     {
     case WM_CREATE:
         {
             //初始化COM端口
             ::CoInitializeEx(NULL,COINIT_APARTMENTTHREADED);
             //创建识别引擎COM实例为共享型
             HRESULT hr=m_cpRecoEngine.CoCreateInstance(CLSID_SpSharedRecognizer);
             //创建识别上下文接口
             if(SUCCEEDED(hr))
             {
                 hr=m_cpRecoEngine->CreateRecoContext(&m_cpRecoCtxt);
             }
             else MessageBox(hwnd,TEXT("error1"),TEXT("error"),S_OK);
             //设置识别消息,使计算机时刻监听语音消息
             if(SUCCEEDED(hr))
             {
                 hr=m_cpRecoCtxt->SetNotifyWindowMessage(hwnd,WM_RECOEVENT,0,0);
             }
             else MessageBox(hwnd,TEXT("error2"),TEXT("error"),S_OK);
             //设置我们感兴趣的事件
             if(SUCCEEDED(hr))
             {
                 ULONGLONG ullMyEvents=SPFEI(SPEI_SOUND_START)|SPFEI(SPEI_RECOGNITION)|SPFEI(SPEI_SOUND_END);
                 hr=m_cpRecoCtxt->SetInterest(ullMyEvents,ullMyEvents);
             }
             else MessageBox(hwnd,TEXT("error3"),TEXT("error"),S_OK);
             //创建语法规则
             b_Cmd_Grammar=TRUE;
             if(FAILED(hr))
             {
                 MessageBox(hwnd,TEXT("error4"),TEXT("error"),S_OK);
             }
             hr=m_cpRecoCtxt->CreateGrammar(GID_CMD_GR,&m_cpCmdGramma);
             WCHAR wszXMLFile[20]=L"er.xml";
             MultiByteToWideChar(CP_ACP,0,(LPCSTR)"er.xml",-1,wszXMLFile,256);
             hr=m_cpCmdGramma->LoadCmdFromFile(wszXMLFile,SPLO_DYNAMIC);
             if(FAILED(hr))
             {
                 MessageBox(hwnd,TEXT("error5"),TEXT("error"),S_OK);
             }
             b_initSR=TRUE;
             //在开始识别时,激活语法进行识别
             hr=m_cpCmdGramma->SetRuleState(NULL,NULL,SPRS_ACTIVE);
             return 0;
         }
     case WM_RECOEVENT:
         {
             RECT rect;
             GetClientRect(hwnd,&rect);
             hdc=GetDC(hwnd);
             USES_CONVERSION;
             CSpEvent event;
             while(event.GetFrom(m_cpRecoCtxt)==S_OK)
             {
                 switch(event.eEventId)
                 {
                 case SPEI_RECOGNITION:
                     {
                         static const WCHAR wszUnrecognized[]=L"<Unrecognized>";
                         CSpDynamicString dstrText;
                         //取得识别结果
                         if(FAILED(event.RecoResult()->GetText(SP_GETWHOLEPHRASE,SP_GETWHOLEPHRASE,TRUE,&dstrText,NULL))) { dstrText=wszUnrecognized; } BSTR SRout; dstrText.CopyToBSTR(&SRout); char* lpszText2 = _com_util::ConvertBSTRToString(SRout); if(b_Cmd_Grammar) { if (strcmp("我是学霸",lpszText2)==0) { MessageBoxA(0, lpszText2, lpszText2, 0); //DrawText(hdc,TEXT("我是学霸"),-1,&rect,DT_SINGLELINE|DT_CENTER|DT_VCENTER); } if (strcmp("清华土匪", lpszText2) == 0) { MessageBoxA(0, lpszText2, lpszText2, 0); //DrawText(hdc,TEXT("我是学霸"),-1,&rect,DT_SINGLELINE|DT_CENTER|DT_VCENTER); } if (strcmp("吴伟", lpszText2) == 0) { MessageBoxA(0, lpszText2, lpszText2, 0); speak(L"吴伟帅的惊动了党中央国务院"); //DrawText(hdc,TEXT("我是学霸"),-1,&rect,DT_SINGLELINE|DT_CENTER|DT_VCENTER); } if (strcmp("曾彬", lpszText2) == 0) { MessageBoxA(0, lpszText2, lpszText2, 0); speak(L"曾彬猥琐的惊动了妇女偶像"); //DrawText(hdc,TEXT("我是学霸"),-1,&rect,DT_SINGLELINE|DT_CENTER|DT_VCENTER); } } } } } return TRUE; } case WM_PAINT: hdc=BeginPaint(hwnd,&ps); EndPaint(hwnd,&ps); return 0; case WM_DESTROY: PostQuitMessage(0); return 0; } return DefWindowProc(hwnd,message,wParam,lParam); } #pragma comment(lib, "ole32.lib") //CoInitialize CoCreateInstance需要调用ole32.dll int speak(wchar_t *str) { ISpVoice * pVoice = NULL; ::CoInitialize(NULL); //获取ISpVoice接口: long hr = CoCreateInstance(CLSID_SpVoice, NULL, CLSCTX_ALL, IID_ISpVoice, (void **)&pVoice); hr = pVoice->Speak(str, 0, NULL);
     pVoice->Release();
     pVoice = NULL;
     //千万不要忘记:   
     ::CoUninitialize();
     return TRUE;
 }

程序片段(05):热键.c
内容概要:热键

//01.区分:普通应用程序和Win应用程序!
// 主函数(入口点)的不同:
// 普通应用程序:main();
// Win应用程序:WinMain();
// 注:普通应用程序是含有main();函数的,而Dll动态库当中
// 没有main();函数
#include <Windows.h>

//02.Windows应用程序入口函数格式:
// APIENTRY:
// Win应用程序入口标识("APIENTRY")
// cInstance:
// 当前实例(当前窗口句柄)
// pInstance:
// 父级实例(父级窗口句柄)
// cmdLine:
// 命令行参数
// mcmdshow:
// 控制显示或者隐藏
int APIENTRY WinMain(HINSTANCE cInstance, HINSTANCE pInstance, LPSTR cmdLine, int mcmdshow)
{
    //03.RegisterHotKey();函数格式:
    // NULL:表示给系统窗口注册热键
    // 0x001:表示给组合键编号(16进制整数形式)
    // MOD_CONTROL | MOD_ALT:
    // 表示组合键除开最后一个字符
    // 'M':表示组合键最后一个字符
    RegisterHotKey(NULL, 0x001, MOD_CONTROL | MOD_ALT, 'M');
    RegisterHotKey(NULL, 0x002, MOD_CONTROL | MOD_ALT, 'N');
    RegisterHotKey(NULL, 0x003, MOD_CONTROL | MOD_ALT, 'B');

    //04.基于Windows窗体的消息机制控制:
    // 1.循环获取消息队列当中的消息
    // 2.根据消息对列的获取情况判断
    // 注:获取队列消息+对获取情况判断
    MSG msg;//描述消息的结构体
    while (GetMessage(&msg, NULL, 0, 0))//死循环
    {
        if (WM_HOTKEY == msg.wParam)//热键判断
        {
            if (0x001 == msg.wParam)
            {
                //05.控制按键状态的两个宏指令:
                // KEYEVENTF_EXTENDEDKEY(0)-->按下
                // KEYEVENTF_KEYUP(2)-->松开 
                //06.字母键的大小写:
                // 1.大写字母:w就是大写字母
                // 2.小写字母:(shift+w)组合小写字母键
                keybd_event('S', 0, KEYEVENTF_EXTENDEDKEY, 0);//按下
                keybd_event('S', 0, KEYEVENTF_KEYUP, 0);//松开
                keybd_event('W', 0, 0, 0);
                keybd_event('W', 0, 2, 0);
                keybd_event('W', 0, 0, 0);
                keybd_event('W', 0, 2, 0);
                //07.鼠标事件状态控制:
                // 采用mouse_event();函数控制
                mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);
                mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
            }
            else if (0x002 == msg.wParam)
            {
                //08.按键('\')用VK_OME_102进行表示
                keybd_event(VK_OEM_102, 0, 0, 0);//按下
                keybd_event(VK_OEM_102, 0, 2, 0);//松开
                keybd_event('A', 0, 0, 0);
                keybd_event('A', 0, 2, 0);
                keybd_event('N', 0, 0, 0);
                keybd_event('N', 0, 2, 0);
                keybd_event('G', 0, 0, 0);
                keybd_event('G', 0, 2, 0);//表示顺序按下\ang
                //09.MapVirtualKey映射:
                // 字典虚拟键映射
                keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 0, 0);//按下
                keybd_event(VK_RETURN, MapVirtualKey(VK_RETURN, 0), 2, 0);//松开
            }
            else
            {
                keybd_event('A', 0, 0, 0);
                keybd_event('A', 0, 2, 0);
                keybd_event('D', 0, 0, 0);
                keybd_event('D', 0, 2, 0);
                keybd_event('S', 0, 0, 0);
                keybd_event('S', 0, 2, 0);
                mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);
                mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
            }
            //MessageBoxA(0, "即将推出!", "确定", 0);
            //10.取消注册热键:
            // NULL:表示系统
            // 0x001:组合键标识ID
            //UnregisterHotKey(NULL, 0x001);
            //UnregisterHotKey(NULL, 0x001);
            //UnregisterHotKey(NULL, 0x003);
        }
    }
}
//11.注册的快捷键给系统:
// 于是其他程序在运行过程中;
// 只要触发了快捷键,就会自动触发按键函数
// 触动按键(进行字符输入)

//the modifier of hot key:
//#define MOD_ALT 0x0001
//#define MOD_CONTROL 0x0002
//#define MOD_SHIFT 0x0004
//--------------------------------------
//#define MOD_LEFT 0x8000
//#define MOD_RIGHT 0x4000
//--------------------------------------
//#define MOD_ON_KEYUP 0x0800
//#define MOD_IGORE_ALL_MODIFIER 0x0400

你可能感兴趣的:(CCPP)