贝塞尔曲线 java_贝塞尔曲线理论及实现——Java篇

贝塞尔曲线贝塞尔曲线(The Bézier Curves),是一种在计算机图形学中相当重要的参数曲线(2D,3D的称为曲面)。贝塞尔曲线于1962年,由法国工程师皮埃尔·贝塞尔(Pierre Bézier)所发表,他运用贝塞尔曲线来为汽车的主体进行设计。

f7975793b3af06badf366d8788ef59ca.png线性曲线给定点P0、P1,线性贝塞尔曲线只是一条两点之间的直线。这条线由下式给出:

2e0db88dbe84e0b2d67a6c5c73d1d61b.png当参数t变化时,其过程如下:

6c442d982ac3755d75dc9a0f84a6869a.gif线性贝塞尔曲线函数中的t会经过由P0至P1的B(t)所描述的曲线。例如当t=0.25时,B(t)即一条由点P0至P1路径的四分之一处。就像由0至1的连续t,B(t)描述一条由P0至P1的直线。二次曲线二次方贝塞尔曲线的路径由给定点P0、P1、P2的函数B(t)追踪:

57f79f1878e09a05b8d9d9721df30e18.png为建构二次贝塞尔曲线,可以中介点Q0和Q1作为由0至1的t:* 由P0至P1的连续点Q0,描述一条线性贝塞尔曲线。* 由P1至P2的连续点Q1,描述一条线性贝塞尔曲线。* 由Q0至Q1的连续点B(t),描述一条二次贝塞尔曲线。二次曲线看起来就是这样的:

b4517069f649b6e916491524c1d48344.gif三次曲线为建构高阶曲线,便需要相应更多的中介点。对于三次曲线,可由线性贝塞尔曲线描述的中介点Q0、Q1、Q2,和由二次曲线描述的点R0、R1所建构。P0、P1、P2、P3四个点在平面或在三维空间中定义了三次方贝塞尔曲线。曲线起始于P0走向P1,并从P2的方向来到P3。一般不会经过P1或P2;这两个点只是在那里提供方向资讯。P0和P1之间的间距,决定了曲线在转而趋进P3之前,走向P2方向的“长度有多长”。曲线的参数形式为:

0df5aaa4201ccef8c8ce8821677c7738.png看起来就是这样的:

bf616d7129e278fd092a74753c913414.gif高阶曲线更高阶的贝塞尔曲线,可以用以下公式表示:用

602cc7c6746d94bcb1beb2f7f81ff068.png表示由点P0、P1、…、Pn所决定的贝塞尔曲线。则有:

0889fe0f24fc0729f82c6922d9852178.png更多的关于贝塞尔曲线的内容,你可以去查阅各种数学书。

代码实现其实也非常简单,把公式套过来直接用起来就OK了。这里我以3次曲线为例

0df5aaa4201ccef8c8ce8821677c7738.png

for (double k = t; k <= 1 + t; k += t) {

x = (1 - k) * (1 - k) * (1 - k) * ps[0].getX() + 3 * k * (1 - k) * (1 - k) * ps[1].getX()

+ 3 * k * k * (1 - k) * ps[2].getX() + k * k * k * ps[3].getX();

y = (1 - k) * (1 - k) * (1 - k) * ps[0].getY() + 3 * k * (1 - k) * (1 - k) * ps[1].getY()

+ 3 * k * k * (1 - k) * ps[2].getY() + k * k * k * ps[3].getY();

g2.drawOval((int) x, (int) y, 1, 1);

}

以上是核心算法,下面是完整代码:

package com.opentcs.customization;

import java.awt.Color;

import java.awt.Dimension;

import java.awt.Graphics;

import java.awt.Graphics2D;

import java.awt.RenderingHints;

import java.awt.event.MouseEvent;

import java.awt.event.MouseListener;

import java.awt.event.MouseMotionListener;

import java.awt.geom.Ellipse2D;

import java.awt.geom.Point2D;

import javax.swing.JFrame;

import javax.swing.JPanel;

