MSDN的代码

  1. COLORREF pixel;
  2. int maxY = imgOriginal.GetHeight(), maxX = imgOriginal.GetWidth();
  3. byte r,g,b,avg;
  4. for (int y=0; y<maxY; y++) {
  5. for (int x=0; x<maxX; x++) {
  6. pixel = imgOriginal.GetPixel(x,y);
  7. r = GetRValue(pixel);
  8. g = GetGValue(pixel);
  9. b = GetBValue(pixel);
  10. avg = (r+ g+ b)/3;
  11. imgOriginal.SetPixelRGB(x,y,avg,avg,avg);
  12. }}

这种方式效率很低, 因为每次调用getpixel,都包含着程序的进栈和出栈。所以,面对大量需要处理的数据,采用直接访问内存地址的方法。

  1. byte* pRealData;
  2. pRealData=(byte*)imgOriginal.GetBits();
  3. int pit=imgOriginal.GetPitch();
  4. int bitCount=imgOriginal.GetBPP()/8;
  5. for (int y=0; y<maxY; y++) {
  6. for (int x=0; x<maxX; x++) {
  7. int grayVal=(int)(int)(*(pRealData + pit*y + x*bitCount))*0.3
  8. + (int)(int)(*(pRealData + pit*y + x*bitCount +1))*0.59
  9. + (int)(int)(*(pRealData + pit*y + x*bitCount +2))*0.11;
  10. *(pRealData + pit*y + x*bitCount)=grayVal;
  11. *(pRealData + pit*y + x*bitCount +1)=grayVal;
  12. *(pRealData + pit*y + x*bitCount +2)=grayVal;
  13. //如果是8位灰度图像,直接读取一个BYTE位为灰度值
  14. //如果是24位RGB图像,则依次读取pixAddr,pixAddr+1,pixAddr+2为B、G、R分量值
  15. }}

用两种方法对同一张图片(3264*2448像素)进行处理,前者需要1分钟,后者只需1秒左右。

所以,后者比前者至少快60倍

直接访问内存地址的另一种方式:

        int i,j,temp;
int pixel[4];
int width = yuantu.GetWidth();
int height = yuantu.GetHeight();
int widthBytes = yuantu.GetPitch();
bianyuantu.Create(width,height,yuantu.GetBPP()); if(yuantu.IsIndexed())
{
yuantu.GetColorTable(0,256,colorTable);
bianyuantu.SetColorTable(0,256,colorTable);
}
BYTE *pYuantuData = (BYTE*)yuantu.GetBits();
BYTE *pBianyuantuData =(BYTE*)bianyuantu.GetBits(); for(j=0;j<height-1;j++)
{
for(i=0;i<width-1;i++)
{
pixel[0]=pYuantuData[j*widthBytes+i];
pixel[1]=pYuantuData[j*widthBytes+i+1];
pixel[2]=pYuantuData[(j+1)*widthBytes+i];
pixel[3]=pYuantuData[(j+1)*widthBytes+i+1];
temp=(int)sqrt((double)((pixel[0]-pixel[3])*(pixel[0]-pixel[3])
+(pixel[1]-pixel[2])*(pixel[1]-pixel[2]))); //罗伯特算子
pBianyuantuData[j*widthBytes+i]=temp;
}
}

  

彩色图像转化为灰度图的处理方式

//真彩色图像变为灰度图,直接修改像素点的值

  1. void PixelsChangedToGray(CImage *pImage)
  2. {
  3. int  nByte,j,i,nWidth,nHeight,nBytesPerPixel;
  4. BYTE  *pPixelLine,cNewPixelValue;
  5. nWidth=pImage->GetWidth();   nHeight=pImage->GetHeight();
  6. nBytesPerPixel= pImage->GetBPP()/8;
  7. for (i=0;i<nHeight;i++){
  8. pPixelLine =(BYTE*) pImage->GetPixelAddress(0,i);
  9. nByte=0;
  10. for (j=0;j<nWidth;j++){      cNewPixelValue=(BYTE)(0.11*pPixelLine[nByte]
  11. +0.59*pPixelLine[nByte+1]
  12. +0.30*pPixelLine[nByte+2]);
  13. pPixelLine[nByte] = pPixelLine[nByte+1] = pPixelLine[nByte+2]
  14. = cNewPixelValue;
  15. nByte+=nBytesPerPixel;
  16. }
  17. }
  18. }

//非真彩色图像变为灰度图,修改调色板信息

  1. void PaletteChangedToGray(CImage *pImage)
  2. {
  3. RGBQUAD  ColorTabs[256];
  4. int  i,nColorTableEntries,nNewGrayColor;
  5. nColorTableEntries=pImage->GetMaxColorTableEntries();
  6. pImage->GetColorTable(0,nColorTableEntries,ColorTabs);
  7. for (i=0;i<nColorTableEntries;i++){
  8. nNewGrayColor=(int)(0.11*ColorTabs[i].rgbBlue
  9. + 0.59*ColorTabs[i].rgbGreen
  10. + 0.30*ColorTabs[i].rgbRed);
  11. ColorTabs[i].rgbBlue = (BYTE)nNewGrayColor;
  12. ColorTabs[i].rgbGreen = (BYTE)nNewGrayColor;
  13. ColorTabs[i].rgbRed = (BYTE)nNewGrayColor;
  14. }
  15. pImage->SetColorTable(0,nColorTableEntries,ColorTabs);
  16. }

