转自:http://blog.csdn.net/heliang7/article/details/7309394

主要问题是如何在图片做旋转后计算出新图片的长宽。

在java 2d和基本math库的帮助下,其实利用简单的计算就可以知道。

以下算法只是计算出旋转小于90度时的公式。当旋转大于90时,可以先把问题域换算到锐角的情况,再进行计算即可。

如下图所示,需要计算出来的是len_delta的长度,就是有双竖线的位置,它是新图片要增加的宽。(要增加的高度同理可得。)

其实只要知道len的长度,还有len和len_delta的夹角,就可以算出len_delta的长度了。

1. len的长度。注意到它是等腰三角形的底边,顶角为angel, 容易得到len=2*R*sin(angel/2)

2. len和len_delta的夹角。先可以计算出angel_alpha,也就是等腰三角形的底角 angel_alpha = (PI - angel) / 2

然后是R和原图像的底边的夹角angel_delta,显然其tan值是原图片的高宽比(注意计算增加的高度时是宽高比)。用arctan求出其角度。

len和len_delta的夹角 = PI - angel_alpha - angel_delta

3. len_delta = len * cos(len和len_delta的夹角)

java代码示例如下

    1. <pre name="code" class="java">import java.awt.Dimension;
    2. import java.awt.Graphics2D;
    3. import java.awt.Image;
    4. import java.awt.Rectangle;
    5. import java.awt.image.BufferedImage;
    6. public class RotateImage {
    7. public static BufferedImage Rotate(Image src, int angel) {
    8. int src_width = src.getWidth(null);
    9. int src_height = src.getHeight(null);
    10. // calculate the new image size
    11. Rectangle rect_des = CalcRotatedSize(new Rectangle(new Dimension(
    12. src_width, src_height)), angel);
    13. BufferedImage res = null;
    14. res = new BufferedImage(rect_des.width, rect_des.height,
    15. BufferedImage.TYPE_INT_RGB);
    16. Graphics2D g2 = res.createGraphics();
    17. // transform
    18. g2.translate((rect_des.width - src_width) / 2,
    19. (rect_des.height - src_height) / 2);
    20. g2.rotate(Math.toRadians(angel), src_width / 2, src_height / 2);
    21. g2.drawImage(src, null, null);
    22. return res;
    23. }
    24. public static Rectangle CalcRotatedSize(Rectangle src, int angel) {
    25. // if angel is greater than 90 degree, we need to do some conversion
    26. if (angel >= 90) {
    27. if(angel / 90 % 2 == 1){
    28. int temp = src.height;
    29. src.height = src.width;
    30. src.width = temp;
    31. }
    32. angel = angel % 90;
    33. }
    34. double r = Math.sqrt(src.height * src.height + src.width * src.width) / 2;
    35. double len = 2 * Math.sin(Math.toRadians(angel) / 2) * r;
    36. double angel_alpha = (Math.PI - Math.toRadians(angel)) / 2;
    37. double angel_dalta_width = Math.atan((double) src.height / src.width);
    38. double angel_dalta_height = Math.atan((double) src.width / src.height);
    39. int len_dalta_width = (int) (len * Math.cos(Math.PI - angel_alpha
    40. - angel_dalta_width));
    41. int len_dalta_height = (int) (len * Math.cos(Math.PI - angel_alpha
    42. - angel_dalta_height));
    43. int des_width = src.width + len_dalta_width * 2;
    44. int des_height = src.height + len_dalta_height * 2;
    45. return new java.awt.Rectangle(new Dimension(des_width, des_height));
    46. }
    47. }</pre><br>
    48. <pre></pre>
    49. <p>JUnit测试代码如下,将图片路径改为你自己准备测试用的图片路径即可。</p>
    50. <p></p>
    51. <pre name="code" class="java"><pre name="code" class="java">import java.awt.image.BufferedImage;
    52. import java.io.File;
    53. import java.io.IOException;
    54. import javax.imageio.ImageIO;
    55. import junit.framework.Assert;
    56. import org.junit.Test;
    57. import Jugnoo.RotateImage;
    58. public class RotateImageTest {
    59. @Test
    60. public void testRotate() throws IOException {
    61. BufferedImage src = ImageIO.read(new File("d:/dog.jpg"));
    62. BufferedImage des = RotateImage.Rotate(src, 30);
    63. Assert.assertNotNull(des);
    64. Assert.assertTrue(ImageIO.write(des, "jpg", new File("d:/dog2.jpg")));
    65. // bigger angel
    66. des = RotateImage.Rotate(src, 150);
    67. Assert.assertNotNull(des);
    68. Assert.assertTrue(ImageIO.write(des, "jpg", new File("d:/dog3.jpg")));
    69. // bigger angel
    70. des = RotateImage.Rotate(src, 270);
    71. Assert.assertNotNull(des);
    72. Assert.assertTrue(ImageIO.write(des, "jpg", new File("d:/dog4.jpg")));
    73. }
    74. }</pre><br>
    75. <br>
    76. <pre></pre>
    77. <p></p>
    78. <pre></pre>
    79. <pre></pre>

