gpt4 book ai didi

C# - 转换 8 位或 16 位灰度原始像素数据

转载 作者:行者123 更新时间:2023-11-30 14:16:52 24 4
gpt4 key购买 nike

我需要能够将 8 位或 16 位灰度像素数据转换为 .NET 框架可以支持的文件格式。

我可用的数据是宽度、高度、方向(左下)和像素格式,即 4096 级灰度(12 位分辨率),每个像素 2 个字节。

例如每个像素的范围是 0 到 4096,每个像素是 2 个字节。

我已经尝试将 PixelFormat.Format16bppGrayScale 与 Bitmap 构造函数一起使用,它会引发 GDI+ 异常。我读过的所有内容都说不支持这种格式,而且 MSDN 是错误的。

我想将此像素缓冲区转换为 .NET 位图格式(例如 Format32bppArgb),并尽可能减少图像质量损失。

有人知道吗?

最佳答案

请参阅下面的示例,该示例预先计算了一个查找表 (LUT) 并使用它来转换每个像素。这个版本涵盖了您的 12 位情况;对于 8 位,代码非常相似,但很难跨像素格式进行推广。

从 12 位 GS 转换为有效的 8 位 GS 将丢失数据。但是,您可以调整 LUT 表以关注具有更好对比度的更小范围的输入值(例如 DICOM Window Center/Window Width )。

class Program
{
static void Main( string[] args )
{
// Test driver - create a Wedge, convert to Bitmap, save to file
//
int width = 4095;
int height = 1200;
int bits = 12;

byte[] wedge = Wedge( width, height, bits );

Bitmap bmp = Convert( wedge, width, height, bits );

string file = "wedge.png";

bmp.Save( file );

Process.Start( file );
}

static Bitmap Convert( byte[] input, int width, int height, int bits )
{
// Convert byte buffer (2 bytes per pixel) to 32-bit ARGB bitmap

var bitmap = new Bitmap( width, height, PixelFormat.Format32bppArgb );

var rect = new Rectangle( 0, 0, width, height );

var lut = CreateLut( bits );

var bitmap_data = bitmap.LockBits( rect, ImageLockMode.WriteOnly, bitmap.PixelFormat );

ConvertCore( width, height, bits, input, bitmap_data, lut );

bitmap.UnlockBits( bitmap_data );

return bitmap;
}

static unsafe void ConvertCore( int width, int height, int bits, byte[] input, BitmapData output, uint[] lut )
{
// Copy pixels from input to output, applying LUT

ushort mask = (ushort)( ( 1 << bits ) - 1 );

int in_stride = output.Stride;
int out_stride = width * 2;

byte* out_data = (byte*)output.Scan0;

fixed ( byte* in_data = input )
{
for ( int y = 0; y < height; y++ )
{
uint* out_row = (uint*)( out_data + ( y * in_stride ) );

ushort* in_row = (ushort*)( in_data + ( y * out_stride ) );

for ( int x = 0; x < width; x++ )
{
ushort in_pixel = (ushort)( in_row[ x ] & mask );

out_row[ x ] = lut[ in_pixel ];
}
}
}
}

static uint[] CreateLut( int bits )
{
// Create a linear LUT to convert from grayscale to ARGB

int max_input = 1 << bits;

uint[] lut = new uint[ max_input ];

for ( int i = 0; i < max_input; i++ )
{
// map input value to 8-bit range
//
byte intensity = (byte)( ( i * 0xFF ) / max_input );

// create ARGB output value A=255, R=G=B=intensity
//
lut[ i ] = (uint)( 0xFF000000L | ( intensity * 0x00010101L ) );
}

return lut;
}

static byte[] Wedge( int width, int height, int bits )
{
// horizontal wedge

int max = 1 << bits;

byte[] pixels = new byte[ width * height * 2 ];

for ( int y = 0; y < height; y++ )
{
for ( int x = 0; x < width; x++ )
{
int pixel = x % max;

int addr = ( ( y * width ) + x ) * 2;

pixels[ addr + 1 ] = (byte)( ( pixel & 0xFF00 ) >> 8 );
pixels[ addr + 0 ] = (byte)( ( pixel & 0x00FF ) );
}
}

return pixels;
}
}

关于C# - 转换 8 位或 16 位灰度原始像素数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6832948/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com