使用C與C++/CLI實現Sobel Edge Detector。

http://www.cnblogs.com/oomusou/archive/2008/07/23/sobel_edge_detector.html

 /*
(C) OOMusou 2007 http://oomusou.cnblogs.com Filename : sobel_edge.c
Compiler : Visual C++ 8.0
Description : Demo the how to use sobel detector on gray level image
Release : 07/23/2008 1.0
*/
#include <stdio.h>
#include <stdlib.h>
#include <math.h> #define MASK_N 2
#define MASK_X 3
#define MASK_Y 3
#define WHITE 255
#define BLACK 0 unsigned char *image_s = NULL; // source image array
unsigned char *image_t = NULL; // target image array
FILE *fp_s = NULL; // source file handler
FILE *fp_t = NULL; // target file handler unsigned int width, height; // image width, image height
unsigned int rgb_raw_data_offset;// RGB raw data offset
unsigned char bit_per_pixel; // bit per pixel
unsigned short byte_per_pixel; // byte per pixel // bitmap header
unsigned char header[] = {
0x42, // identity : B
0x4d, // identity : M
, , , , // file size
, , // reserved1
, , // reserved2
, , , , // RGB data offset
, , , , // struct BITMAPINFOHEADER size
, , , , // bmp width
, , , , // bmp height
, , // planes
, , // bit per pixel
, , , , // compression
, , , , // data size
, , , , // h resolution
, , , , // v resolution
, , , , // used colors
, , , // important colors
}; // sobel mask
int mask[MASK_N][MASK_X][MASK_Y] = {
{{-,-,-},
{ , , },
{ , , }}, {{-, , },
{-, , },
{-, , }}
}; int read_bmp(const char *fname_s) {
fp_s = fopen(fname_s, "rb");
if (fp_s == NULL) {
printf("fopen fp_s error\n");
return -;
} // move offset to 10 to find rgb raw data offset
fseek(fp_s, , SEEK_SET);
fread(&rgb_raw_data_offset, sizeof(unsigned int), , fp_s); // move offset to 18 to get width & height;
fseek(fp_s, , SEEK_SET);
fread(&width, sizeof(unsigned int), , fp_s);
fread(&height, sizeof(unsigned int), , fp_s); // get bit per pixel
fseek(fp_s, , SEEK_SET);
fread(&bit_per_pixel, sizeof(unsigned short), , fp_s);
byte_per_pixel = bit_per_pixel / ; // move offset to rgb_raw_data_offset to get RGB raw data
fseek(fp_s, rgb_raw_data_offset, SEEK_SET); image_s = (unsigned char *)malloc((size_t)width * height * byte_per_pixel);
if (image_s == NULL) {
printf("malloc images_s error\n");
return -;
} image_t = (unsigned char *)malloc((size_t)width * height * byte_per_pixel);
if (image_t == NULL) {
printf("malloc image_t error\n");
return -;
} fread(image_s, sizeof(unsigned char), (size_t)(long)width * height * byte_per_pixel, fp_s); return ;
} // convert RGB to gray level int
int color_to_int(int r, int g, int b) {
return (r + g + b) / ;
} int sobel(double threshold) {
unsigned int x, y, i, v, u; // for loop counter
unsigned char R, G, B; // color of R, G, B
double val[MASK_N] = {0.0};
int adjustX, adjustY, xBound, yBound;
double total; for(y = ; y != height; ++y) {
for(x = ; x != width; ++x) {
for(i = ; i != MASK_N; ++i) {
adjustX = (MASK_X % ) ? : ;
adjustY = (MASK_Y % ) ? : ;
xBound = MASK_X / ;
yBound = MASK_Y / ; val[i] = 0.0;
for(v = -yBound; v != yBound + adjustY; ++v) {
for (u = -xBound; u != xBound + adjustX; ++u) {
if (x + u >= && x + u < width && y + v >= && y + v < height) {
R = *(image_s + byte_per_pixel * (width * (y+v) + (x+u)) + );
G = *(image_s + byte_per_pixel * (width * (y+v) + (x+u)) + );
B = *(image_s + byte_per_pixel * (width * (y+v) + (x+u)) + ); val[i] += color_to_int(R, G, B) * mask[i][u + xBound][v + yBound];
}
}
}
} total = 0.0;
for (i = ; i != MASK_N; ++i) {
total += val[i] * val[i];
} total = sqrt(total); if (total - threshold >= ) {
// black
*(image_t + byte_per_pixel * (width * y + x) + ) = BLACK;
*(image_t + byte_per_pixel * (width * y + x) + ) = BLACK;
*(image_t + byte_per_pixel * (width * y + x) + ) = BLACK;
}
else {
// white
*(image_t + byte_per_pixel * (width * y + x) + ) = WHITE;
*(image_t + byte_per_pixel * (width * y + x) + ) = WHITE;
*(image_t + byte_per_pixel * (width * y + x) + ) = WHITE;
}
}
} return ;
} int write_bmp(const char *fname_t) {
unsigned int file_size; // file size fp_t = fopen(fname_t, "wb");
if (fp_t == NULL) {
printf("fopen fname_t error\n");
return -;
} // file size
file_size = width * height * byte_per_pixel + rgb_raw_data_offset;
header[] = (unsigned char)(file_size & 0x000000ff);
header[] = (file_size >> ) & 0x000000ff;
header[] = (file_size >> ) & 0x000000ff;
header[] = (file_size >> ) & 0x000000ff; // width
header[] = width & 0x000000ff;
header[] = (width >> ) & 0x000000ff;
header[] = (width >> ) & 0x000000ff;
header[] = (width >> ) & 0x000000ff; // height
header[] = height &0x000000ff;
header[] = (height >> ) & 0x000000ff;
header[] = (height >> ) & 0x000000ff;
header[] = (height >> ) & 0x000000ff; // bit per pixel
header[] = bit_per_pixel; // write header
fwrite(header, sizeof(unsigned char), rgb_raw_data_offset, fp_t); // write image
fwrite(image_t, sizeof(unsigned char), (size_t)(long)width * height * byte_per_pixel, fp_t); fclose(fp_s);
fclose(fp_t); return ;
} int main() {
read_bmp("lena.bmp"); // 24 bit gray level image
sobel(90.0);
write_bmp("lena_sobel.bmp");
}
 /*
(C) OOMusou 2007 http://oomusou.cnblogs.com Filename : sobel_edge.cpp
Compiler : C++/CLI / Visual C++ 8.0
Description : Demo the how to use sobel detector on gray level image
Release : 07/23/2008 1.0
*/ #include "stdafx.h"
#include <cmath> using namespace System::Drawing;
using namespace System::Drawing::Imaging; const int MASK_N = ;
const int MASK_X = ;
const int MASK_Y = ; // Convert RGB to gray level int
int colorToInt(Color %color) {
return (color.R + color.G + color.B) / ;
} void edgeDetector(Bitmap^ oriImg, Bitmap^ resImg, const int mask[MASK_N][MASK_X][MASK_Y], const double &threshold) {
double val[MASK_N] = {0.0}; for(int y = ; y != oriImg->Height; ++y) {
for(int x = ; x != oriImg->Width; ++x) {
for(int i = ; i != MASK_N; ++i) {
int adjustX = (MASK_X % ) ? : ;
int adjustY = (MASK_Y % ) ? : ;
int xBound = MASK_X / ;
int yBound = MASK_Y / ; val[i] = 0.0;
for(int v = -yBound; v != yBound + adjustY; ++v) {
for (int u = -xBound; u != xBound + adjustX; ++u) {
if (x + u >= && x + u < oriImg->Width && y + v >= && y + v < oriImg->Height) {
val[i] += colorToInt(oriImg->GetPixel(x + u, y + v)) * mask[i][u + xBound][v + yBound];
}
}
}
} double total = 0.0;
for (int i = ; i != MASK_N; ++i) {
total += val[i] * val[i];
} total = sqrt(total); if (total - threshold >= )
resImg->SetPixel(x , y, Color::Black);
else
resImg->SetPixel(x, y, Color::White);
}
}
} int main() {
const int mask[MASK_N][MASK_X][MASK_Y] = {
{{-,-,-},
{ , , },
{ , , }}, {{-, , },
{-, , },
{-, , }}
}; Bitmap^ oriImg = gcnew Bitmap("lena.bmp");
Bitmap^ resImg = gcnew Bitmap(oriImg->Width, oriImg->Height); const double threshold = 90.0;
edgeDetector(oriImg, resImg, mask, threshold); resImg->Save("lena_sobel.bmp"); return ;
}

