C#生成影像金字塔的原理实例

这里首先先讲一下金字塔的原理:

影像金字塔就是把一个原始图像处理成一个类似于塔状的影像结构(请不要纠结于这个概念)

在影像金字塔中最精细层的分辨率为16*16,下一层是8*8,依次为4*4,2*2,1*1

如下图所示:

图中网格顶点表示原始图像16*16;红色差号代表8*8;黑色圆圈代表4*4.

按照上述原理,那么编程实现不过就是隔点采样,那么相邻两幅图像在对应点上的灰度值应该相等,当然我是这么想的。

C#在Drawing命名空间下的Bitmap类里面有一个处理影像金字塔的函数,


代码如下:

public void mipmaping()

{

Bitmap jpgsource =new  Bitmap(@"E:\height512.jpg");

Bitmap jpgTarget = new Bitmap(jpgsource, jpgsource.Width>>1, jpgsource.Height>>1);

jpgTarget.Save(@"E:\height256.jpg");

}

但是经过我逐像素对比之后,发现这两张图上对应点的像素并不相等,这说明一个问题:C#里的这个方法肯定不是按照逐点采样,而是采用某种方法把图像拉伸的。这似乎不太符合金字塔的原理,所以我又重新写了一个金字塔的函数:


代码如下:

public void Rescale()

{

Bitmap myImage1024 = new Bitmap(@"E:\height.jpg");

int width = myImage1024.Height; ;

int height = myImage1024.Width; ;

Console.WriteLine(DateTime.Now.ToString());

for (int power = 1; power < 3; power++)

{

Bitmap myImage = new Bitmap(width >> power, height >> power);

int row = myImage.Height;

int column = myImage.Width;

for (int i = 0; i < row; i++)

for (int j = 0; j < column; j++)

{   //这里是精要,实际上就是按照采样间距 踩点

int i1024 = getcoor(i, power); int j1024 = getcoor(j, power);

Color color1024 = myImage1024.GetPixel(i1024, j1024);

myImage.SetPixel(i, j, color1024);

}

myImage.Save(string.Format(@"E:\myjpg\height{0}.jpg", width >> power));

}

Console.WriteLine(DateTime.Now.ToString());

}

public int getcoor(int i,int power)

{

if (power > 1)

return 2 * getcoor(i,power-1) +1;

else

return 2 * i + 1;

}

按照我写的这个函数来建造金字塔,此时相邻图层之间的对应点像素值就相等了。

(0)

相关推荐

  • 关于c#连接ftp进行上传下载实现原理及代码

    复制代码 代码如下: using System; using System.Collections.Generic; using System.Text; using System.Net; using System.IO; namespace ftponload { class Program { static void Main(string[] args) { //上传文件的方法 onload("D://outPut.txt"); //下载文件的方法 fload(); } pub

  • C#:foreach与yield语句的介绍

    1. foreach语句 C#编译器会把foreach语句转换为IEnumerable接口的方法和属性. 复制代码 代码如下: foreach (Person p in persons) { Console.WriteLine(p); } foreach语句会解析为下面的代码段. •调用GetEnumerator()方法,获得数组的一个枚举•在while循环中,只要MoveNext()返回true,就一直循环下去•用Current属性访问数组中的元素 复制代码 代码如下: IEnumerator

  • C#使用yield关键字让自定义集合实现foreach遍历的方法

    foreach遍历是C#常见的功能,而本文通过实例形式展现了C#使用yield关键字让自定义集合实现foreach遍历的方法.具体步骤如下: 一般来说当我们创建自定义集合的时候为了让其能支持foreach遍历,就只能让其实现IEnumerable接口(可能还要实现IEnumerator接口) 但是我们也可以通过使用yield关键字构建的迭代器方法来实现foreach的遍历,且自定义的集合不用实现IEnumerable接口 注意:虽然不用实现IEnumerable接口 ,但是迭代器的方法必须命名为

  • C#正则表达式Regex类用法实例分析

    本文实例讲述了C#正则表达式Regex类用法,分享给大家供大家参考.具体实现方法如下: 一.在C#中,要使用正则表达式类,请在源文件开头处添加以下语句: 复制代码 代码如下: using System.Text.RegularExpressions; 二.RegEx类常用的方法 1.静态Match方法 使用静态Match方法,可以得到源中第一个匹配模式的连续子串. 静态的Match方法有2个重载,分别是 复制代码 代码如下: Regex.Match(string input, string pa

  • C#中foreach原理以及模拟的实现

    本文实例讲述了C#中foreach原理以及模拟的实现方法,分享给大家供大家参考.具体如下: 复制代码 代码如下: public class Person:IEnumerable     //定义一个person类  并且 实现IEnumerable 接口  (或者不用实现此接口 直接在类 //里面写个GetEnumerator()方法) {         string[] names = { "小杨", "科比布莱恩特", "凯文杜兰特", &

  • C#实现跨线程操作控件方法

    本文实例讲述了C#实现跨线程操作控件方法,分享给大家供大家参考.具体实现方法如下: 由于在.net平台下Winform.wpf禁止跨线程直接访问控件,因此在必须跨线程访问控件的时候采用异步方式. 1.winform项目中跨线程访问控件: 编写一个Winform小实例:在做winform项目中,有时为了将系统运行的状态实时显示到Form中,因此添加一个RichTextbox控件实时显示系统运行日志.本例实现的操作是将日志以字符串的形式写入RichTextbox控件,因为是实时显示,所以涉及到跨线程

  • asp.net c#采集需要登录页面的实现原理及代码

    首先说明:代码片段是从网络获取,然后自己修改.我想好的东西应该拿来分享. 实现原理:当我们采集页面的时候,如果被采集的网站需要登录才能采集.不管是基于Cookie还是基于Session,我们都会首先发送一个Http请求头,这个Http请求头里面就包含了网站需要的Cookie信息.当网站接收到发送过来的Http请求头时,会从Http请求头获取相关的Cookie或者Session信息,然后由程序来处理,决定你是否有权限访问当前页面. 好了,原理搞清楚了,就好办了.我们所要做的仅仅是在采集的时候(或者

  • C# 解析 Excel 并且生成 Csv 文件代码分析

    今天工作中遇到一个需求,就是获取 excel 里面的内容,并且把 excel 另存为 csv,因为本人以前未接触过,所以下面整理出来的代码均来自网络,具体参考链接已丢失,原作者保留所有权利! 例子: 复制代码 代码如下: using System; using System.Data; namespace ExportExcelToCode {     class ExcelOperater     {         public void Operater()         {      

  • Base64编码解码原理及C#编程实例

    一. Base64编码由来 为什么会有Base64编码呢?因为有些网络传送渠道并不支持所有的字节,例如传统的邮件只支持可见字符的传送,像ASCII码的控制字符就不能通过邮件传送.这样用途就受到了很大的限制,比如图片二进制流的每个字节不可能全部是可见字符,所以就传送不了.最好的方法就是在不改变传统协议的情况下,做一种扩展方案来支持二进制文件的传送.把不可打印的字符也能用可打印字符来表示,问题就解决了.Base64编码应运而生,Base64就是一种基于64个可打印字符来表示二进制数据的表示方法. 二

  • C#之CLR内存原理初探

    本文初步讲述了C#的CLR内存原理.这里所关注的内存里面说没有寄存器的,所以我们关注的只有托管堆(heap),栈(stack), 字符串常量池(其中string是一个很特殊的对象) 首先我们看两个方法: void M1() { string name = "Tom"; M2(name); } void M2(string name2) { int length = 10; double rate = 10.0; name2 = "Joe"; return; } 这里

随机推荐