转自: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. 两个Beta函数类型的积分及其一般形式

    \[\Large\displaystyle \int_{0}^{1}\frac{\sqrt[4]{x\left ( 1-x \right )^{3}}}{\left ( 1+x \right )^{3 ...

  2. linux查看公网ip的方法

    curl ifconfig.me 或者 curl cip.cc

  3. LeetCode练题——67. Add Binary

    1.题目 67. Add Binary——easy Given two binary strings, return their sum (also a binary string). The inp ...

  4. 《JavaScript高级程序设计》读书笔记(四)变量、作用域和内存问题

    内容---理解基本类型和引用类型的值---理解执行环境---理解垃圾收集 --JavaScript变量松散类型的本质--决定了它只是在特定时间用于保存特定值的一个名字而已--变量的值及其数据类型可以在 ...

  5. 5种JVM调优配置方法概览!!!

    本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...

  6. Fluent_Python_Part4面向对象,08-ob-ref,对象引用、可变性和垃圾回收

    第四部分第8章,对象引用.可变性和垃圾回收 1. 创建对象之后才会把变量分配给对象 变量是对象的标注,是对象的别名,是对象的引用,并不是对象存储的地方. 例子1. 证明赋值语句的右边先执行 class ...

  7. ubuntu-查看所有用户

    cat /etc/shadow :后面的全是用户

  8. 【PAT甲级】1050 String Subtraction (20 分)

    题意: 输入两个串,长度小于10000,输出第一个串去掉第二个串含有的字符的余串. trick: ascii码为0的是NULL,减去'0','a','A',均会导致可能减成负数. AAAAAccept ...

  9. KEAZ128 时钟配置

    本文介绍如何用KEAZ128评估版(FRDM-KEAZ128Q80)配置为40MHz core freqency/20MHz bus frequency. 1.了解器件时钟特性 参见NXP KEA12 ...

  10. 2.1.1Remove Duplicates from Sorted Arr

    /* 题目:2.1.1 Remove Duplicates from Sorted Array Given a sorted array, remove the duplicates in place ...