每天进步一点点------如何实现Sobel Edge Detector? (Image Processing) (C/C++)的更多相关文章

  1. Canny Edge Detector

    Canny边缘检测算法有自己的理论和经验性的推导, 没仔细看/没看明白. 它的步骤如下: 对原图的灰度图进行高斯滤波 求一阶导数, 得到每个像素点的梯度强度和方向. 非最大抑制. 对每个edge ca ...

  2. Understanding Convolution in Deep Learning

    Understanding Convolution in Deep Learning Convolution is probably the most important concept in dee ...

  3. 32个最热CPLD-FPGA论坛

    1. OPENCORES.ORG这里提供非常多,非常好的PLD了内核,8051内核就可以在里面找到.进入后,选择project或者由http//www.opencores.org/browse.cgi ...

  4. [转]FPGA网站推荐

    1. OPENCORES.ORG这里提供非常多,非常好的PLD了内核,8051内核就可以在里面找到.进入后,选择project或者由http//www.opencores.org/browse.cgi ...

  5. Computer Vision_33_SIFT:Remote Sensing Image Registration With Modified SIFT and Enhanced Feature Matching——2017

    此部分是计算机视觉部分,主要侧重在底层特征提取,视频分析,跟踪,目标检测和识别方面等方面.对于自己不太熟悉的领域比如摄像机标定和立体视觉,仅仅列出上google上引用次数比较多的文献.有一些刚刚出版的 ...

  6. OpenCV2马拉松第14圈——边缘检測(Sobel,prewitt,roberts)

    收入囊中 差分在边缘检測的角色 Sobel算子 OpenCV sobel函数 OpenCV Scharr函数 prewitt算子 Roberts算子 葵花宝典 差分在边缘检測究竟有什么用呢?先看以下的 ...

  7. Sobel Derivatives

    https://docs.opencv.org/2.4/doc/tutorials/imgproc/imgtrans/sobel_derivatives/sobel_derivatives.html ...

  8. 线特征---Edge Drawing(七)

    http://ceng.anadolu.edu.tr/cv/edgedrawing/ References C. Topal, C. Akinlar, Edge Drawing: A Combined ...

  9. 2.7 Sobel导数

    OpenCV函数 Sobel(src_gray,grad_x/grad_y,ddepth,x_order,y_order,scale,delta,BORDER_DEFAULT ) Scharr( ) ...

