简体   繁体   English

Powershell代码在powershell_ise上运行良好,但在powershell上中断

[英]Powershell code works good on powershell_ise but breaks on powershell

I'm trying to write a code that "blinks" the lock key leds really fast (using -comObject to sendkeys). 我正在尝试编写一个非常快的“闪烁”锁定键指示灯的代码(使用-comObject来发送键)。 The code run really slow on Powershell (from CMD) and miss some keypress, but seems to work great on Powershell_ise. 该代码在Powershell(来自CMD)上运行非常慢,并且错过了一些按键操作,但是在Powershell_ise上似乎运行良好。

The code reads a file to binary and then transfer each bit to num/scroll lock. 该代码将文件读取为二进制文件,然后将每个位传输到num / scroll锁。 This needs to run really fast - as fast as I can. 这需要运行得非常快-尽快。

This is the code: 这是代码:

$wsh = New-Object -ComObject "WScript.Shell"
$bytes = [Byte[]] (Get-Content  $env:temp\temp1536.txt -Encoding Byte -ReadCount 0) | ForEach-Object {[System.Convert]::ToString($_,2)}
##($i=0; $i -le $byte.length; $i++){

 foreach ($byte in $bytes) {
 #$byte;
   while($byte.length -ne 1 ){
    if($byte[1] -eq '1'){
        #echo "1";
        $wsh.SendKeys('{SCROLLLOCK}');
        [System.Threading.Thread]::Sleep(40);   
        $wsh.SendKeys('{SCROLLLOCK}');
    } Else {
        #echo "0";
        $wsh.SendKeys('{NUMLOCK}');
        [System.Threading.Thread]::Sleep(40);
        $wsh.SendKeys('{NUMLOCK}');
    }
    $byte=$byte.Substring(1);
    [System.Threading.Thread]::Sleep(50);


   }
   #echo " ";
   #echo "1";

   $wsh.SendKeys('{CAPSLOCK}');
   [System.Threading.Thread]::Sleep(55);

   $wsh.SendKeys('{CAPSLOCK}');
   [System.Threading.Thread]::Sleep(20);

 }

Anyone knows why this happens? 有人知道为什么会这样吗?

EDIT: I added a video showing the lag of the lock key blinking on Powershell Console Vs. 编辑:我添加了一个视频,该视频显示了Powershell Console Vs上的锁定键闪烁的滞后时间。 Powershell_ISE http://youtu.be/OnOmr50OBhs Powershell_ISE http://youtu.be/OnOmr50OBhs

I tried this on Powershell V3.0/4.0 on Windows 7 我在Windows 7的Powershell V3.0 / 4.0上尝试过

I used this text file name -'temp1536.txt' in %temp% folder The file is imported to binary and then light the led accordingly. 我在%temp%文件夹中使用了此文本文件名-'temp1536.txt'。文件已导入到二进制文件,然后相应地点亮了led。

$bytes = [Byte[]] (Get-Content  $env:temp\temp1536.txt -Encoding Byte -ReadCount 0) | ForEach-Object {[System.Convert]::ToString($_,2)}

I couldn't get your code working (it never steps into if($byte[1] -eq '1') ), but here is the version using keybd_event \\ GetKeyState Win32 API that works fine in my ISE and console. 我无法使您的代码正常工作(它永远不会进入if($byte[1] -eq '1') ),但这是使用keybd_event \\ GetKeyState Win32 API的版本,在我的ISE和控制台中运行良好。 Adapted from code posted here . 改编自此处发布的代码。

$Keyboard = @'
using System.Runtime.InteropServices;

namespace My
{
    public class Keyboard
    {
        private const byte VK_SCROLL = 0x91;
        private const byte VK_NUMLOCK = 0x90;
        private const byte VK_CAPITAL = 0x14;
        private const uint KEYEVENTF_KEYUP = 0x2;

        [DllImport("user32.dll", EntryPoint = "keybd_event", SetLastError = true)]
        static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, uint dwExtraInfo);

        [DllImport("user32.dll", EntryPoint = "GetKeyState", SetLastError = true)]
        static extern short GetKeyState(uint nVirtKey);

        // SCROLLLOCK
        public static void SetScrollLockKey(bool newState)
        {
            bool scrollLockSet = GetKeyState(VK_SCROLL) != 0;
            if (scrollLockSet != newState)
            {
                keybd_event(VK_SCROLL, 0, 0, 0);
                keybd_event(VK_SCROLL, 0, KEYEVENTF_KEYUP, 0);
            }
        }
        public static bool GetScrollLockState()
        {
            return GetKeyState(VK_SCROLL) != 0;
        }

        // NUMLOCK
        public static void SetNumLockKey(bool newState)
        {
            bool scrollLockSet = GetKeyState(VK_NUMLOCK) != 0;
            if (scrollLockSet != newState)
            {
                keybd_event(VK_NUMLOCK, 0, 0, 0);
                keybd_event(VK_NUMLOCK, 0, KEYEVENTF_KEYUP, 0);
            }
        }

        public static bool GetNumLockState()
        {
            return GetKeyState(VK_NUMLOCK) != 0;
        }

        // CAPSLOCK
        public static void SetCapsLockKey(bool newState)
        {
            bool scrollLockSet = GetKeyState(VK_NUMLOCK) != 0;
            if (scrollLockSet != newState)
            {
                keybd_event(VK_CAPITAL, 0, 0, 0);
                keybd_event(VK_CAPITAL, 0, KEYEVENTF_KEYUP, 0);
            }
        }

        public static bool GetCapsLockState()
        {
            return GetKeyState(VK_CAPITAL) != 0;
        }
    }
}
'@

Add-Type -TypeDefinition $Keyboard -ErrorAction Stop

$Bytes = Get-Content -Path '.\led.txt' -Encoding Byte

foreach ($byte in $Bytes) {
    if($byte)
    {
        [My.Keyboard]::SetScrollLockKey($true)
        [System.Threading.Thread]::Sleep(40)
        [My.Keyboard]::SetScrollLockKey($false)
    }
    else
    {
        [My.Keyboard]::SetNumLockKey($true)
        [System.Threading.Thread]::Sleep(40)
        [My.Keyboard]::SetNumLockKey($false)
    }

    [My.Keyboard]::SetCapsLockKey($true)
    [System.Threading.Thread]::Sleep(55)

    [My.Keyboard]::SetNumLockKey($false)
    [System.Threading.Thread]::Sleep(20)
}

It seems to be a problem in some PCs only. 仅在某些PC中这似乎是一个问题。 Can't really understand why. 真的不明白为什么。 But I got it to work fine on some computers. 但是我知道它可以在某些计算机上正常工作。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM