项目笔记---Windows Service调用Windows API问题
概要
此文来自于最近一个“诡异”的Windows API调用发现Windows Service在调用某些Windows API的过程中失效,在经过漫长的Baidu,之后终于在StackOverFlow上找到了答案,今天希望把这个问题记录下来,方便大家Baidu -。-
需求是什么?

注:PDA一端通过Socket传输一条一维码(如1231223123123123),服务端接收到消息后,在当前(即当前有焦点的任何窗口)模拟键盘敲击一维码。
最开始服务端是用WinForm实现的,测试成功,所以就想当然希望将服务端重构成Windows服务,实现后台监听并处理的功能,可是事与愿违,各位别着急,请继续往下看。
代码实现及问题是什么?
Socket通讯我是用《异步Socket通讯》实现的,这里不详细介绍,先将模拟键盘的代码贴出来,此代码在Windows Form下测试可用:
internal class Win32API
{
#region Constant
private const uint KEYEVENTF_EXTENDEDKEY = 0x1;
private const uint KEYEVENTF_KEYUP = 0x2;
#endregion #region Public Property
public static IServiceLog Log
{
get;
set;
}
#endregion #region External Import
/// <summary>
/// Keybd_events the specified b vk.
/// </summary>
/// <param name="bVk">The b vk.</param>
/// <param name="bScan">The b scan.</param>
/// <param name="dwFlags">The dw flags.</param>
/// <param name="dwExtraInfo">The dw extra information.</param>
[System.Runtime.InteropServices.DllImport("user32.dll")]
static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, uint dwExtraInfo); /// <summary>
/// Messages the box.
/// </summary>
/// <param name="h">The h.</param>
/// <param name="m">The m.</param>
/// <param name="c">The c.</param>
/// <param name="type">The type.</param>
/// <returns></returns>
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern int MessageBox(int h, string m, string c, int type); /// <summary>
/// Gets the focus.
/// </summary>
/// <returns></returns>
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern IntPtr GetFocus(); #endregion #region Public Method
/// <summary>
/// Sends the key.
/// </summary>
/// <param name="str">The string.</param>
public static void SendKey(string str)
{
var charray = str.ToCharArray();
for (int i = ; i < charray.Length; i++)
{
CharToInt(charray[i]);
}
KeyBoardDo(, );
} #endregion #region Private Method
/// <summary>
/// Keys the board do.
/// </summary>
/// <param name="key">The key.</param>
private static void KeyBoardDo(int[] key)
{
foreach (int k in key)
{
keybd_event((byte)k, 0x45, KEYEVENTF_EXTENDEDKEY | , );
}
foreach (int k in key)
{
keybd_event((byte)k, 0x45, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, );
}
} /// <summary>
/// Keys the board do.
/// </summary>
/// <param name="key">The key.</param>
/// <param name="sheft">The sheft.</param>
private static void KeyBoardDo(int key, int sheft)
{
Log.WriteInfo("KeyBoard Simulate Char: " + key);
if (sheft == )
{
keybd_event((byte)sheft, , KEYEVENTF_EXTENDEDKEY | , );
System.Threading.Thread.Sleep();
}
keybd_event((byte)key, , KEYEVENTF_EXTENDEDKEY | , );
System.Threading.Thread.Sleep();
keybd_event((byte)key, , KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, );
System.Threading.Thread.Sleep();
if (sheft == )
{
keybd_event((byte)sheft, 0x45, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, );
System.Threading.Thread.Sleep();
}
} /// <summary>
/// Characters to int.
/// </summary>
/// <param name="ch">The ch.</param>
/// <returns></returns>
private static int CharToInt(char ch)
{
int chint = ;
int sheft = ;
switch (ch)
{
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case '':
chint = ;
KeyBoardDo(chint, );
break;
case 'A':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'B':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'C':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'D':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'E':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'F':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'G':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'H':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'I':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'J':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'K':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'L':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'M':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'N':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'O':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'P':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'Q':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'R':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'S':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'T':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'U':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'V':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'W':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'X':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'Y':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'Z':
chint = ;
KeyBoardDo(chint, sheft);
break;
case 'a':
chint = ;
KeyBoardDo(chint, );
break;
case 'b':
chint = ;
KeyBoardDo(chint, );
break;
case 'c':
chint = ;
KeyBoardDo(chint, );
break;
case 'd':
chint = ;
KeyBoardDo(chint, );
break;
case 'e':
chint = ;
KeyBoardDo(chint, );
break;
case 'f':
chint = ;
KeyBoardDo(chint, );
break;
case 'g':
chint = ;
KeyBoardDo(chint, );
break;
case 'h':
chint = ;
KeyBoardDo(chint, );
break;
case 'i':
chint = ;
KeyBoardDo(chint, );
break;
case 'j':
chint = ;
KeyBoardDo(chint, );
break;
case 'k':
chint = ;
KeyBoardDo(chint, );
break;
case 'l':
chint = ;
KeyBoardDo(chint, );
break;
case 'm':
chint = ;
KeyBoardDo(chint, );
break;
case 'n':
chint = ;
KeyBoardDo(chint, );
break;
case 'o':
chint = ;
KeyBoardDo(chint, );
break;
case 'p':
chint = ;
KeyBoardDo(chint, );
break;
case 'q':
chint = ;
KeyBoardDo(chint, );
break;
case 'r':
chint = ;
KeyBoardDo(chint, );
break;
case 's':
chint = ;
KeyBoardDo(chint, );
break;
case 't':
chint = ;
KeyBoardDo(chint, );
break;
case 'u':
chint = ;
KeyBoardDo(chint, );
break;
case 'v':
chint = ;
KeyBoardDo(chint, );
break;
case 'w':
chint = ;
KeyBoardDo(chint, );
break;
case 'x':
chint = ;
KeyBoardDo(chint, );
break;
case 'y':
chint = ;
KeyBoardDo(chint, );
break;
case 'z':
chint = ;
KeyBoardDo(chint, );
break;
case ';':
chint = ;
KeyBoardDo(chint, );
break;
case '=':
chint = ;
KeyBoardDo(chint, );
break;
case ',':
chint = ;
KeyBoardDo(chint, );
break;
case '-':
chint = ;
KeyBoardDo(chint, );
break;
case '.':
chint = ;
KeyBoardDo(chint, );
break;
case '/':
chint = ;
KeyBoardDo(chint, );
break;
case '`':
chint = ;
KeyBoardDo(chint, );
break;
case '[':
chint = ;
KeyBoardDo(chint, );
break;
case ']':
chint = ;
KeyBoardDo(chint, );
break;
case '\'':
chint = ;
KeyBoardDo(chint, );
break;
}
return chint;
}
#endregion }
同样的代码在Windows Service调用没有任何效果,原因就出在这里:Session 0 Isolation,简而言之就是在Win XP以及之前的windows服务(系统服务和用户编写的服务)都运行在最高权限的Session 0 中,但是在Vista以及之后的系统中用户所编写的服务被隔离开,为的是系统安全性考虑,被隔离开的服务中受影响的是一些基于图形界面的系统调用都将失效,如“User32.dll”中的MessageBox调用,同时也包含本文中的模拟键盘的keybd_event调用。
有兴趣的朋友可以自行测试下,原理很简单,这里就不给出更多的代码了。

