5

我正在用python制作一个屏幕截图程序。我目前的问题是 PIL.ImageGrab.grab() 给了我与 2 秒后相同的输出。例如,因为我想我不清楚,在下面的程序中,几乎所有的图像都是相同的,具有相同的 Image.tostring() 值,即使我在 PIL.ImageGrab 期间移动了我的屏幕。抓取循环正在执行。

>>> from PIL.ImageGrab import grab
>>> l = []
>>> import time
>>> for a in l:
        l.append(grab())
        time.sleep(0.01)


    >>> for a in range(0, 30):
        l.append(grab())
        time.sleep(0.01)


>>> b = []
>>> for a in l:
        b.append(a.tostring())


>>> len(b)
30
>>> del l
>>> last = []
>>> a = 0
>>> a = -1
>>> last = ""
>>> same = -1
>>> for pic in b:
        if b == last:
            same = same + 1
        last = b


>>> same
28
>>> 

这是一个问题,因为所有图像都相同,但 1. 30 中的 1 是不同的。这将制作一个绝对可怕的质量视频。请告诉我是否有比 PIL.ImageGrab.grab() 更好的替代品。我需要捕获整个屏幕。谢谢!

编辑:

到目前为止,看起来最好的选择是使用 pywin32。我现在正在使用它,但它非常慢。我并不真正关心兼容性,因为现在这个项目是个人的。每次算出程序的帧率,pywin32都这么慢,处于负数。请告诉我是否有更快的替代方案。谢谢!

4

2 回答 2

5

PIL.ImageGrab 有很多替代品。

如果您想要跨平台,几乎每个主要的窗口库都具有屏幕抓取功能。这是使用PyQt4的示例:

import sys
from PyQt4.QtGui import QPixmap, QApplication
app = QApplication(sys.argv)
QPixmap.grabWindow(QApplication.desktop().winId()).save('screenshot.jpg', 'jpg')

如果您想要与 Qt 认为的“桌面”的默认设置不同的东西,您需要深入研究 PyQt 或 Qt 文档。

缺点是这些跨平台的窗口库通常在安装、分发,甚至有时你如何构建程序方面都非常繁重。

另一种方法是使用特定于 Windows 的代码。虽然可以通过 ctypes 直接访问 Windows API,但更简单的解决方案是PyWin32

import win32gui, win32ui, win32con, win32api
hwin = win32gui.GetDesktopWindow()
width = win32api.GetSystemMetrics(win32con.SM_CXVIRTUALSCREEN)
height = win32api.GetSystemMetrics(win32con.SM_CYVIRTUALSCREEN)
left = win32api.GetSystemMetrics(win32con.SM_XVIRTUALSCREEN)
top = win32api.GetSystemMetrics(win32con.SM_YVIRTUALSCREEN)
hwindc = win32gui.GetWindowDC(hwin)
srcdc = win32ui.CreateDCFromHandle(hwindc)
memdc = srcdc.CreateCompatibleDC()
bmp = win32ui.CreateBitmap()
bmp.CreateCompatibleBitmap(srcdc, width, height)
memdc.SelectObject(bmp)
memdc.BitBlt((0, 0), (width, height), srcdc, (left, top), win32con.SRCCOPY)
bmp.SaveBitmapFile(memdc, 'screenshot.bmp')

与 Win32 一样,您必须准确指定“桌面”的含义;此版本为当前会话指定“虚拟屏幕”,如果您在控制台本地运行,则它是所有监视器的组合,如果您在终端服务上运行,则为远程视图。如果您想要不同的东西,您必须阅读 MSDN 上的相关 Win32 文档。但在大多数情况下,从该文档转换为 PyWin32 是微不足道的。

(顺便说一句,即使我说“Win32”,同样适用于 Win64。)

于 2012-09-25T21:36:14.210 回答
4

现代跨平台解决方案是使用Python MSS

from mss import mss
from PIL import Image

def capture_screenshot():
    # Capture entire screen
    with mss() as sct:
        monitor = sct.monitors[1]
        sct_img = sct.grab(monitor)
        # Convert to PIL/Pillow Image
        return Image.frombytes('RGB', sct_img.size, sct_img.bgra, 'raw', 'BGRX')

img = capture_screenshot()
img.show()

在我的慢速笔记本电脑上,此功能可以以高达 27 fps 的速度返回屏幕截图。

于 2018-07-02T07:04:39.900 回答