Java实现图片内容无损任意角度旋转的更多相关文章

  1. JAVA对图片的任意角度旋转,以及镜像操作

    package relevantTest;/* * 该代码实现了对图像的水平镜像变换,垂直镜像变换,任意角度旋转,jtf的实时监控,以及对图像的缩放变换,以及按钮的若隐若现效果. * 在对图像进行任意 ...

  2. 在word中输入任意角度旋转图片

    Sub 图片旋转任意角度() Dim sha As Shape, isa As InlineShape Static s As Integer Application.ScreenUpdating = ...

  3. Winform以任意角度旋转PictureBox中的图片的方法

    方法1: private void RotateFormCenter(PictureBox pb, float angle) { Image img = pb.Image; int newWidth ...

  4. C# 使用 GDI+ 实现添加中心旋转(任意角度)的文字

    这篇文章是 GDI+ 总结系列的第三篇,如果对 GDI+ 的基础使用不熟悉的朋友可以先看第一篇文章<C# 使用 GDI+ 画图>. 需求 需求是要实现给图片添加任意角度旋转的文字,文字的旋 ...

  5. 在图片不被裁剪时opencv绕图片中任意点旋转任意角度

    opencv绕图片中任意角度旋转任意角度   最近在做项目需要把把图片绕图片中任意点旋转任意角度,考虑到自己旋转需要编写插值算法,所以想到了用opencv,但是网上都是围绕图片中点旋转任意角度的,都是 ...

  6. JQuery插件让图片旋转任意角度且代码极其简单

    引入下方的jquery.rotate.js文件,然后通过$("选择器").rotate(角度);可以旋转任意角度, 例如$("#rotate-image").r ...

  7. JQuery插件让图片旋转任意角度且代码极其简单 - 摘自网友

    JQuery插件让图片旋转任意角度且代码极其简单 2012-04-01 09:57:03     我来说两句      收藏    我要投稿 引入下方的jquery.rotate.js文件,然后通过$ ...

  8. 大厂前端工程师教你如何使用css3绘制任意角度扇形+动画

    这里只是做下原理解释,原理:使用两个半圆做角度拼接.比如想绘制一个缺口朝右,缺口弧度30度角的扇形 资源网站搜索大全https://55wd.com 那么将由一个旋转65度角的半圆A+一个旋转-65度 ...

  9. 使用css实现任意大小,任意方向, 任意角度的箭头

    使用css实现任意大小,任意方向, 任意角度的箭头 网页开发中,经常会使用到 下拉箭头,右侧箭头 这样的箭头. 一般用css来实现: { display: inline-block; margin: ...

随机推荐

  1. 解决<%@taglib prefix="s" uri="/struts-tags"%>显示找不到

    问题: jsp中使用<%@taglib prefix="s" uri="/struts-tags"%>显示找不到 解决方法: 在web.xml中插入 ...

  2. Abaqus-GUI开发-RSG

    目录 1. GUI开发简介 2. 目标和消息 2.1消息类型和消息ID 2.2消息映射 3. 控件创建 1. GUI开发简介 Abaqus GUI程序开发时,可以采用两种方式创建GUI图形界面. (1 ...

  3. 原生js按回车键实现登录

    这篇文章主要介绍了原生JS按回车键实现登录的方法,众所周知,这是在web程序设计中的一个非常实用的小技巧,主要用于表单提交,包括注册.登录等等功能,具有很好的用户体验,有着非常广泛的实用价值,需要的朋 ...

  4. VisualTreeHelper 向下提取 元素

    private ChildType FindVisualChild<ChildType>(DependencyObject obj) where ChildType : Dependenc ...

  5. IDEA导入maven项目不自动识别

    解决办法: 选中module的pom.xml,右键,选择" add as maven project",即可刷新为maven项目

  6. 常见排序的Java实现

    插入排序 1.原理:在有序数组中从后向前扫描找到要插入元素的位置,将元素插入进去. 2.步骤:插入元素和依次和前一个元素进行比较,若比前一个元素小就交换位置,否则结束循环. 3.代码: package ...

  7. 吴裕雄--天生自然Numpy库学习笔记:NumPy 高级索引

    import numpy as np x = np.array([[1, 2], [3, 4], [5, 6]]) y = x[[0,1,2], [0,1,0]] print (y) import n ...

  8. ASA防火墙忘记密码之后的恢复步骤

    ASA的密码恢复方法与路由器相似,修改配置寄存器的值,启动时绕过startup-config配置文件,重新配置密码. 密码恢复的步骤如下: (1)重新插拔电源线,重启ASA (2)按ESC键或Ctrl ...

  9. 02-14Android学习进度报告十四

    今天我学习了关于构建一个可复用的自定义BaseAdapter的知识. 首先将Entity设置成泛型 代码示例: public class MyAdapter<T> extends Base ...

  10. 「AT2381 [AGC015C] Nuske vs Phantom Thnook」

    题目大意 给出一个01矩阵,这个矩阵有一个特殊的性质: 对于任意两个 \(1\) 之间最多只有 \(1\) 条由 \(1\) 构成的路径.每次询问给出一个矩形范围,查询在这个范围内的联通快个数. 分析 ...