2016-08-04 64 views
1

我比較了skia和windows上的gdi繪畫。兩者都繪製了98000條隨機線。令我驚訝的是,skia的效率遠遠低於gdi(skia繪畫花費1600ms,而gdi花費0ms)。我的測試代碼粘貼在下面。有什麼建議麼?爲什麼窗戶上的skia效率很差

bool PaintCompare() { 
    //generate ramdon points 
    std::default_random_engine e(std::chrono::high_resolution_clock::now().time_since_epoch().count()); 
    std::uniform_real_distribution<float> u(10, 500); 
    SkPoint pts[100]; 
    for (int i = 0; i<100; i++) 
     pts[i].set(u(e), u(e)); 
    SkPaint paint; 
    paint.setColor(SkColorSetRGB(255, 0, 0)); 

    //create skia canvas 
    sk_sp<SkSurface> rasterSurface(
     SkSurface::MakeRasterN32Premul(600, 600)); 
    SkCanvas* canvas = rasterSurface->getCanvas(); 

    //draw lines with skia 
    auto start = std::chrono::high_resolution_clock::now(); 
    for (int i = 0; i<1000; i++) 
    { 
     for (int j = 1; j<99; j++) 
     { 
      canvas->drawLine(pts[j].fX, pts[j].fY, pts[j + 1].fX, pts[j + 1].fY, paint); 
     } 
    } 
    auto cost = std::chrono::duration_cast<std::chrono::microseconds>(std::chrono::high_resolution_clock::now() - start); 
    sk_sp<SkImage> img(rasterSurface->makeImageSnapshot()); 
    if (!img) { return false; } 
    SkBitmap skBmp; 
    if (!img->asLegacyBitmap(&skBmp, SkImage::kRO_LegacyBitmapMode)) { 
     return false; 
    } 

    //show bitmap on hdc 
    BITMAPINFO bmi; 
    memset(&bmi, 0, sizeof(bmi)); 
    bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER); 
    bmi.bmiHeader.biWidth = 600; 
    bmi.bmiHeader.biHeight = -600; // top-down image 
    bmi.bmiHeader.biPlanes = 1; 
    bmi.bmiHeader.biBitCount = 32; 
    bmi.bmiHeader.biCompression = BI_RGB; 
    bmi.bmiHeader.biSizeImage = 0; 

    HDC hdc = GetDC(); 
    LPVOID pBits = NULL; 
    HBITMAP hBmp = CreateDIBSection(hdc, &bmi, DIB_RGB_COLORS, &pBits, 0, 0); 
    skBmp.copyPixelsTo(pBits, skBmp.getSize()); 
    CDC memdc; 
    memdc.CreateCompatibleDC(hdc); 
    memdc.SelectBitmap(hBmp); 
    BitBlt(hdc, 0, 0, 600, 600, memdc, 0, 0, SRCCOPY); 
    memdc.DeleteDC(); 


    //draw with gdi 
    CPen pen; 
    pen.CreatePen(PS_SOLID, 1, RGB(0, 255, 0)); 
    RECT rc{ 0,0,600,600 }; 
    CBitmap bmp; 
    bmp.CreateCompatibleBitmap(hdc, 600, 600); 
    memdc.CreateCompatibleDC(hdc); 
    memdc.SelectBitmap(bmp); 
    memdc.FillSolidRect(&rc, RGB(0, 0, 0)); 
    memdc.SelectPen(pen); 
    start = std::chrono::high_resolution_clock::now(); 
    for (int i = 0; i<1000; i++) 
    { 
     for (int j = 1; j<99; j++) 
     { 
      memdc.MoveTo(pts[j].fX, pts[j].fY); 
      memdc.LineTo(pts[j + 1].fX, pts[j + 1].fY); 
     } 
    } 
    auto cost2 = std::chrono::duration_cast<std::chrono::microseconds>(std::chrono::high_resolution_clock::now() - start); 

    //copy bitmap to window 
    BitBlt(hdc, 700, 0, 600, 600, memdc, 0, 0, SRCCOPY); 
    ReleaseDC(hdc); 
    memdc.DeleteDC(); 

    //wchar_t buf[256]; 
    //wsprintf(buf, L"left cost=%I64d, right cost=%I64d", cost.count(), cost2.count()); 
    //GetParent().SetWindowText(buf); 

    //cost == 1596615 microseconds 
    //cost2 == 107253 microseconds 
} 
+0

請發佈[最小,完整,可驗證示例](http://stackoverflow.com/help/mcve)。 –

+1

對不起,我不知道如何在此發佈更多的代碼。我粘貼在上面的代碼是核心。正如你所看到的,我花費在用skia和gdi繪製的時間上。成本和成本2是可以包含花費時間的可變因素。 – qianyi

回答

2

我終於找到了問題。我在調試模式下給出結果!

在調試模式下,skia與柵格後端比gdi慢20倍。 但是在發佈模式下,帶有光柵後端的skia比gdi慢4-5倍。

我有另一個測試,skia使用opengl作爲後端。結果顯示skia和gdi差不多同時度過。 skia比gdi慢大約15%。

相關問題