正文

转贴 J2ME插值算法实现图片的放大缩小2007-05-08 20:17:00

【评论】 【打印】 【字体: 】 本文链接:http://blog.pfan.cn/sword2008/25718.html

分享到:

前段时间接触了一些数字图像处理的问题,在1位师兄的指导下,在j2me平台,完成了一些基本的2D图像处理算法。就当是对这段知识做一下总结,决定把这些算法写出来,和各位朋友共同探讨。这篇文章先介绍图像放大缩小的实现,程序是以Nokia S40的机器为平台实现的。
1、实现图形缩放的基本思想:
图像的变形变换,简单的说就是把源图像每个点坐标通过变形运算转为目标图像相应点的新坐标,但是这样会导致一个问题就是目标点的坐标通常不会是整数。所以我们在做放大变换时,需要计算生成没有被映射到的点;而在缩小变换时,需要删除一些点。这里我们采用最简单的一种插值算法:“最近邻域法”。顾名思义,就是把非整数坐标作一个四舍五入,取最近的整数点。
看下面的一个图片放大的例子,左图为原始图像,右图为放大1倍的图像。里面的数字,表示所在像素的信息

1 2 1 1 2 2

5 4 5 5 4 4

2、对于图片像素的操作:
获取Image图片像素信息:
标准的midp1.0没有提供获取图片像素信息的函数,对于NOKIA的机器,我们可以采用Nokia SDK提供的API获取像素信息。具体程序如下:
g = image.getGraphics()
DirectGraphics dg = DirectUtils.getDirectGraphics(g);
dg.getPixels(short[] pixels, int offset, int scanlength, int x,int y, int width, int height, int format)
参数介绍:
short[] pixels: 用于接收像素信息的数组
int offset:这篇文章中的用到的地方,添0就可以了
int scanlength:添图片的宽度就行了
int x:添0
int y:添0
int width:图片宽度
int height:图片高度
int format:444,表示图形格式,好像Nokia S40的机器都是采用444格式表示RGB颜色的。就是红,绿,蓝各用4位表示,至于可以表示透明色ARGB的4444格式,应该是机器硬件实现的。
想具体了解Nokia SDK的信息,可以查看Nokia SDK的帮助文档。

使用像素信息数组生成Image图片:
image = Image.createImage(w, h);
g = image.getGraphics()
DirectGraphics dg = DirectUtils.getDirectGraphics(g);
dg.drawPixels(short[] pixels,boolean transparency, int offset, int scanlength, int x, int y, int width,int height, int manipulation, int format)
short[] pixels:像素信息数组
boolean transparency:是否包含alpha位信息
int offset:添 0
int scanlength:添图片的宽度就行了
int x:添 0
int y:添 0
int width:图片宽度
int height:图片高度
int manipulation:添 0
int format:444