public class BezierDemo extends JPanel implements MouseListener, MouseMotionListener {

private static final long serialVersionUID = 1L;

private Point2D[] ps;

private Ellipse2D.Double[] ellipse;

private static final double SIDELENGTH = 8;

private int numPoints;

private double t = 0.002;

public BezierDemo() {

numPoints = 0;

ps = new Point2D[4];

ellipse = new Ellipse2D.Double[4];

this.addMouseListener(this);

this.addMouseMotionListener(this);

}

@Override

protected void paintComponent(Graphics g) {

// TODO Auto-generated method stub

super.paintComponent(g);

Graphics2D g2 = (Graphics2D) g;

g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);

g2.setColor(Color.BLUE);

for (int i = 0; i < numPoints; i++) { // 绘制圆点 if (i > 0 && i < (numPoints - 1)) { g2.fill(ellipse[i]); } else { g2.draw(ellipse[i]); } // 绘制点之间的连接线 if (numPoints > 1 && i < (numPoints - 1))

g2.drawLine((int) ps[i].getX(), (int) ps[i].getY(), (int) ps[i + 1].getX(), (int) ps[i + 1].getY());

}

if (numPoints == 4) {

double x, y;

g2.setColor(Color.RED);

for (double k = t; k <= 1 + t; k += t) {

x = (1 - k) * (1 - k) * (1 - k) * ps[0].getX() + 3 * k * (1 - k) * (1 - k) * ps[1].getX()

+ 3 * k * k * (1 - k) * ps[2].getX() + k * k * k * ps[3].getX();

y = (1 - k) * (1 - k) * (1 - k) * ps[0].getY() + 3 * k * (1 - k) * (1 - k) * ps[1].getY()

+ 3 * k * k * (1 - k) * ps[2].getY() + k * k * k * ps[3].getY();

g2.drawOval((int) x, (int) y, 1, 1);

}

}

}

@Override

public Dimension getPreferredSize() {

// TODO Auto-generated method stub

return new Dimension(600, 600);

}

public static void main(String[] agrs) {

JFrame f = new JFrame();

BezierDemo t2 = new BezierDemo();

f.add(t2);

f.pack();

f.setVisible(true);

f.setLocationRelativeTo(null);

f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

}

@Override

public void mouseClicked(MouseEvent e) {

// TODO Auto-generated method stub

if (numPoints < 4) {

double x = e.getX();

double y = e.getY();

ps[numPoints] = new Point2D.Double(x, y);

Ellipse2D.Double current = new Ellipse2D.Double(x - SIDELENGTH / 2, y - SIDELENGTH / 2, SIDELENGTH,

SIDELENGTH);

ellipse[numPoints] = current;

numPoints++;

repaint();

}

}

private int flag = -1;

@Override

public void mousePressed(MouseEvent e) {

// TODO Auto-generated method stub

if (!find((Point2D) e.getPoint()))

flag = -1;

}

private boolean find(Point2D p) {

for (int i = 0; i < numPoints; i++) {

if (ellipse[i].contains(p)) {

flag = i;

return true;

}

}

return false;

}

@Override

public void mouseReleased(MouseEvent e) {

// TODO Auto-generated method stub

}

@Override

public void mouseEntered(MouseEvent e) {

// TODO Auto-generated method stub

}

@Override

public void mouseExited(MouseEvent e) {

// TODO Auto-generated method stub

}

@Override

public void mouseDragged(MouseEvent e) {

// TODO Auto-generated method stub

if (flag < 5 && flag >= 0) {

double x = e.getX();

double y = e.getY();

ps[flag] = new Point2D.Double(x, y);

Ellipse2D.Double current = new Ellipse2D.Double(x - SIDELENGTH / 2, y - SIDELENGTH / 2, SIDELENGTH,

SIDELENGTH);

ellipse[flag] = current;

repaint();

}

}

@Override

public void mouseMoved(MouseEvent e) {

// TODO Auto-generated method stub

}

}

效果图:

e9230b8178762e78a9ebca47629fe1f8.gif

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/552114.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java连接access_关于k8s下使用Ingress保持长连接的异常情况排查

