Kinect for windows 开发入门 五:彩色数据获取和使用

2024-06-12 18:32

本文主要是介绍Kinect for windows 开发入门 五:彩色数据获取和使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景知识

1.      WriteableBitmap之于Bitmap,就好像StringBuilder之于String。可以减少内存消耗。WriteableBitmap在初始化时需要指定高度,宽度和格式。上一节示例中,获取每帧图像都需要创建并初始化一个新的Bitmap,对GPU来说是一个极大的负担。用WriteableBitmap替代可以很大程度上提高性能。

2.      色彩数据流的解析式根据格式而定的。ColorImageFormat枚举用来指定格式。下表列出了支持的格式。简单的说支持四种颜色格式RGBYUVBayerInfrared;两种分辨率640*4801280*960;最快支持30帧每秒;

ColorImageFormat

Description

含义

InfraredResolution640x480Fps30

16 bits, using the top 10 bits from a PixelFormats.Gray16  format (with the 6 least significant bits always set to 0) whose resolution is  640 x 480 and frame rate is 30 frames per second. Introduced in 1.6.

格式:红外(有亮度无彩色)

分辨率:640*480

最高频:30/

RawBayerResolution1280x960Fps12

Bayer data (8 bits per pixel, layout in alternating  pixels of red, green and blue) whose resolution is 1280 x 960 and frame rate  is 12 frames per second. Introduced in 1.6.

格式:Bayer

分辨率:1280*960

最高频:12/

RawBayerResolution640x480Fps30

Bayer data (8 bits per pixel, layout in alternating  pixels of red, green and blue) whose resolution is 640 x 480 and frame rate  is 30 frames per second. Introduced in 1.6.

格式:Bayer

分辨率:640*480

最高频:30/

RawYuvResolution640x480Fps15

Raw YUV data whose resolution is 640 x 480 and frame  rate is 15 frames per second.

格式:YUV

分辨率:640*480

最高频:15/

RgbResolution1280x960Fps12

RBG data whose resolution is 1280 x 960 and frame rate  is 12 frames per second.

格式:RGB

分辨率:1280*960

最高频:12/

RgbResolution640x480Fps30

RBG data whose resolution is 640 x 480 and frame rate is  30 frames per second.

格式:RGB

分辨率:640*480

最高频:30/

YuvResolution640x480Fps15

YUV data whose resolution is 640 x 480 and frame rate is  15 frames per second.

格式:YUV

分辨率:640*480

最高频:15/

Undefined

The format is not defined.

图像格式未定义

3.      示例中使用的颜色格式是Bgr32,每个像素占4个字节,每个字节8个位。第一个字节是蓝色通道,第二个是绿色,第三个是红色,第四个待用(表示像素的Alpha?或者透明度)。

4.      Stride是指图像中一行像素所占的字节(RGBwidth*height*BytesPerPixel=640*480*4).

5.      Stream会为每一帧图像加一个编号(不一定连续)和Timestamp

6.      获取ColorStream有两种方式,主动和被动,示例中采用被动模式(多用),即被动的持续接受sensor传过来的信息;主动则是,根据需要定时地去获取,可参考“拉”的模式

7.      本示例展示了一些简单的图像处理。下面代码中以EditColor_开头的方法就是各种处理方法。

 

