C# Bitmap圖像處理加速的實現(xiàn)
BitmapData類
BitmapData類專門用于位圖處理,與Bitmap的不同點在于,它使用指針直接修改內(nèi)存,而Bitmap是使用SetPixel()方法間接修改顏色,因此其效率遠遠超過SetPixel()
傳統(tǒng)代碼
以灰度處理為例,為了便于演示,此處的灰度算法采用 Gray=(R+G+B) / 3
private void Gray_Tradition()
{
for(int i = 0; i < bitmap.Width; i++)
{
for(int j = 0; j < bitmap.Height; j++)
{
Color color = bitmap.GetPixel(i, j);
int RGB = (color.R + color.G + color.B) / 3;
bitmap.SetPixel(i, j, Color.FromArgb(255, RGB, RGB, RGB));
}
}
}
使用BitmapData的代碼
private void Gray_BitmapData()
{
int width = bitmap.Width, height = bitmap.Height;//圖片的寬度和高度
//在內(nèi)存中以讀寫模式鎖定Bitmap
BitmapData bitmapData = bitmap.LockBits(
new Rectangle(0, 0, width, height),
ImageLockMode.ReadWrite,
PixelFormat.Format24bppRgb);
//圖片像素點數(shù)組的長度,由于一個像素點占了3個字節(jié),所以要乘上3
int size = width * height * 3;
//緩沖區(qū)數(shù)組
byte[] srcArray = new byte[size];
//獲取第一個像素的地址
IntPtr ptr = bitmapData.Scan0;
//把像素值復制到緩沖區(qū)
Marshal.Copy(ptr, srcArray, 0, size);
int p;
for (int i = 0; i < width; i++)
{
for (int j = 0; j < height; j++)
{
//定位像素點位置
p = j * width * 3 + i * 3;
//計算灰度值
byte color = (byte)((srcArray[p] + srcArray[p + 1] + srcArray[p + 2]) / 3);
srcArray[p] = srcArray[p + 1] = srcArray[p + 2] = color;
}
}
//從緩沖區(qū)復制回BitmapData
Marshal.Copy(srcArray, 0, ptr, size);
//從內(nèi)存中解鎖
bitmap.UnlockBits(bitmapData);
}
效率對比
代碼
private void onTest()
{
double t1, t2;
Stopwatch watch = new Stopwatch();
watch.Start();
Gray_BitmapData();
watch.Stop();
t1 = watch.Elapsed.TotalMilliseconds;
watch.Reset();
watch.Start();
Gray_Tradition();
watch.Stop();
t2 = watch.Elapsed.TotalMilliseconds;
MessageBox.Show("BitmapData=" + (long)t1 + "\nTradition=" + (long)t2);
}
圖片信息

耗時

可以看到傳統(tǒng)方法的耗時是使用BitmapData方法的106倍,需要整整14秒,而BitmapData僅用了0.1秒
GPU加速
使用CUDA生成dll后,可以在GPU上高效處理圖像,但是這種方式需要使用dll,而且異常繁瑣,因此只適合對效率有極高要求時使用
生成Dll
#include "cuda_runtime.h"
#include "device_launch_parameters.h"
#include <stdio.h>
#include <Windows.h>
__global__ void DoInKernel(byte* o, int num)
{
int i = blockIdx.x * blockDim.x + threadIdx.x;
if (i >= num) return;
byte* ori = o + i * 3;
ori[0] = ori[1] = ori[2] = (ori[0] + ori[1] + ori[2]) / 3;
}
extern "C" _declspec(dllexport) void Gray(byte * oriArray, int num) {
int size = num * 3 * sizeof(byte);
byte* dev_ori;
//在GPU上分配內(nèi)存
cudaMalloc((void**)&dev_ori, size);
//把數(shù)組復制到顯存
cudaMemcpy(dev_ori, oriArray, size, cudaMemcpyHostToDevice);
//計算
DoInKernel << <num / 1024 + 1, 1024 >> > (dev_ori, num);
//從顯存復制到內(nèi)存
cudaMemcpy(oriArray, dev_ori, size, cudaMemcpyDeviceToHost);
//釋放
cudaFree(dev_ori);
}
實際上GPU的thread和block數(shù)量應該根據(jù)實際數(shù)組大小來動態(tài)調(diào)整,但是這里為了演示方便,直接定義1024個線程
調(diào)用Dll
[DllImport("CUDA.dll", EntryPoint = "Gray", CallingConvention = CallingConvention.Cdecl)]
public static extern void Gray(IntPtr ori, int num);
此時不需要定義緩沖區(qū)數(shù)組了,可以直接把數(shù)據(jù)復制到顯存中使用
private void Gray_GPU()
{
int width = bitmap.Width, height = bitmap.Height;//圖片的寬度和高度
//在內(nèi)存中以讀寫模式鎖定Bitmap
BitmapData bitmapData = bitmap.LockBits(
new Rectangle(0, 0, width, height),
ImageLockMode.ReadWrite,
PixelFormat.Format24bppRgb);
//圖片像素點數(shù)組的長度,由于一個像素點占了3個字節(jié),所以要乘上3
int size = width * height * 3;
//獲取第一個像素的地址
IntPtr ptr = bitmapData.Scan0;
Gray(ptr, width * height);
//從內(nèi)存中解鎖
bitmap.UnlockBits(bitmapData);
pictureBox1.Refresh();
}
耗時
由于加載dll需要時間,因此第二次執(zhí)行的耗時才是真正的GPU執(zhí)行時間

僅用了34毫秒
到此這篇關(guān)于C# Bitmap圖像處理加速的實現(xiàn)的文章就介紹到這了,更多相關(guān)C# Bitmap圖像處理加速內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
WPF實現(xiàn)XAML轉(zhuǎn)圖片的示例詳解
這篇文章主要為大家詳細介紹了如何利用WPF實現(xiàn)XAML轉(zhuǎn)圖片,文中的示例代碼講解詳細,對我們學習或工作有一定幫助,感興趣的小伙伴可以了解一下2022-11-11
C#調(diào)用Win32的API函數(shù)--User32.dll
這篇文章主要介紹了C#調(diào)用Win32_的API函數(shù)--User32.dll,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-01-01
基于C#實現(xiàn)將圖片轉(zhuǎn)換為PDF文檔
將圖片(JPG、PNG)轉(zhuǎn)換為PDF文件可以幫助我們更好地保存和分享圖片,所以本文將介紹如何使用C#將JPG/PNG圖片轉(zhuǎn)換為PDF文檔,需要的可以參考下2024-12-12
小白2分鐘學會Visual Studio如何將引用包打包到NuGet上
這篇文章主要介紹了小白2分鐘學會Visual Studio如何將引用包打包到NuGet上,只需兩步完成打包上傳操作,需要的朋友可以參考下2021-09-09
c#解析jobject的數(shù)據(jù)結(jié)構(gòu)
這篇文章介紹了c#解析jobject數(shù)據(jù)結(jié)構(gòu)的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-07-07