写在前面的话应某位友人需求&#xff0c;帮整理下工作中的发生的一些值得记录的文章。于是在友人描述后&#xff0c;为其整理为了文章&#xff0c;供大家一起参考探讨。问题描述在我们中应用有一个使用到Http Long Poll的场景&#xff0c;它需要一个http请求保持最长30秒&#…

python重复命令_Python简化重复命令教案

教材分析&#xff1a; 本节主要内容是 Python 程序设计的第三节&#xff0c; 也是这一单元的重点和难点&#xff0c; 是在掌握循序结构程序的基础上真正接 触结构化程序设计&#xff0c;能够理解并掌握循环语句的含义&#xff0c;其他语句就很容易理解了。 学情分析&#xff1a…

java调用另一个类的变量_调用 另一个类(同包/不同包)中的方法,成员方法,成员变量...

调用 另一个类中的1, 方法(方法区别于成员方法的标志就是有无static)homework05.Test52.randomNumber(100,20,8); //直接调用即可, 包名.类名.方法名();同一个包中可不写包名.1 packagehomework06;23 importhomework05.Test52;45 importjava.util.Arrays;67 public classTes…

罗斯蒙特电磁流量计8723说明书_罗斯蒙特8732E电磁流量计对环境和温度的限制

今天我们来说说美国罗斯蒙特8732E电磁流量计对环境和温度的限制&#xff01;工作温度-40 到 60C(-40 到 140F)&#xff0c;无本地操作界面-20 到 60C(-4 到 140F)&#xff0c;有本地操作界面当温度低于 -20C 时&#xff0c;本地操作界面 (LOI) 将无显示储存-40 到 85C(-40 到 1…

python字符串定义符_python入门——定义字符串

坚持每天更新&#xff0c;帮助入门python。kali linux 小伙伴们&#xff0c;大家好&#xff0c;今下午我们一起学习在python中定义字符串。 那么什么是字符串呢&#xff1f;字符串或串(String)是由数字、字母、下划线组成的一串字符。说白了&#xff0c;就是一堆字符。 在pytho…

java asynchronize_Java 中synchronize函数的实例详解

Java 中synchronize函数的实例详解java中的一个类的成员函数若用synchronized来修饰&#xff0c;则对应同一个对象&#xff0c;多个线程像调用这个对象的这个同步函数时必须等到上一个线程调用完才能由下一个线程调用。那么如果一个类同时有两个成员函数是由synchronized修饰如…

python递归和循环的区别_递归与伪递归区别,Python 实现递归与尾递归

递归函数在函数内部&#xff0c;可以调用其他函数。如果一个函数在内部调用自身本身&#xff0c;这个函数就是递归函数。(1) 递归就是在过程或函数里调用自身。(2) 在使用递归策略时&#xff0c;必须有一个明确的递归结束条件&#xff0c;称为递归出口。递归一般用于解决三类问…

java基础排序_Java排序算法-Java入门|Java基础课程

1、 课程目标排序是任何语言都会使用到的功能之一&#xff0c;然成果排序的算法有很多&#xff0c;对空间的要求及其时间效率也不尽相同。本文章以Java语言示例&#xff0c;通过对空间要求、时间效率要求&#xff0c;来对比各种排序算法的使用场景2、适用对象Java语言初学者Jav…

python逐行打印_python中逐行打印

方法一&#xff1a;readline函数 f open("./code.txt")      #返回一个文件对象 line f.readline() #调用文件的 readline()方法 whileline: print(line, end ) #在 Python 3中使用 line f.readline() f.close() 方法二&#xff1a;一次读取多行数据 with ope…

多个python脚本同时执行_Python实现脚本锁功能(同时只能执行一个脚本)

1. 文件锁脚本启动前检查特定文件是否存在&#xff0c;不存在就启动并新建文件&#xff0c;脚本结束后删掉特定文件。通过文件的判断来确定脚本是否正在执行。方法实现也比较简单&#xff0c;这里以python脚本为例#codingutf-8## 文件锁脚本测试#import os #操作系统import tim…

耳机不分主从是什么意思_开学必备高性价蓝牙耳机,学生党时尚配件推荐

