C#图像:2.图像处理

类库项目ImageProcessingLibrary代码

里面是几种图像处理函数,可以生成DLL文件被其他(下面)项目添加引用

using System;
using System.Collections.Generic;
using System.Drawing;
using System.Drawing.Drawing2D;
using System.Drawing.Imaging;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;namespace ImageProcessingLibrary
{public static class ImageProcessing{/// 获取直方图数组,并绘制直方图/// <param name="image">需要处理的图像</param>/// <param name="indexColor">处理的颜色索引值,Blue:0,Green:1,Red:2</param>/// <param name="histogram">直方图统计数组</param>/// <returns>绘制好的直方图</returns>public static Bitmap GetHistogram(Bitmap image, int indexColor, out int[] histogram){histogram = new int[256];                               //直方图统计数组BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadOnly, PixelFormat.Format24bppRgb);  //将图像锁定到内存中byte[] datas = new byte[data.Stride * image.Height];   //图像数组Marshal.Copy(data.Scan0, datas, 0, datas.Length);     //将图像在内存中的数据复制到图像数组中for (int y = 0; y < image.Height * data.Stride; y += data.Stride) //data.Stride代表图像一行数据的字节总数/步长为data.Stride{//外层循环是遍历行for (int x = 0; x < image.Width * 3; x += 3)//遍历当前行中的每个像素/每个像素由三个字节(RGB)组成//每个颜色分量(红色、绿色或蓝色)可以有的不同强度级别就是2^8,即256个级别{int index = y + x;                     //颜色在内存中的索引/每个索引偏移量3字节(对应R,G,B)histogram[datas[index + indexColor]]++;//增加直方图中对应颜色分量出现的次数}}image.UnlockBits(data);byte maxValue = 0;                             //直方图中的最大值for (int value = 1; value < 256; value++){if (histogram[value] > histogram[maxValue]) maxValue = (byte)value;}Bitmap imageHistogram = new Bitmap(256, 256);Graphics GHistogram = Graphics.FromImage(imageHistogram);GHistogram.Clear(Color.Blue);for (int value = 1; value< 256; value++){int length = byte.MaxValue * histogram[value] / histogram[maxValue];GHistogram.DrawLine(new Pen(Color.FromArgb(value, value, value), 1f), value,256, value, 256 - length);                            //绘制直方图}Font font = new Font("宋体", 9f);//绘制统计标识for (int value = 32; value < 256; value += 32){int count = histogram[maxValue] / 8 * value / 32;Pen pen = new Pen(Color.Lime);pen.DashStyle = DashStyle.DashDot;SizeF sizeCount = GHistogram.MeasureString(count.ToString(), font);GHistogram.DrawLine(pen, 0, 255 - value, 255, 255 - value);//绘制数量等级线GHistogram.DrawString(count.ToString(), font, Brushes.Red, 5, 255 - value - sizeCount.Height / 2);SizeF sizeValue = GHistogram.MeasureString(value.ToString(), font);GHistogram.DrawLine(Pens.Red, value, 250, value, 255);//绘制颜色值等级线GHistogram.DrawString(value.ToString(), font, Brushes.Red, value - sizeValue.Width / 2, 240);}font.Dispose();return imageHistogram;}/// 将图像进行二值化处理/// <param name="image">需要处理的图像</param>/// <param name="indexColor">处理的颜色索引值,Blue:0,Green:1,Red:2</param>/// <param name="thresholdMin">阈值下限</param>/// <param name="thresholdMax">阈值上限</param>public static void BinaryImage(Bitmap image, int indexColor, int thresholdMin, int thresholdMax){//将图像锁定到内存中BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size), ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);byte[] datas = new byte[data.Stride * image.Height];           //图像数组Marshal.Copy(data.Scan0, datas, 0, datas.Length);              //将图像在内存中的数据复制到图像数组中for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int index = y + x;//根据阈值将图像分成黑色和白色,其中阈值内的为黑色,阈值外的为白色if (datas[index + indexColor] >= thresholdMin && datas[index + indexColor] <= thresholdMax)datas[index] = datas[index + 1] = datas[index + 2] = 0;elsedatas[index] = datas[index + 1] = datas[index + 2] = 255;}}Marshal.Copy(datas, 0, data.Scan0, datas.Length);      //将图像数组复制到内存中image.UnlockBits(data);                                //将图像从内存中解锁}/// 将彩色图像转换成灰度图像/// <param name="image">需要处理的图像</param>/// <returns></returns>public static void GreyImage(Bitmap image){BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);     //将图像锁定到内存中byte[] datas = new byte[data.Stride * image.Height];           //图像数组Marshal.Copy(data.Scan0, datas, 0, datas.Length);              //将图像在内存中的数据复制到图像数组中//计算灰度矩阵for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int index = y + x;                             //颜色在内存中的索引byte Blue = datas[index];                      //蓝色值byte Green = datas[index + 1];                 //绿色值byte Red = datas[index + 2];                   //红色值datas[index] = datas[index + 1] = datas[index + 2] =(byte)((Red * 19595 + Green * 38469 + Blue * 7472) >> 16);//灰度值}}Marshal.Copy(datas, 0, data.Scan0, datas.Length);      //将图像数组复制到内存中image.UnlockBits(data);}/// 对图像进行边缘提取/// <param name="image">灰度图像</param>/// <param name="threshold">边缘阈值</param>public static void ExtractEdge(Bitmap image){BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);      //将图像锁定到内存中byte[] datas = new byte[data.Stride * image.Height];           //图像数组byte[] edges = new byte[data.Stride * image.Height];           //图像边界Marshal.Copy(data.Scan0, datas, 0, datas.Length);              //将图像在内存中的数据复制到图像数组中for (int y = data.Stride; y < image.Height * data.Stride - data.Stride; y += data.Stride){for (int x = 3; x < image.Width * 3 - 3; x += 3){int index = y + x;byte grey = datas[index];                      //像素灰度值int value = 0;//将像素点的灰度值与周围8个点的灰度值分别求差的绝对值,再求和for (int yy = -data.Stride; yy <= data.Stride; yy += data.Stride){for (int xx = -3; xx <= 3; xx += 3){if (yy == 0 && xx == 0) continue;index = x + y + xx + yy;value += Math.Abs(grey - datas[index]);}}//上一步求和的结果除以8,然后赋值给边界数组edges[index] = edges[index + 1] = edges[index + 2] = (byte)(value >> 3);}}Marshal.Copy(edges, 0, data.Scan0, datas.Length);      //将边界数组复制到内存中image.UnlockBits(data);                                //将图像从内存中解锁}/// 将图像进制中值滤波处理/// <param name="image">需要处理的图像</param>public static void MedianFilter(Bitmap image){BitmapData dataImage = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);//将图像锁定到内存中unsafe{byte* pImage = (byte*)dataImage.Scan0.ToPointer(); //获取图像头指针for (int y = 1; y < image.Height - 1; y++){for (int x = 1; x < image.Width - 1; x++){List<byte>[] datas = new List<byte>[3];        //创建存放八邻域像素颜色值的列表for (int k = 0; k < 3; k++) datas[k] = new List<byte>();for (int yy = -1; yy < 2; yy++){for (int xx = -1; xx < 2; xx++){int index = (y + yy) * dataImage.Stride + (x + xx) * 3;  //八邻域像素索引//将八邻域像素颜色值添加到列表中for (int k = 0; k < 3; k++) datas[k].Add(pImage[index + k]);}}for (int k = 0; k < 3; k++) datas[k].Sort();            //对八邻域颜色值排序int indexMedian = y * dataImage.Stride + x * 3;for (int k = 0; k < 3; k++){pImage[indexMedian + k] = datas[k][4]; //取排序后的中间值作为像素颜色值datas[k].Clear();}datas = null;}}}image.UnlockBits(dataImage);}/// <summary>/// 利用自动对比度增强方法,对图像进行对比度增强处理/// </summary>/// <param name="image">需要处理的灰度图像</param>/// <param name="sourceMin">定义原始图像中像素值的最小范围。</param>/// <param name="sourceMax">定义原始图像中像素值的最大范围。</param>/// <param name="destMin">定义增强后图像中像素值想要映射到的最小范围</param>/// <param name="destMax">定义增强后图像中像素值想要映射到的最大范围</param>public static void EnhanceImage(Bitmap image, int sourceMin, int sourceMax, int destMin, int destMax){//将图像锁定到内存中BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size), ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);byte[] datas = new byte[data.Stride * image.Height];   //图像数组Marshal.Copy(data.Scan0, datas, 0, datas.Length);      //将图像在内存中的数据复制到图像数组中for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int index = y + x;                             //颜色在内存中的索引float value = (float)(destMax - destMin) / (float)(sourceMax - sourceMin)* (float)(datas[index] - sourceMin) + destMin;     //灰度值映射关系datas[index] = datas[index + 1] = datas[index + 2]= value < byte.MinValue ? byte.MinValue : (value > byte.MaxValue ? byte.MaxValue : (byte)value);}}Marshal.Copy(datas, 0, data.Scan0, datas.Length);      //将图像数组复制到内存中image.UnlockBits(data);}/// 连通区域标记/// <param name="image">二值图像</param>public static void ConnectRegion(Bitmap image){BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size), ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);  //将图像锁定到内存中byte[] datas = new byte[data.Stride * image.Height];       //图像数组Marshal.Copy(data.Scan0, datas, 0, datas.Length);          //将图像在内存中的数据复制到图像数组中int sign = 40;for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int index = y + x;                                  //颜色在内存中的索引if (datas[index] == 0){SignRegionIterative(datas, x, y, sign, data.Stride, image.Width, image.Height);sign += 40;}}}Marshal.Copy(datas, 0, data.Scan0, datas.Length);      //将图像数组复制到内存中image.UnlockBits(data);}/// <summary>/// 区域标记(可被迭代版本SignRegionIterative替换)/// </summary>/// <param name="datas"></param>/// <param name="x">初始点(x, y)</param>/// <param name="y"></param>/// <param name="sign"></param>/// <param name="stride"></param>/// <param name="width"></param>/// <param name="height"></param>static void SignRegion(byte[] datas, int x, int y, int sign, int stride, int width, int height){datas[y + x] = datas[y + x + 1] = datas[y + x + 2] = (byte)sign;       //标记区域if (x > 0 && datas[y + x - 3] == 0){SignRegion(datas, x - 3, y, sign, stride, width, height);  //搜索左边像素}if (x < width * 3 - 3 && datas[y + x + 3] == 0){SignRegion(datas, x + 3, y, sign, stride, width, height);  //搜索右边像素}if (y > 0 && datas[y - stride + x] == 0){SignRegion(datas, x, y - stride, sign, stride, width, height); //搜索上边像素}if (y < stride * height && datas[y + stride + x] == 0){SignRegion(datas, x, y + stride, sign, stride, width, height); //搜索下边像素}}static void SignRegionIterative(byte[] datas, int x, int y, int sign, int stride, int width, int height){Stack<Tuple<int, int>> stack = new Stack<Tuple<int, int>>();//使用了一个栈来保存待处理的像素坐标stack.Push(Tuple.Create(x, y));//将初始坐标(x, y)压入栈中while (stack.Count > 0)//循环会一直执行,直到栈为空{var current = stack.Pop();//每次循环中,我们从栈顶弹出一个坐标,并将其标记int currentX = current.Item1;int currentY = current.Item2;// 标记当前点  if (currentX >= 0 && currentX < width * 3 && currentY >= 0 && currentY < height * stride){int index = currentY + currentX;datas[index] = datas[index + 1] = datas[index + 2] = (byte)sign;// 搜索相邻的可标记像素并压入栈  if (currentX > 0 && datas[currentY + currentX - 3] == 0){stack.Push(Tuple.Create(currentX - 3, currentY)); // 左边像素  }if (currentX < width * 3 - 3 && datas[currentY + currentX + 3] == 0){stack.Push(Tuple.Create(currentX + 3, currentY)); // 右边像素  }if (currentY > 0 && datas[currentY - stride + currentX] == 0){stack.Push(Tuple.Create(currentX, currentY - stride)); // 上边像素  }if (currentY < stride * (height - 1) && datas[currentY + stride + currentX] == 0){stack.Push(Tuple.Create(currentX, currentY + stride)); // 下边像素  }}}}/*在这个迭代版本中,我们使用了一个栈来保存待处理的像素坐标。开始时,我们将初始坐标(x, y)压入栈中。然后,* 我们进入一个循环,该循环会一直执行,直到栈为空。在每次循环中,我们从栈顶弹出一个坐标,并将其标记。然后,我们检查该坐标的左、右、上、下四个相邻像素,如果它们的颜色值为0(即未被标记),我们就将它们压入栈中以便后续处理。请注意,我们在压入相邻像素之前进行了边界检查,以确保我们不会尝试访问数组之外的元素。这个迭代版本应该能够避免递归版本可能导致的栈溢出问题。*//// <summary>/// 缩放函数/// </summary>/// <param name="image"></param>/// <param name="horizon"></param>/// <param name="verticale"></param>public static void Dilation(Bitmap image, float horizon, float verticale){BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);//将图像锁定到内存中byte[] datasOld = new byte[data.Stride * image.Height];        //图像数组byte[] datasNew = new byte[data.Stride * image.Height];        //图像数组//将图像在内存中的数据复制到图像数组中Marshal.Copy(data.Scan0, datasOld, 0, datasOld.Length);for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int xNew = x / 3, yNew = y / data.Stride;int xOld = (int)(xNew / horizon);int yOld = (int)(yNew / verticale);if (xOld < 0 || xOld >= image.Width || yOld < 0 || yOld >= image.Height) continue;int indexOld = yOld * data.Stride + xOld * 3;int indexNew = y + x;datasNew[indexNew] = datasNew[indexNew + 1] = datasNew[indexNew + 2] = datasOld[indexOld];}}Marshal.Copy(datasNew, 0, data.Scan0, datasOld.Length);  //将图像数组复制到内存中image.UnlockBits(data);                                  //将图像从内存中解锁}/// <summary>/// 平移函数/// </summary>/// <param name="image"></param>/// <param name="horizon"></param>/// <param name="verticale"></param>public static void Translation(Bitmap image, float horizon, float verticale){BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb);//将图像锁定到内存中byte[] datasOld = new byte[data.Stride * image.Height];        //图像数组byte[] datasNew = new byte[data.Stride * image.Height];        //图像数组//将图像在内存中的数据复制到图像数组中Marshal.Copy(data.Scan0, datasOld, 0, datasOld.Length);for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int xNew = x / 3, yNew = y / data.Stride;int xOld = (int)(xNew - horizon);int yOld = (int)(yNew - verticale);if (xOld < 0 || xOld >= image.Width || yOld < 0 || yOld >= image.Height) continue;int indexOld = yOld * data.Stride + xOld * 3;int indexNew = y + x;datasNew[indexNew] = datasNew[indexNew + 1] = datasNew[indexNew + 2] = datasOld[indexOld];}}Marshal.Copy(datasNew, 0, data.Scan0, datasOld.Length);  //将图像数组复制到内存中image.UnlockBits(data);                                  //将图像从内存中解锁}/// <summary>/// 旋转函数/// </summary>/// <param name="image"></param>/// <param name="angle"></param>/// <param name="center"></param>public static void Rotation(Bitmap image, float angle, Point center){BitmapData data = image.LockBits(new Rectangle(new Point(), image.Size),ImageLockMode.ReadWrite, PixelFormat.Format24bppRgb); //将图像锁定到内存中byte[] datasOld = new byte[data.Stride * image.Height];        //图像数组byte[] datasNew = new byte[data.Stride * image.Height];        //图像数组//将图像在内存中的数据复制到图像数组中Marshal.Copy(data.Scan0, datasOld, 0, datasOld.Length);for (int y = 0; y < image.Height * data.Stride; y += data.Stride){for (int x = 0; x < image.Width * 3; x += 3){int xNew = x / 3, yNew = y / data.Stride;int xOld = (int)((xNew - center.X) * Math.Cos(angle * Math.PI / 180)- (yNew - center.Y) * Math.Sin(angle * Math.PI / 180) + center.X);int yOld = (int)((xNew - center.X) * Math.Sin(angle * Math.PI / 180)+ (yNew - center.Y) * Math.Cos(angle * Math.PI / 180) + center.Y);if (xOld < 0 || xOld >= image.Width|| yOld < 0 || yOld >= image.Height) continue;int indexOld = yOld * data.Stride + xOld * 3;int indexNew = y + x;datasNew[indexNew] = datasNew[indexNew + 1] = datasNew[indexNew + 2] = datasOld[indexOld];}}Marshal.Copy(datasNew, 0, data.Scan0, datasOld.Length); //将图像数组复制到内存中image.UnlockBits(data);                                 //将图像从内存中解锁}}}
/*假设颜色分量是8位的,那么每个颜色分量(红色、绿色或蓝色)可以有的不同强度级别就是2^8,即256个级别。* 这是因为8位可以表示从0到255的整数,总共256个不同的数值。在数字图像处理中,8位颜色深度是常见的,* 因为它提供了足够的动态范围来表示大多数自然和人工颜色的细微差别,同时保持数据量相对较小。当你说“直方图大小为256”时,你指的是直方图的横坐标(即颜色强度的可能值)有256个不同的条目,
每个条目对应一个特定的颜色强度值(从0到255)。直方图的纵坐标通常表示该颜色强度值在图像中出现的频率或像素数量。因此,如果我们想为8位颜色分量的图像构建直方图,我们将创建一个大小为256的数组,数组的每个元素初始化为0。
然后,我们遍历图像的每个像素,对于每个像素的特定颜色分量(如红色、绿色或蓝色),我们增加直方图中对应颜色强度值的计数。
这个过程最终会给我们一个表示图像中每个颜色强度出现频率的直方图。*//*datas[index] = datas[index + 1] = datas[index + 2] = some_value;
这里实际上是在进行连续赋值。some_value首先被赋给datas[index + 2](蓝色通道),然后这个值又被赋给datas[index + 1]
(绿色通道),最后再次被赋给datas[index](红色通道)。由于赋值操作符=是从右到左结合的,
所以这行代码的效果是三个颜色通道都被赋予了相同的值。在图像处理的上下文中,这种操作通常用于灰度化(将彩色图像转换为灰度图像,即每个通道具有相同的亮度值),
或者在进行某些类型的颜色校正或调整时,需要对所有颜色通道进行统一的变换。*//*datas[index] = datas[index + 1] = datas[index + 2]  = value < byte.MinValue ? byte.MinValue : (value > byte.MaxValue ? byte.MaxValue : (byte)value);
value < byte.MinValue ? byte.MinValue : ... :这部分是一个条件运算符(也称作三元运算符)。
它检查value是否小于byte.MinValue(即0)。如果是,则直接返回byte.MinValue(0),因为负值在字节表示中是无效的。... : (value > byte.MaxValue ? byte.MaxValue : ...) :如果value不小于byte.MinValue,
代码会进入这个分支,并检查value是否大于byte.MaxValue(即255)。如果是,则返回byte.MaxValue(255),
因为超出255的值也无法用一个字节表示。... : (byte)value :如果value既不小于byte.MinValue也不大于byte.MaxValue,则将其强制转换为byte类型,
这会自动截去小数部分,只保留整数部分。*//*float value = (float)(destMax - destMin) / (float)(sourceMax - sourceMin)  * (float)(datas[index] - sourceMin) + destMin;
(float)(destMax - destMin):计算目标值域的范围,即最大值与最小值之间的差。(float)(sourceMax - sourceMin):计算原始值域的范围,即原图像中像素值的最大值与最小值之间的差。(float)(destMax - destMin) / (float)(sourceMax - sourceMin):计算目标值域范围与原始值域范围的比例。这个比例将用于调整原始像素值,使其适应新的值域。(float)(datas[index] - sourceMin):计算当前像素值相对于原始值域最小值的偏移量。* (float)(datas[index] - sourceMin):将上述比例与原始像素值的偏移量相乘,得到在新值域中的相对偏移量。+ destMin:将新值域中的相对偏移量加上新值域的最小值,从而得到映射后的绝对像素值。float value = ...:将最终计算出的映射后的像素值存储在value变量中*//*。像素灰度的变换公式如下:
GreyNew=(destMax - destMin) / (sourceMax - sourceMin) * (GreyOld - sourceMin) + destMin;
其中,GreyNew为增强后的灰度值,GreyOld为增强前的灰度值,sourceMax和sourceMin表示增强前的灰度范围,
destMax和destMin表示增强后的灰度范围。从实例效果图的直方图可以看出,图像在增强前的灰度分布在0到100之间,
所以图像整体颜色比较灰暗,增强后的灰度范围是整个灰度范围,所以增强后的图像更有层次感,也更清晰。*/

 下面是各个独立窗体应用程序,都需要添加引用上面的类库(DLL),复制测试这些代码需要自己在......项目名\bin\Debug路径下添加命名自己的图片(可参考C#&图像第一章

1.基于差分的边缘检测与提取
using ImageProcessingLibrary;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;namespace EdgeExtraction
{public partial class FormEdgeExtraction : Form{public FormEdgeExtraction(){InitializeComponent();}protected override void OnPaint(PaintEventArgs e)//重写OnPaint事件函数{Graphics G = e.Graphics;Bitmap image = new Bitmap("apple.jpg");                    //加载图像ImageProcessing.GreyImage(image);                          //生成灰度图像Rectangle rectImage = new Rectangle(new Point(), image.Size);G.DrawImage(image, rectImage);rectImage.Offset(rectImage.Width, 0);ImageProcessing.ExtractEdge(image);                        //提取边缘G.DrawImage(image, rectImage);rectImage.Offset(-rectImage.Width, rectImage.Height);Bitmap image2 = image.Clone() as Bitmap;ImageProcessing.BinaryImage(image2, 0, 20, 255);           //在灰度20到255的范围提取边界G.DrawImage(image2, rectImage);image2.Dispose();rectImage.Offset(rectImage.Width, 0);ImageProcessing.BinaryImage(image, 0, 40, 255);            //在灰度40到255的范围提取边界G.DrawImage(image, rectImage);image.Dispose();}}
}
2.利用中值滤波使图像变得平滑
using ImageProcessingLibrary;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;namespace MedianFilte
{public partial class FormMedianFilter : Form{public FormMedianFilter(){InitializeComponent();}protected override void OnPaint(PaintEventArgs e)//重写OnPaint事件函数{Graphics G = e.Graphics;Bitmap image = new Bitmap("apple.jpg");                    //加载图像Random random = new Random();Rectangle rectImage = new Rectangle(new Point(), image.Size);for (int i = 0; i < 500; i++)//随机产生噪点{ image.SetPixel(random.Next(image.Width), random.Next(image.Height), Color.FromArgb(random.Next(int.MaxValue)));}G.DrawImage(image, rectImage);                             //绘制原始图像ImageProcessing.MedianFilter(image);//中值滤波处理rectImage.Offset(0, rectImage.Height);G.DrawImage(image, rectImage);                             //绘制中值滤波后的图像image.Dispose();}}
}
3.增强图像对比度
using ImageProcessingLibrary;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;namespace ImageEnhancement
{public partial class FormImageEnhancement : Form{public FormImageEnhancement(){InitializeComponent();}protected override void OnPaint(PaintEventArgs e)//重写OnPaint事件函数{Graphics G = e.Graphics;Bitmap image = new Bitmap("jdq.gif");                     //加载图像Size newSize = new Size(image.Width / 2, image.Height / 2);//图像尺寸缩半Rectangle rectImage = new Rectangle(new Point(), newSize);Rectangle rectHistogram = new Rectangle(rectImage.Width, rectImage.Top, 256, 256);int[] histogram;Bitmap imageHistogram = ImageProcessing.GetHistogram(image, 0, out histogram);G.DrawImage(image, rectImage);                             //绘制原始图像G.DrawImage(imageHistogram, rectHistogram);                //绘制原始直方图ImageProcessing.EnhanceImage(image, 0, 255, 0, 200);imageHistogram = ImageProcessing.GetHistogram(image, 0, out histogram);rectImage.Offset(0, Math.Max(rectImage.Height, 256));//位置最少偏移图片高度rectHistogram.Offset(0, Math.Max(rectImage.Height, 256));G.DrawImage(image, rectImage);                             //绘制增强图像G.DrawImage(imageHistogram, rectHistogram);                //绘制增强直方图image.Dispose();imageHistogram.Dispose();}}
}
4.目标提取与区域标记 
using ImageProcessingLibrary;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;namespace ConnectRgeion
{public partial class FormConnectRgeion : Form{public FormConnectRgeion(){InitializeComponent();}protected override void OnPaint(PaintEventArgs e)//重写OnPaint事件函数{Graphics G = e.Graphics;Bitmap image = new Bitmap("657.png");                    //加载图像Rectangle rectImage = new Rectangle(new Point(), image.Size);G.DrawImage(image, rectImage); //原始图ImageProcessing.BinaryImage(image, 0, 0, 210);                     //通过二值化将目标分割出来rectImage.Offset(  rectImage.Width,0);G.DrawImage(image, rectImage);ImageProcessing.MedianFilter(image);//中值滤波处理;rectImage.Offset(-rectImage.Width, image.Height);G.DrawImage(image, rectImage);ImageProcessing.ConnectRegion(image);                     //连通区域标记rectImage.Offset(rectImage.Width, 0);G.DrawImage(image, rectImage);image.Dispose();}}
}
5.图像的变形 
using ImageProcessingLibrary;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;namespace Transformation
{public partial class FormTransformation : Form{public FormTransformation(){InitializeComponent();}protected override void OnPaint(PaintEventArgs e)//重写OnPaint事件函数{Graphics G = e.Graphics;Bitmap image = new Bitmap("lina.jpg");                 //加载图像Size newSize = new Size(image.Width / 3, image.Height / 3);//图像尺寸缩半Rectangle rectImage = new Rectangle(new Point(), newSize);G.DrawImage(image, rectImage);                                     //绘制原始图像rectImage.Offset(rectImage.Width, 0);ImageProcessing.Dilation(image, 0.8f, 0.8f);                       //对图像进行缩小处理G.DrawImage(image, rectImage);rectImage.Offset(-rectImage.Width, rectImage.Height);ImageProcessing.Translation(image, image.Width / 10, image.Height / 10); //将图像平移居中-居中还和和图像缩小有关(0.5f-4/0.8f-10)G.DrawImage(image, rectImage);rectImage.Offset(rectImage.Width, 0);//将图像围绕中心点逆时针旋转45°ImageProcessing.Rotation(image, 45, new Point(image.Width / 2, image.Height / 2));G.DrawImage(image, rectImage);image.Dispose();}/// <summary>/// 根据加载的图片大小动态调整Windows Forms窗口的大小/// </summary>/// <param name="imagePath"></param>private void LoadImageAndResizeForm(){try{Bitmap image = new Bitmap("lina.jpg");// 调整窗口大小以适应图片,并考虑额外的空间  this.Size = new Size(image.Width*1+ SystemInformation.Border3DSize.Width * 10,image.Height*1 + SystemInformation.CaptionHeight + SystemInformation.Border3DSize.Height * 10);}catch (Exception ex){MessageBox.Show("加载图片时出错: " + ex.Message);}}private void MainForm_Load(object sender, EventArgs e){// 使用绝对路径//string imagePath = @"D:\Users\图像处理\WindowsFormsCeShi\bin\Debug\lina.jpg";// LoadImageAndResizeForm(imagePath);LoadImageAndResizeForm();//  MessageBox.Show("窗体正在初始化");}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/6560.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

IoTDB 入门教程——导读

文章目录 一、基础篇二、实战篇三、企业篇四、问题篇 Apache IoTDB&#xff08;物联网数据库&#xff09;是一体化收集、存储、管理与分析物联网时序数据的软件系统。 Apache IoTDB 采用轻量式架构&#xff0c;具有高性能和丰富的功能&#xff0c;并与Apache Hadoop、Spark和Fl…

n-Track Studio Suite for Mac激活版:打造您的专属音频工作室

n-Track Studio Suite for Mac是一款功能强大的数字音频工作站软件&#xff0c;让您在家中就能享受到专业录音棚的待遇。无论是录制人声、乐器还是MIDI序列&#xff0c;都能轻松应对。 n-Track Studio Suite for Mac激活版下载 这款软件拥有实时音高校准、时间拉伸和自动补足功…

深度学习500问——Chapter08:目标检测(7)

文章目录 8.3.8 RFBNet 8.3.9 M2Det 8.3.8 RFBNet RFBNet有哪些创新点 1. 提出RF block&#xff08;RFB&#xff09;模块 RFBNet主要想利用一些技巧使得轻量级模型在速度和精度上达到很好的trade-off的检测器。灵感来自人类视觉的感受野结构Receptive Fields&#xff08;RFs…

为什么要备份数据?我经历的数据灾难

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github&#xff1a;codetoys&#xff0c;所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的&#xff0c;可以在任何平台上使用。 &#xff08;注&#xff1a;这…

快速构建vscode pytest 开发测试环境

如果不想用 heavy 的pycharm vscode 也是1个很好的选择 安装python SDK pacman -S python [gatemanmanjaro-x13 tmp]$ pacman -Q python python 3.11.8-1安装Vscode 很多中方法 yay -S visual-studio-code-bin [gatemanmanjaro-x13 tmp]$ pacman -Q | grep -i visual visua…

JVM-02

字节码文件是一种特殊的文件格式&#xff0c;它包含了将源代码转换为机器可执行代码所需的指令集。字节码文件通常是由编译器将源代码编译为字节码的中间表示形式。 在Java中&#xff0c;字节码文件的扩展名为.class&#xff0c;它存储了编译后的Java代码。这些字节码文件可以在…

[Java EE] 多线程(七): 锁策略

&#x1f338;个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 &#x1f3f5;️热门专栏:&#x1f355; Collection与数据结构 (90平均质量分)https://blog.csdn.net/2301_80050796/category_12621348.html?spm1001.2014.3001.5482 &#x1f9c0;Java …

基于MQTT通信开发的失物招领小程序

项目架构设计 这个项目采用前后端分离的方式&#xff0c;重新设计了两条链路来支撑程序的信息获取和传递 前端的小程序页面再启动页面渲染时&#xff0c;直接通过DBAPI从后端数据库获取信息&#xff0c;直接渲染在小程序中项目中给DBAPI的定位是快速从后端获取信息&#xff0…

如何在 Ubuntu 服务器上使用 GlusterFS 创建冗余存储池

简介 冗余和高可用性对于各种服务器活动都是必要的。在数据存储方面存在单点故障对于任何关键数据来说都是非常危险的配置。 虽然许多数据库和其他软件允许您在单个应用程序的上下文中分散数据&#xff0c;但其他系统可以在文件系统级别操作&#xff0c;以确保数据在写入磁盘时…

SQL:NOT IN与NOT EXISTS不等价

在对SQL语句进行性能优化时&#xff0c;经常用到一个技巧是将IN改写成EXISTS&#xff0c;这是等价改写&#xff0c;并没有什么问题。问题在于&#xff0c;将NOT IN改写成NOT EXISTS时&#xff0c;结果未必一样。 目录 一、举例验证二、三值逻辑简述三、附录&#xff1a;用到的S…

微信小程序:14.什么是wxs,wxs的使用

wxs是小程序独有的一套脚本语言&#xff0c;结合wxml&#xff0c;可以构建出页面的结构 wxs的应用场景 wxml中无法调用在页面的js中定义的函数&#xff0c;但是wxml可以调用wxs中定义的函数。因此小程序中wxs的典型应用场景就是过滤器 wxs和js的关系 wxs有自己的数据类型 …

【Flask 系统教程 5】视图进阶

类视图 在 Flask 中&#xff0c;除了使用函数视图外&#xff0c;你还可以使用类视图来处理请求。类视图提供了一种更为结构化和面向对象的方式来编写视图函数&#xff0c;使得代码组织更清晰&#xff0c;并且提供了更多的灵活性和可扩展性。 创建类视图 要创建一个类视图&am…

SVM直观理解

https://tangshusen.me/2018/10/27/SVM/ https://www.bilibili.com/video/BV16T4y1y7qj/?spm_id_from333.337.search-card.all.click&vd_source8272bd48fee17396a4a1746c256ab0ae SVM是什么? 先来看看维基百科上对SVM的定义: 支持向量机&#xff08;英语&#xff1a;su…

使用Python实现二维码生成工具

二维码的本质是什么&#xff1f; 二维码本质上&#xff0c;就是一段字符串。 我们可以把任意的字符串&#xff0c;制作成一个二维码图片。 生活中使用的二维码&#xff0c;更多的是一个 URL 网址。 需要用到的模块 先看一下Python标准库&#xff0c;貌似没有实现这个功能的…

一文了解python机器学习Sklearn

1.3 安装和配置Sklearn 要使用Sklearn库&#xff0c;首先需要安装Python和相应的库。在本教程中&#xff0c;我们将使用Python 3.x版本。可以使用以下命令安装Sklearn库&#xff1a; pip install scikit-learn安装完成后&#xff0c;可以在Python代码中导入Sklearn库&#xf…

code-server容器webpack的ws无法连接解决方法

TLDR 通过指定client的wsrul去连接ws devServer.client.webSocketURL ‘wss://<Forwarded uri>/ws’ 拓扑 1、code-server: 用于编写代码、启动webpack dev-server 服务&#xff1b;[https://<domain>:8001] 2、webpack: 用于浏览dev-server服务&#xff1b;[ht…

在视频中使用时间卷积和半监督训练进行三维人体姿态估计

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 摘要Abstract文献阅读&#xff1a;在视频中使用时间卷积和半监督训练进行三维人体姿态估计1、文献摘要2、提出方法2.1、时间扩张卷积模型2.2、半监督方法2.3、与传统…

UE4 Widget制作搜索框

效果&#xff1a; 一、控件层级结构 1.父控件层级结构 2.子控件层级结构 二、蓝图 1.先清除掉创建子项&#xff08;注意&#xff1a;这里使用的是reverse循环&#xff01;&#xff09; 2.判断是否含有关键字&#xff0c;创建子控件

【Android学习】日期和时间选择对话框

实现功能 实现日期和时间选择的对话框&#xff0c;具体效果可看下图(以日期为例) 具体代码 1 日期对话框 1.1 xml <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android&quo…

AGI要闻:斯坦福李飞飞首次创业,瞄准“空间智能”;OpenAI下周发布搜索产品挑战谷歌;新的开源 AI 眼镜来了|钛媒体AGI | 最新快讯

多方消息证实&#xff0c;OpenAI将会在北京时间5月10日&#xff08;周五&#xff09;凌晨2点公布搜索引擎新产品消息。 斯坦福大学首位红杉讲席教授 李飞飞 通用人工智能&#xff08;AGI&#xff09;领域又公布了一系列重磅消息。 5月4日凌晨&#xff0c;据路透社&#xff0c…