下面开始介绍具体的算法,首先给出图像缩放的完整函数,然后对代码,分段进行解释
/*********************************
* @todo 图片放大缩小
* @param srcImg 原始图片
* @param desW 变化后图片的宽
* @param desH 变化后图片的高
* @return 处理后的图片
*********************************/
private Image ZoomImage(Image srcImg, int desW, int desH) {
int srcW = srcImg.getWidth(); //原始图像宽
int srcH = srcImg.getHeight(); //原始图像高

short[] srcBuf = new short[srcW * srcH]; //原始图片像素信息缓存

//srcBuf获取图片像素信息
Image desImg = Image.createImage(srcW, srcH);
if (srcImg.isMutable()) { /*如果是可变图像*/
DirectUtils.getDirectGraphics(srcImg.getGraphics()).
getPixels(srcBuf, 0, srcW, 0, 0, srcW, srcH, 444);
} else { /*如果是非可变图像*/
desImg.getGraphics().drawImage(srcImg, 0, 0, 0);
DirectUtils.getDirectGraphics(desImg.getGraphics()).
getPixels(srcBuf, 0, srcW, 0, 0, srcW, srcH, 444);
}

//计算插值表
short[] tabY = new short[desH];
short[] tabX = new short[desW];

int sb = 0;
int db = 0;
int tems = 0;
int temd = 0;
int distance = srcH > desH ? srcH : desH;
for (int i = 0; i <= distance; i++) { /*垂直方向*/
tabY[db] = (short) sb;
tems += srcH;
temd += desH;
if (tems > distance) {
tems -= distance;
sb++;
}
if (temd > distance) {
temd -= distance;
db++;
}
}

sb = 0;
db = 0;
tems = 0;
temd = 0;
distance = srcW > desW ? srcW : desW;
for (int i = 0; i <= distance; i++) { /*水平方向*/
tabX[db] = (short) sb;
tems += srcW;
temd += desW;
if (tems > distance) {
tems -= distance;
sb++;
}
if (temd > distance) {
temd -= distance;
db++;
}
}

//生成放大缩小后图形像素buf
short[] desBuf = new short[desW * desH];
int dx = 0;
int dy = 0;
int sx = 0;
int sy = 0;
int oldy = -1;
for (int i = 0; i < desH; i++) {
if (oldy == tabY[i]) {
System.arraycopy(desBuf, dy - desW, desBuf, dy, desW);
} else {
dx = 0;
for (int j = 0; j < desW; j++) {
desBuf[dy + dx] = srcBuf[sy + tabX[j]];
dx++;
}
sy += (tabY[i] - oldy) * srcW;
}
oldy = tabY[i];
dy += desW;
}

//生成图片
desImg = Image.createImage(desW, desH);
DirectUtils.getDirectGraphics(desImg.getGraphics()).
drawPixels(desBuf, true, 0, desW, 0, 0, desW, desH, 0, 444);
return desImg;
}

首先看函数的头两句,很容易,就是获取原始图片的宽度和高度
int srcW = srcImg.getWidth(); //原始图像宽
int srcH = srcImg.getHeight(); //原始图像高
接下来一句我们要定义一个short型数组,作为获取原始图片像素信息的缓存
short[] srcBuf = new short[srcW * srcH];

 

 

//////////////////////////////////////////////////////////////////////////////////////////

J2ME(MIDP2.0)图像处理演示程序(完整源代码)增强版

J2ME(MIDP2.0)图像处理演示程序(完整源代码)增强版

此程序的创意来自于the3gwireless(http://blog.csdn.net/the3gwireless),第一版的源代码由the3gwireless利用从互联网上获得的资料整理修改完成。

本例的目的是帮助J2ME的爱好者学习和掌握基本的图像处理知识,也希望其它深谙此道的朋友对代码进行进一步完善和增强。

由于此代码主要作为学习使用,时间仓促,程序结构和内容肯定会有许多问题,希望大家谅解。

大家可以任意修改和使用此代码,如果做过修改并要在互联网上公开传播,请一定在本文后注明自己的名字,修改或者增强的功能以及新的版本号.并将新的文件发给sam4it@gmail.com以便在下载服务器上更新

当前实现的功能:

状态条显示从服务器上读取图像,

图像的90度/180度翻转(使用了MIDP2.0自带的方法,但包括了自己实现的代码),

图像的水平镜像,垂直镜像(使用了MIDP2.0自带的方法,但包括了自己实现的代码),

图像的缩小,增大,

图像的淡出渐进,

图像的保存和加载(RMS),

生成拇指图标,

横向纵向滚动图片,

调整图像及背景颜色

版本v1.0.1新增强的功能有:

添加了灰度显示图像(两种方法)

调节图像明暗

反转显示图像

版本v1.0.2新增强的功能有:

支持CMWAP和CMNET两种连接方式

动态更新图库列表

清空手机存储图片功能

系统公告

下载地址:

用手机:

http://www.the3gwireless.com/ImageAlbum/ImageAlbum.jad

http://www.the3gwireless.com/ImageAlbum/ImageAlbum.jar

源码:

http://www.the3gwireless.com/ImageAlbum/source.rar

 

在线测试:

 

http://webstart.mpowerplayer.com/www.the3gwireless.com/ImageAlbum/ImageAlbum.jad.jnlp

截屏:

阅读(4340) | 评论(0)


版权声明:编程爱好者网站为此博客服务提供商,如本文牵涉到版权问题,编程爱好者网站不承担相关责任,如有版权问题请直接与本文作者联系解决。谢谢!

评论

暂无评论
您需要登录后才能评论,请 登录 或者 注册