耳机自从手机出世之后就一直伴随着我们&#xff0c;作为手机的最佳搭档被我们使用&#xff0c;像现如今流行的蓝牙耳机我们就经常使用&#xff0c;大学生也是差不多每天都用得着&#xff0c;听歌、散步、玩游戏、看剧等哪都看得到它的身影&#xff0c;当然蓝牙耳机价格也有高低…

11g java 驱动_Oracle 11g Java驱动包ojdbc6.jar安装到maven库,并查看jar具体版本号

ojdbc6.jar下载Oracle官方宣布的Oracle数据库11g的驱动jar包是ojdbc6.jarojdbc6.jar下载地址&#xff1a;https://www.oracle.com/technetwork/database/enterprise-edition/jdbc-112010-090769.html (Oracle Database 11g Release 2 (11.2.0.4) JDBC Drivers & UCP Downlo…

功放音量调节原理_玩汽车音响,功放和喇叭,应该如何做好匹配?

原标题&#xff1a;玩汽车音响&#xff0c;功放和喇叭&#xff0c;应该如何做好匹配&#xff1f;功放和喇叭搭配使用&#xff0c;离不开合理匹配&#xff0c;那么如何做好两者匹配呢&#xff1f;功放和喇叭要做到三匹配&#xff1a;阻抗匹配、功率匹配和工作频率匹配。只有这样…

python淘宝抢购_Python 实现毫秒级淘宝抢购脚本的示例代码

本篇文章主要介绍了Python 通过selenium实现毫秒级自动抢购的示例代码&#xff0c;通过扫码登录即可自动完成一系列操作&#xff0c;抢购时间精确至毫秒&#xff0c;可抢加购物车等待时间结算的&#xff0c;也可以抢聚划算的商品。 博主不提供任何服务器端程序&#xff0c;也不…

java seekbar_SeekBar的基本使用方法

a)什么是SeekBarb)使用SeekBar的步骤:i.在布局文件当中声明SeekBar: ii.定义一个OnSeekBarChangeListener: private class SeekBarListener implements SeekBar.OnSeekBarChangeListener{public void onProgressChanged(SeekBar seekBar,int progress,Boolean fromUser){System…

线程中如何使用对象_在 Flink 算子中使用多线程如何保证不丢数据?

简介&#xff1a; 本人通过分析痛点、同步批量请求优化为异步请求、多线程 Client 模式、Flink 算子内多线程实现以及总结四部分帮助大家理解 Flink 中使用多线程的优化及在 Flink 算子中使用多线程如何保证不丢数据。分析痛点笔者线上有一个 Flink 任务消费 Kafka 数据&#x…

用python绘制好看的图形_怎么用Python画出好看的词云图?

相信很多人在第一眼看到下面这些图时&#xff0c;都会被其牛逼的视觉效应所吸引&#xff0c;这篇文章就教大家怎么用Python画出这种图。 前期准备 上面的这种图叫做词云图&#xff0c;主要用途是将文本数据中出现频率较高的关键词以可视化的形式展现出来&#xff0c;使人一眼就…

linux回到桌面的命令符_三 基本的base shell 命令

1、启动shell进入 /etc/passwd 看到知道默认的是base shell ctrlaltt 或者双击终端&#xff0c;就进入如下图界面&#xff1b;如果你还是CLI新手&#xff0c;请记住&#xff0c;在输入shell命令之后&#xff0c;需要按回车键才能让shell执行你输入的命令。2、bash 手册man 命令…

java单纯形法_单纯形法 - fjzzq2002 - 博客园

看了集训队答辩&#xff0c;感觉要学习的有杜教筛高级版、线性规划、FFT、仙人掌、高级版线段树不出意外的话一个月内博客内都不会有别的东西了QAQ首先是喜闻乐见的单纯形法解线性规划。今年(2016年)和线性规划有关的集训队论文有两篇&#xff0c;大家可以自行翻一下集训队论文…

python调用js获取异步返回的数据_Python怎么获取js动态加载的数据

展开全部 import selenium from selenium import webdriver from selenium.common.exceptions import NoSuchElementException from selenium.webdriver.common.keys import Keys import time browser webdriver.Firefox() # Get local session of firefox browser.get("…