如何读取屏幕像素?

15
我想读取一个矩形区域或整个屏幕的像素,就好像按下了截图按钮。如何实现?
编辑:可行代码:
void CaptureScreen(char *filename)
{
    int nScreenWidth = GetSystemMetrics(SM_CXSCREEN);
    int nScreenHeight = GetSystemMetrics(SM_CYSCREEN);
    HWND hDesktopWnd = GetDesktopWindow();
    HDC hDesktopDC = GetDC(hDesktopWnd);
    HDC hCaptureDC = CreateCompatibleDC(hDesktopDC);
    HBITMAP hCaptureBitmap = CreateCompatibleBitmap(hDesktopDC, nScreenWidth, nScreenHeight);
    SelectObject(hCaptureDC, hCaptureBitmap); 

    BitBlt(hCaptureDC, 0, 0, nScreenWidth, nScreenHeight, hDesktopDC, 0,0, SRCCOPY|CAPTUREBLT); 

    BITMAPINFO bmi = {0}; 
    bmi.bmiHeader.biSize = sizeof(bmi.bmiHeader); 
    bmi.bmiHeader.biWidth = nScreenWidth; 
    bmi.bmiHeader.biHeight = nScreenHeight; 
    bmi.bmiHeader.biPlanes = 1; 
    bmi.bmiHeader.biBitCount = 32; 
    bmi.bmiHeader.biCompression = BI_RGB; 

    RGBQUAD *pPixels = new RGBQUAD[nScreenWidth * nScreenHeight]; 

    GetDIBits(
        hCaptureDC, 
        hCaptureBitmap, 
        0,  
        nScreenHeight,  
        pPixels, 
        &bmi,  
        DIB_RGB_COLORS
    );  

    // write:
    int p;
    int x, y;
    FILE *fp = fopen(filename, "wb");
    for(y = 0; y < nScreenHeight; y++){
        for(x = 0; x < nScreenWidth; x++){
            p = (nScreenHeight-y-1)*nScreenWidth+x; // upside down
            unsigned char r = pPixels[p].rgbRed;
            unsigned char g = pPixels[p].rgbGreen;
            unsigned char b = pPixels[p].rgbBlue;
            fwrite(fp, &r, 1);
            fwrite(fp, &g, 1);
            fwrite(fp, &b, 1);
        }
    }
    fclose(fp);

    delete [] pPixels; 

    ReleaseDC(hDesktopWnd, hDesktopDC);
    DeleteDC(hCaptureDC);
    DeleteObject(hCaptureBitmap);
}

2
如果答案对您有帮助,请不要忘记接受它们。 - johnsyweb
我知道,我还没有让它正常工作。 - Newbie
也许您需要添加更多信息,例如您尝试了什么以及出现了什么问题。 - johnsyweb
2
你使用了哪些包括/编译器等工具? - Paul
@Paul 如果我猜的话,应该是在Visual Studio上编译的winuser.h和wingdi.h,但我可能漏掉了什么。 - Aidan
5个回答

14

从你的代码开始,省略错误检查...

// Create a BITMAPINFO specifying the format you want the pixels in.
// To keep this simple, we'll use 32-bits per pixel (the high byte isn't
// used).
BITMAPINFO bmi = {0};
bmi.bmiHeader.biSize = sizeof(bmi.bmiHeader);
bmi.bmiHeader.biWidth = nScreenWidth;
bmi.bmiHeader.biHeight = nScreenHeight;
bmi.bmiHeader.biPlanes = 1;
bmi.bmiHeader.biBitCount = 32;
bmi.bmiHeader.biCompression = BI_RGB;

// Allocate a buffer to receive the pixel data.
RGBQUAD *pPixels = new RGBQUAD[nScreenWidth * nScreenHeight];

// Call GetDIBits to copy the bits from the device dependent bitmap
// into the buffer allocated above, using the pixel format you
// chose in the BITMAPINFO.
::GetDIBits(hCaptureDC,
            hCaptureBitmap,
            0,  // starting scanline
            nScreenHeight,  // scanlines to copy
            pPixels,  // buffer for your copy of the pixels
            &bmi,  // format you want the data in
            DIB_RGB_COLORS);  // actual pixels, not palette references

// You can now access the raw pixel data in pPixels.  Note that they are
// stored from the bottom scanline to the top, so pPixels[0] is the lower
// left pixel, pPixels[1] is the next pixel to the right,
// pPixels[nScreenWidth] is the first pixel on the second row from the
// bottom, etc.

// Don't forget to free the pixel buffer.
delete [] pPixels;

我应该为hCaptureBitmap参数输入什么? - Newbie
另外,我应该放哪个窗口的HDC,我的桌面还是自己程序的窗口? - Newbie
你为什么删除了原来的截图代码?我发布的内容应该是添加到你原有的代码中,而不是替换它。这就是为什么我使用了你的变量名hCaptureBitmaphCaptureDC - Adrian McCarthy
找到了代码,现在我更新了,但仍然返回黑色图像。 - Newbie

3
重新阅读您的问题,似乎我们在屏幕截图方面有些离题了。如果您只想检查屏幕上的一些像素,可以使用GetPixel。具体请参考GetPixel
HDC hdcScreen = ::GetDC(NULL);
COLORREF pixel = ::GetPixel(hdcScreen, x, y);
ReleaseDC(NULL, hdcScreen);
if (pixel != CLR_INVALID) {
  int red = GetRValue(pixel);
  int green = GetGValue(pixel);
  int blue = GetBValue(pixel);
  ...
} else {
  // Error, x and y were outside the clipping region.
}

如果你要读取很多像素,最好使用屏幕截图,然后使用GetDIBits。调用GetPixel太多次会很慢。


是的,我想读取一个像素矩形(或者如果整个屏幕更容易一些就读取整个屏幕)。我不知道如何使用GetDIBits函数,我无法理解任何参数术语,也不知道它是如何工作的。 - Newbie
我已经让你的函数工作了,但我想使用更有效的方法来读取多个像素。我无法让GetDIBits函数工作。 - Newbie

2

您可以使用BitBlt()函数来截取屏幕截图。通过nWidth和nHeight参数设置截图大小,通过nXSrc和nYSrc参数设置左上角位置。


1
你可以使用以下代码来读取屏幕像素:
HWND desktop = GetDesktopWindow();
HDC desktopHdc = GetDC(desktop);
COLORREF color = GetPixel(desktopHdc, x, y);

0

HBITMAP不是指针或数组,它是由Windows管理的句柄,仅对Windows有意义。您必须请求Windows将像素复制到某个位置以供使用。

要获取单个像素值,甚至不需要位图,可以使用GetPixel。如果需要访问多个像素,则速度会很慢。

要将位图复制到可以访问的内存中,请使用GetDIBits函数


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接