CImage访问像素及其像素操作总结的更多相关文章

  1. OpenCV(2)-Mat数据结构及访问Mat中像素

    Mat数据结构 一开始OpenCV是基于C语言的,在比较早的教材例如<学习OpenCV>中,讲解的存储图像的数据结构还是IplImage,这样需要手动管理内存.现在存储图像的基本数据结构是 ...

  2. SNMP 原理及配置简述 net-snmp-utils net-snmp 第2版基于SNMP 群体名(community name) 第3版引入了安全性更高的访问控制方法 SNMP协议操作只有4种 Apache的php_snmp 模块

    SNMP 原理及配置简述  net-snmp-utils  net-snmp 第2版基于SNMP 群体名(community name) 第3版引入了安全性更高的访问控制方法 SNMP协议操作只有4种 ...

  3. C# 委托 / 跨线程访问UI / 线程间操作无效: 从不是创建控件“Form1”的线程访问它

    C# 委托 / 跨线程访问UI /  线程间操作无效: 从不是创建控件“Form1”的线程访问它 网上的代码都比较复杂,还是这个简单 见代码, 简易解决办法: 主窗体代码 using System; ...

  4. ADO.NET访问Access(文本数据库)数据操作(CRUD)

    1,ADO.NET访问Access(文本数据库)数据操作(CRUD) 2,DatabaseDesign 文本数据库Northwind.mdb 3,/App_Code 3.1,/App_Code/DBC ...

  5. GG_DataAccess 数据库访问层使用dapper操作

    3.5.GG_DataAccess 数据库访问层使用dapper操作 和Model实体类同理,tt模板已写好,需要的可加qq群:547765059  自己下载.

  6. 《OpenCV3编程入门》访问图像中像素的三类方法

    ·方法一 指针访问:C操作符[ ]; ·方法二 迭代器iterator; ·方法三 动态地址计算; #include <opencv2/core/core.hpp> #include &l ...

  7. 通过一个小Trick实现shader的像素识别/统计操作

    2018/12/14日补充:后来发现compute shader里用AppendStructuredBuffer可以解决这类问题,请看这里:https://www.cnblogs.com/hont/p ...

  8. opencv 中对一个像素的rgb值或像素值进行操作的几个常用小办法【转】

    You can access the Image pixels in many ways:1. One using the Inbuilt macro2. One using the pointer ...

  9. OpenCV 学习笔记(11)像素级别指针操作

    //优化两图的连接处,使得拼接自然 void OptimizeSeam(Mat& img1, Mat& trans, Mat& dst) { int start = MIN(c ...

随机推荐

  1. webrtc 学习资源 http://www.cnblogs.com/lingyunhu/p/3578218.html

    Realtime/Working WebRTC Experiments It is a repository of uniquely experimented WebRTC demos; writte ...

  2. css中块元素和行内元素区别

    行内元素特点 1.和其他元素都在一行上: 2.元素的高度.宽度.行高及顶部和底部边距不可设置: 3.元素的宽度就是它包含的文字或图片的宽度,不可改变. 块元素特点 1.每个块级元素都从新的一行开始,并 ...

  3. touch实现滑动删除

    请用chrome手机模式查看或者在手机上查看(转载请注明出处) <!DOCTYPE html> <html> <head> <meta charset=&qu ...

  4. Rsync+Sersync同步

    Rsync+Sersync同步特点: (1):sersync可以记录下被监听目录中发生变化的(包括增加.删除.修改)具体某一个文件或某一个目录的名字:(2):rsync在同步的时候,只同步发生变化的这 ...

  5. docker 学习(四) springboot + docker

    下面演示: 在Windows上新建一个简单的Springboot工程,生成docker iamge,然后在本地的docker上运行: (1):登录到 https://start.spring.io/, ...

  6. win10文件夹或文件已在另一程序中打开

    我们在对文件或文件夹进行删除.移动.重命名等操作时,系统可能提示“操作无法完成,因为其中的文件夹已在另一程序中打开,请关闭该文件或文件夹,然后重试.”,遇到这种情况我们应该怎么办呢?请看下文. 方法/ ...

  7. springmvc源码分析----入门看springmvc的加载过程

    接上一篇我们写的入门---http://www.cnblogs.com/duanxiaojun/p/6591448.html 今天从这个门里进去我们看springmvc是如何在容器启动的时候将各个模块 ...

  8. A - Combination Lock

    Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Description Scroog ...

  9. 5、overflow、hover

    一.overflow 1.属性介绍 说明: 这个属性定义溢出元素内容区的内容会如何处理.如果值为 scroll,不论是否需要,用户代理都会提供一种滚动机制.因此,有可能即使元素框中可以放下所有内容也会 ...

  10. [翻译] 正式宣布 .NET 5

    原文: Introducing .NET 5 今天,我们宣布 .NET Core 3.0 之后的下一个版本将是 .NET 5 .这将是 .NET 系列的下一个重要版本. 将来只会有一个 .NET ,您 ...