示例代码

        privateKinectSensor kinect;

        privateWriteableBitmapcolorImageBitmap;

        privateInt32RectcolorImageBitmapRect;

        privateint colorImageStride;

        privatebyte[] colorImagePixelData;

 

 

        // Get and set connected Kinect Sensor;

        publicKinectSensor Kinect

        {

            get

            {

                returnthis.kinect;

            }

 

            set

            {

                if (this.kinect != value)

                {

                    if (null != this.kinect)

                   {

                       UninitializeKinectSensor(this.kinect);

                       this.kinect= null;

                   }

                }

 

                if (null != value && KinectStatus.Connected== value.Status)

                {

                    this.kinect = value;

                   InitializeKinectSensor(this.kinect);

                }

            }

        }

 

        // Bind color stream handler

        privatevoid InitializeKinectSensor(KinectSensorkinectSensor)

        {

            if (null != kinectSensor)

            {

                ColorImageStreamcolorStream = kinectSensor.ColorStream;

                colorStream.Enable();

 

                this.colorImageBitmap = newWriteableBitmap(colorStream.FrameWidth,colorStream.FrameHeight, 96, 96, PixelFormats.Bgr32, null);

                this.colorImageBitmapRect = newInt32Rect(0, 0,colorStream.FrameWidth, colorStream.FrameHeight);

                this.colorImageStride =colorStream.FrameWidth * colorStream.FrameBytesPerPixel;

               ColorImageElement.Source = this.colorImageBitmap;

 

               kinectSensor.ColorFrameReady += kinectSensor_ColorFrameReady;

               kinectSensor.Start();

            }

        }

 

        privatevoid UninitializeKinectSensor(KinectSensorkinectSensor)

        {

            if (kinectSensor != null)

            {

               kinectSensor.Stop();

               kinectSensor.ColorFrameReady -= newEventHandler<ColorImageFrameReadyEventArgs>(kinectSensor_ColorFrameReady);

            }

        }

 

        privatevoidkinectSensor_ColorFrameReady(object sender, ColorImageFrameReadyEventArgs e)

        {

            // Use 'using' to dispose the frame after using.

            // 30 frame per second

            using (ColorImageFrame frame =e.OpenColorImageFrame())

            {

                if (null != frame)

                {

                    byte[] pixelData = newbyte[frame.PixelDataLength];

                   frame.CopyPixelDataTo(pixelData);

                   EditColor_HighSaturation(pixelData, frame.BytesPerPixel);

                    this.colorImageBitmap.WritePixels(this.colorImageBitmapRect,pixelData, this.colorImageStride,0);

                }

            }

 

        }

 

        privatevoid EditColor_Inverted(byte[] pixelData, int bytesPerPixel)

        {

            for (int i = 0; i <pixelData.Length; i += bytesPerPixel)

            {

               pixelData[i] = 0x00;//Blue

               pixelData[i + 1] = 0x00;//Green

            }

        }

 

        privatevoid EditColor_Gray(byte[] pixelData, int bytesPerPixel)

        {

            for (int i = 0; i <pixelData.Length; i += bytesPerPixel)

            {

                byte gray = Math.Max(pixelData[i],pixelData[i + 1]);

                gray= Math.Max(gray,pixelData[i + 2]);

               pixelData[i] = gray;

               pixelData[i + 1] = gray;

               pixelData[i + 2] = gray;

            }

        }

 

        privatevoid EditColor_BlackWhite(byte[] pixelData, int bytesPerPixel)

        {

            for (int i = 0; i <pixelData.Length; i += bytesPerPixel)

            {

                byte gray = Math.Min(pixelData[i], pixelData[i + 1]);

                gray= Math.Min(gray,pixelData[i + 2]);

               pixelData[i] = gray;

               pixelData[i + 1] = gray;

               pixelData[i + 2] = gray;

            }

        }

 

        privatevoid EditColor_Apocalyptic(byte[] pixelData, int bytesPerPixel)

        {

            for (int i = 0; i <pixelData.Length; i += bytesPerPixel)

            {

               pixelData[i] = pixelData[i + 1];

               pixelData[i + 2] = (byte)~pixelData[i + 2];

            }

        }

 

        privatevoid EditColor_WashedOut(byte[] pixelData, int bytesPerPixel)

        {

            for (int i = 0; i <pixelData.Length; i += bytesPerPixel)

            {

                double gray =(pixelData[i] * 0.11) + (pixelData[i + 1] * 0.59) + (pixelData[i + 2] * 0.3);

                double desaturation =0.75;

               pixelData[i] = (byte)(pixelData[i] + desaturation * (gray - pixelData[i]));

                pixelData[i + 1] = (byte)(pixelData[i + 1] +desaturation * (gray - pixelData[i + 1]));

               pixelData[i + 2] = (byte)(pixelData[i + 2] + desaturation * (gray - pixelData[i +2]));

            }

        }

 

        privatevoid EditColor_HighSaturation(byte[] pixelData, int bytesPerPixel)

        {

            for (int i = 0; i <pixelData.Length; i += bytesPerPixel)

            {

                if (pixelData[i] < 0x33 ||pixelData[i] > 0xE5)

                {

                   pixelData[i] = 0x00;

                }

                else

                {

                   pixelData[i] = 0Xff;

                }

 

                if (pixelData[i + 1] < 0x33|| pixelData[i + 1] > 0xE5)

                {

                   pixelData[i + 1] = 0x00;

                }

                else

                {

                   pixelData[i + 1] = 0Xff;

                }

 

                if (pixelData[i + 2] < 0x33|| pixelData[i + 2] > 0xE5)

                {

                   pixelData[i + 2] = 0x00;

                }

                else

                {

                   pixelData[i + 1] = 0Xff;

                }

            }

        }

 

        public MainWindow()

        {

           InitializeComponent();

            // Action according to the app start or stop

            this.Loaded += (sender, e) =>DiscoverKinectSensor();

            this.Unloaded += (s, e) => this.kinect = null;

        }

 

        privatevoid DiscoverKinectSensor()

        {

            KinectSensor.KinectSensors.StatusChanged+= KinectSensors_StatusChanged;

            this.Kinect = KinectSensor.KinectSensors.FirstOrDefault(x=> x.Status == KinectStatus.Connected);

        }

 

        // Monitor Kinect Sensor status, get the first one if it'sconnected, remove if it's disconnected.

        privatevoid KinectSensors_StatusChanged(object sender, StatusChangedEventArgs e)

        {

            switch (e.Status)

            {

                caseKinectStatus.Connected:

                    if (null == this.kinect)

                   {

                       this.kinect= e.Sensor;

                   }

                    break;

                caseKinectStatus.Disconnected:

                    if (this.kinect == e.Sensor)

                    {

                       this.kinect= null;

                       this.kinect= KinectSensor.KinectSensors.FirstOrDefault(x => x.Status == KinectStatus.Connected);

                       if (this.kinect == null)

                       {

                           // Message to show that all kinects weredisconnected.

                       }

                   }

                    break;

            }

        }

 