随机推荐

  1. Wannafly Camp 2020 Day 2J 邦邦的2-SAT模板

    #include <bits/stdc++.h> using namespace std; int main() { int n; cin>>n; cout<<n& ...

  2. nginx 无法启动:bind() to 0.0.0.0:443 failed

    bind to 0.0.0.0:443 failed.其实就是443端口被其它程序占用,要结束占用443端口导致nginx不能启动的应用. CMD: 1.查看所有程序使用的端口 netstat -ao ...

  3. 2020牛客寒假算法基础集训营2 J.求函数 (线段树 推公式 单点修改 区间查询)

    https://ac.nowcoder.com/acm/contest/3003/J 题解: #include<bits/stdc++.h> typedef long long ll; u ...

  4. C++类this指针为空时的几个误区

    代码: class test{ public: static void f1(){cout<<y<<endl;} void f2(){cout<<y<< ...

  5. Java连载84-Collection的常用方法、迭代器

    一.Collections的常用方法介绍 1.承接上次连载,先介绍几个简单的常用方法 package com.bjpowernode.java_learning; import java.util.* ...

  6. Linux忘记 root密码的解决办法

    很多朋友经常会忘记Linux系统的root密码,linux系统忘记root密码的情况该怎么办呢?重新安装系统吗?当然不用!进入单用户模式更改一下root密码即可. 步骤如下: 重启linux系统 重启 ...

  7. AcWing 1016. 最大上升子序列和

    #include<iostream> using namespace std ; ; int f[N]; int a[N]; int main() { int n; cin>> ...

  8. 构造和改变一个string的其他方法

    构造一个string的其他方法: 1string s1(cp, n),cp为c风格数组名或一个指向C风格数组的指针,则执行的操作为拷贝从cp开始包括cp在内的接下来n个字符给s1,n的默认值为size ...

  9. not under version control

    表示这个文件没有在SVN的控制之下 你在执行commit操作的时候,只有处于SVN控制之下的文件才能被commit,从update得到的文件是在SVN控制之下的(比如原来就存在的文本文件,你可以对其修 ...

  10. NAND FLASH驱动框架以及程序实现

    1.NAND FLASH的硬件连接: 实验用的NAND FLASH芯片为K9F2G08U0C,它是三星公司的存储芯片,它的大小为256M.它的接线图如下所示: 它的每个引脚的分别为LDATA0-LDA ...