请问有人遇到过"System.InvalidOperationException: 进程已退出,因此请求的信息不可用。"吗?

一树秋叶 2016-04-11 03:10:08
正确流程应该是:
1,软件在开机时自动启动
2,启动后点击某个textbox文本输入框
3,点击后启动并弹出系统自带的OSK(屏幕键盘)软件
4,修改OSK软件的位置

现在的问题是开机启动并点击textbox后可以弹出OSK软件,但是还没有改变位置就报了如下错误:
"System.InvalidOperationException: 进程已退出,因此请求的信息不可用。
at System.Diagnostics.Process.EnsureState(State state)
at System.Diagnostics.Process.get_MainWindowHandle()
at textBox1_MouseClick(object sender, MouseEventArgs e)
at.............."


研究后发现一个更奇怪的现象就是, 报了错之后强制关闭,再打开还报同样的错,再强制关闭,第三次第四次第五次。。。打开之后就都正常了,不会报错了而且位置也按预定代码执行了。

下面是代码:
private void textBox1_MouseClick(object sender, MouseEventArgs e)
{
bool have = false;//是否已经启动了OSK
Process[] pro = Process.GetProcesses();//获取已开启的所有进程

for (int i = 0; i < pro.Length; i++)
{
//判断此进程是否是要查找的进程
if (pro[i].ProcessName.ToString().ToLower() == "osk")
{
SetWindowPos(pro[i].MainWindowHandle, HWND_TOPMOST, 0, 0, 800, 600, SWP_NOSIZE | SWP_NOZORDER | SWP_NOZORDER | SWP_SHOWWINDOW);//如果OSK存在,则设置OSK出现在(0,0)位置
SetForegroundWindow(this.Handle);
have = true;
break;
}
}

if (have == false)
{
bool haveWarn = false;//是否报过错误提醒了
bool haveOsk = false;//是否成功启动了OSK软件

Process p = Process.Start(Application.StartupPath + "\\osk.exe");
p.WaitForInputIdle();
Thread.Sleep(500);
System.Timers.Timer wait3s = new System.Timers.Timer(3000);//3s启动不了OSK则报错误提醒
wait3s.Elapsed += (s, ee) =>
{
wait3s.Enabled = false;
haveWarn = true;
this.Invoke(new Action(() =>
{
Common.FrmWarning myWarningWindow = new Common.FrmWarning("错误提醒", "无法打开虚拟键盘,请检查后重新尝试。");
myWarningWindow.Owner = this;
myWarningWindow.ShowDialog();
}));
};

while (haveOsk == false)
{
if (wait3s.Enabled == false)
{
wait3s.Enabled = true;
}

//Process[] pro2 = Process.GetProcesses();
//foreach(Process h in pro2)
//{
// if (h.ProcessName.ToString().ToLower() == "osk")
// {
// haveOsk = true;
// }
//}

if (p.HasExited == false)
{
haveOsk = true;
}

if (haveWarn)
{
wait3s.Enabled = false;
return;
}

Application.DoEvents();
}

wait3s.Enabled = false;
SetWindowPos(p.MainWindowHandle, HWND_TOPMOST, 0, 0, 800, 600, SWP_NOSIZE | SWP_NOZORDER | SWP_NOZORDER | SWP_SHOWWINDOW);//如果OSK存在,则设置OSK出现在(0,0)位置
//MoveWindow(p.MainWindowHandle, 0, 0, 800, 200, true);
SetForegroundWindow(this.Handle);
have = false;
}
}


我发现错误应该是产生在没有OSK则启动OSK后的SetWindowPos方法的第一个参数p.MainWindowHandle上,可是我不知道应该怎么修改才能确保开机启动的那次就可以不报错正常运行。
求问大家是我程序逻辑错误还是系统原因??
感激不尽!
...全文
2958 8 打赏 收藏 转发到动态 举报
写回复
用AI写文章
8 条回复
切换为时间正序
请发表友善的回复…
发表回复
一树秋叶 2016-04-12
  • 打赏
  • 举报
回复
引用 6 楼 bbwolfcool 的回复:
你在一个按钮里面写那么多代码,不用异步,不是造死么,还搞轮询,一个try catch没有,服了, 你把主线程阻塞住,Sleep了,就是程序不报错,又咋样,这玩意能用么?
谢谢您的提醒。我现在就是要测试可能的所有异常,所以还没有加try catch,可能这样的思路是错误的? Thread.Sleep(500);这一句是因为去掉会出现一个问题,也是经过询问后某位朋友给出的解答。愿意听听您对这里的看法,向您学习一下。
一树秋叶 2016-04-12
  • 打赏
  • 举报
回复
引用 4 楼 dongxinxi 的回复:
简单的办法 if (pro[i].ProcessName.ToString().ToLower() == "osk" && pro[i].Responding) Responding属性是向目标进程发送一个window测试消息,如果超过5秒还未响应则视为进程无响应,此时如果你去SetPosition基本也是得不到响应的,所以你可以选择继续等待个2秒,再次判断Responding,直到为true你再设置位置就可以了
多谢多谢!解释的很明朗,我去试一试。
bbwolfcool 2016-04-11
  • 打赏
  • 举报
回复
你在一个按钮里面写那么多代码,不用异步,不是造死么,还搞轮询,一个try catch没有,服了, 你把主线程阻塞住,Sleep了,就是程序不报错,又咋样,这玩意能用么?
  • 打赏
  • 举报
回复
你可以直接使用Process[] pro = Process.GetProcessesByName("osk"); 下面就不用判断进程名了
  • 打赏
  • 举报
回复
简单的办法 if (pro[i].ProcessName.ToString().ToLower() == "osk" && pro[i].Responding) Responding属性是向目标进程发送一个window测试消息,如果超过5秒还未响应则视为进程无响应,此时如果你去SetPosition基本也是得不到响应的,所以你可以选择继续等待个2秒,再次判断Responding,直到为true你再设置位置就可以了
  • 打赏
  • 举报
回复
进程从启动到窗口初始化完毕是需要个过程的(虽然很短) EnsureState()正是检测访问权限,以及对方进程和窗口的状态,进程刚启动时,有可能主窗口句柄都还没来得及创建 过一会儿又可以了,因为操作系统刚开机时大量的CPU时间都优先分配系统内核操作了,等过一会儿,系统喘上口气了,你的用户进程才变得流畅起来
一树秋叶 2016-04-11
  • 打赏
  • 举报
回复
求大神帮助啊。。。。。
一树秋叶 2016-04-11
  • 打赏
  • 举报
回复
自顶一下。。。。。

111,124

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Creator Browser
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