注:Vista之后系统服务权限图。
结语
说点题外话,其实这个问题就是一句话的事:Win7下Windows服务不能调用键盘模拟事件,为什么在百度搜来搜去,基本没有什么有价值的信息,包括CSDN等等。后来,不得不开着翻墙软件去Google,或者再去StackOverFlow提问,搜索问题,这里也不去抱怨什么,只希望作为码农的我以及跟我一样的人能更熟练的掌握英语,Happy To Search On Google.
引用
Session 0 Isolation: https://msdn.microsoft.com/en-us/library/windows/hardware/dn653293(v=vs.85).aspx
StackOverFlow Discuss:http://stackoverflow.com/questions/16959963/how-to-reset-windows-idle-timer-through-a-windows-service
项目笔记---Windows Service调用Windows API问题的更多相关文章
- C# 编写Windows Service(windows服务程序)
C# 编写Windows Service(windows服务程序) Windows Service简介: 一个Windows服务程序是在Windows操作系统下能完成特定功能的可执行的应用程序.W ...
- C# Windows Service调用IBM Lotus Notes发送邮件
近日研究了下IBM Lotus Mail,这货果然是麻烦,由于公司策略,没有开放smtp,很多系统邮件都没有办法发送,于是入手google学习Lotus Mail,想做成Windows服务,提供wcf ...
- windows service宿主web api使用"依赖注入"和“控制反转”的技术实践
前言 自从几年前抛弃wcf,使用web api 来做服务器端开发之后,就不再迷惑了.但是因为本来从事传统行业管理软件开发,一般都以分布式应用开发为主.纯BS还是比较少,于是比较喜欢用windows s ...
- 如何将.NET 4.0写的Windows service和Web API部署到docker上面
Web API. 看这篇文章: https://docs.microsoft.com/en-us/aspnet/mvc/overview/deployment/docker-aspnetmvc Win ...
- C# 编写Windows Service(windows服务程序)【转载】
[转]http://www.cnblogs.com/bluestorm/p/3510398.html Windows Service简介: 一个Windows服务程序是在Windows操作系统下能完成 ...
- C# 创建Windows Service(Windows服务)程序
本文介绍了如何用C#创建.安装.启动.监控.卸载简单的Windows Service 的内容步骤和注意事项. 一.创建一个Windows Service 1)创建Windows Service项目 2 ...
- Windows Desktop 调用 WinRT api
<Reference Include="Windows"> <HintPath>..\..\..\..\..\..\Program Files (x86)\ ...
- HTML5项目笔记4:使用Audio API设计绚丽的HTML5音乐播放器
HTML5 有两个很炫的元素,就是Audio和 Video,可以用他们在页面上创建音频播放器和视频播放器,制作一些效果很不错的应用. 无论是视屏还是音频,都是一个容器文件,包含了一些音频轨道,视频轨道 ...
- Create a new Windows service on windows server2012
netsh http add iplisten ipaddress=0.0.0.0:15901 sc.exe create "FPPService" binPath= " ...
随机推荐
- C++ new(1)
如果找工作的同学看一些面试的书,我相信都会遇到这样的题:sizeof 不是函数,然后举出一堆的理由来证明 sizeof 不是函数.在这里,和 sizeof 类似,new 和 delete 也不是函数, ...
- 【软件使用】Windows下的Objective-C集成开发环境搭建(IDE)
Objective-C是苹果软件的编程语言,想要上机学习.调试,有一个集成开发环境(IDE)方便很多.有三类方法搭建Objective-C的集成开发环境: 1) 使用苹果的平台,集成开发环境使用X ...
- UIView.frame的骗局
如果你刚刚开始接触IOS编程, 刚刚接触UIKit, 肯定会被 frame, bounds, center, layer.anchorPoint, layer.position 这些乱七八糟得属性折腾 ...
- java GUI,贷款服务器
本习题来自<java语言程序设计--进阶篇>第30章,网络编程的习题. 题目描述:为一个客户端编写一个服务器.客户端向服务器发送贷款信息(年利率.贷款年限和贷款总额).服务器计算月偿还额和 ...
- Redis漏洞?阿里云被攻击!
今天运维那边过来说阿里云服务器进程被占用很多,后来查了一下进程发现了这个玩意: 小编我看不懂,经运维先森仔细研究,发现这是被注入进来的一个进程,服务器被当成了肉鸡,专门用来跑比特币的,这样对方就不需要 ...
- 第2章 面向对象的设计原则(SOLID):2_里氏替换原则(LSP)
2. 里氏替换原则(Liskov Substitution Principle,LSP) 2.1 定义 (1)所有使用基类的地方必须能透明地使用子类替换,而程序的行为没有任何变化(不会产生运行结果错误 ...
- 利用appscan进行自动化定期安全测试
Appscan的强大众所周知,如果可以自动执行定期安全测试,岂不是美事一件? 事实上,appscan提供了计划扫描的选项,配合windows的计划任务,可以按需设定. 1.打开appscan中的“工具 ...
- sublime配置全攻略
大家好,今天给大家分享一款编辑器:sublime text2 我用过很多编辑器, EditPlus.EmEditor.Notepad++.Notepad2.UltraEdit.Editra.V ...
- JavaScript---基本语法
字符串方法:str.lengthstr.charAt(i):取字符串中的某一个;str.indexOf('e');找第一个出现的位置;找不到返回-1;str.lastIndexOf('e'):找最后一 ...
- PHP基础18:require和include
<?phph //1.通过 include 或 require 语句,可以将 PHP 文件的内容插入另一个 PHP 文件(在服务器执行它之前) //include 和 require 语句是相同 ...