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. git bash使用端口转发连接服务器

    之前的配置是 url = user@xx.xx.xx.xx:/home/tutu/thelib/ww.git xx.xx.xx.xx是服务器的外网地址,其内网地址是zz.zz.zz.zz 但是现在服务 ...

  2. java 基于百度地图API GPS经纬度解析地址

    首先这是百度地图api 的接口地址,基于接口的参数,不过多介绍,其中都提供相应的介绍: http://lbsyun.baidu.com/index.php?title=webapi/guide/web ...

  3. 骨牌覆盖问题 KxM

    前面我们说了一些简单的骨牌覆盖问题,有了上面的经验,我们可以尝试解决K*M的 思路和上一篇文章所提到的3*N的 很类似: 依然是矩阵快速幂.我们需要把一个小的边固定下来作为的已知边,然后进行矩阵快速幂 ...

  4. hdu-5003 Osu!(水题)

    题目链接: Osu! time Limit: 2000/1000 MS (Java/Others)     Memory Limit: 65536/65536 K (Java/Others) Prob ...

  5. ComboBox的真实值和显示值

    一.类型 /// <summary> /// 下拉框值类型 /// </summary> public class TextAndValue { /// <summary ...

  6. java:calendar类及一些比较实用的utils(一)

    在java编程中经常会用到时间日期的计算.比较.格式化等等操作,刚开始接触Calendar类时,还是在初学习期间,小小白一枚,看着这个好复杂,懒惰心理作祟也就没有怎么去学习,后来在项目中经常用到,索性 ...

  7. json对象和json数组

    json字符串对象和json字符串数组:JSONArray跟JSONObject的区别就是JSONArray比JSONObject多中括号[] jsonObject: "Row": ...

  8. JAVA中的BIO,NIO,AIO

    在了解BIO,NIO,AIO之前先了解一下IO的几个概念: 1.同步 用户进程触发IO操作并等待或者轮询的去查看IO操作是否就绪, 例如自己亲自出马持银行卡到银行取钱 2.异步 用户触发IO操作以后, ...

  9. Flex AIR操作文件系统

    1.各种文件操作http://blog.csdn.net/zdingxin/article/details/6635376 2.file.browseForDirectory("请选择一个目 ...

  10. POJ - 2251 Dungeon Master 多维多方向BFS

    Dungeon Master You are trapped in a 3D dungeon and need to find the quickest way out! The dungeon is ...