效果演示

翻转:

灰化:

黑白:

血腥:

冲洗:

饱和:

 

这篇关于Kinect for windows 开发入门 五:彩色数据获取和使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1054996

相关文章

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

MySQL 删除数据详解(最新整理)

《MySQL删除数据详解(最新整理)》:本文主要介绍MySQL删除数据的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、前言二、mysql 中的三种删除方式1.DELETE语句✅ 基本语法: 示例:2.TRUNCATE语句✅ 基本语

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

从入门到精通MySQL联合查询

《从入门到精通MySQL联合查询》:本文主要介绍从入门到精通MySQL联合查询,本文通过实例代码给大家介绍的非常详细,需要的朋友可以参考下... 目录摘要1. 多表联合查询时mysql内部原理2. 内连接3. 外连接4. 自连接5. 子查询6. 合并查询7. 插入查询结果摘要前面我们学习了数据库设计时要满

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

MyBatisPlus如何优化千万级数据的CRUD

《MyBatisPlus如何优化千万级数据的CRUD》最近负责的一个项目,数据库表量级破千万,每次执行CRUD都像走钢丝,稍有不慎就引起数据库报警,本文就结合这个项目的实战经验,聊聊MyBatisPl... 目录背景一、MyBATis Plus 简介二、千万级数据的挑战三、优化 CRUD 的关键策略1. 查