java画山_Java第n次入门之递归山脉的实现

2024-03-03 03:59

本文主要是介绍java画山_Java第n次入门之递归山脉的实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一,递归山脉的要求

给定左右两个点X1(Lx,Ly),X2(Rx,Ry),一个y轴动态范围-range~range,在该动态范围内随机选取一个数num,选取一个中点M,中点的横坐标为(Lx+Rx)/2,纵坐标为(Ly+Ry)/2+num,连接左端点与中点、中点与右端点。如此反复,再分别取左端点X1和中点M的中点、中点M和右端点X2的中点,range范围按一定比例缩小,连接两点形成递归山脉。

620cd5b5aceb073c456b7daf952a2e3b.png

二,创新点

之前我们调用递归的时候每循环一次都调用一次,后面的结果覆盖前面的结果,形成最后的效果,这造成了之前的画的一些图的冗余。在本次项目中,我们采用不一样的思想,在循环部分只做计算,当最终条件满足时再画图,这样就是最后每一小段之间连接,不会造成小段覆盖大段的冗余。

三,实现过程

(1)创建界面,绑定监听

package com.yzd1223.RecurMountain;

import java.awt.FlowLayout;

import java.awt.Graphics;

import javax.swing.JFrame;

public class RecurMountain {

public void ShowUI() {

JFrame jf = new JFrame("MyPad");

jf.setSize(800, 600);//画板宽800 高600

jf.setLocationRelativeTo(null);

jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

FlowLayout flayout = new FlowLayout();//设定流式布局

jf.setLayout(flayout);

jf.setVisible(true);//实现窗体可视化

DrawListener dlistener = new DrawListener();

jf.addMouseListener(dlistener);//界面注册鼠标监听器

Graphics g = jf.getGraphics();//得到窗体画笔

dlistener.g=g;//将窗体画笔赋给监听画笔

}

public static void main(String[] args) {//主函数

RecurMountain Rmountain = new RecurMountain();

Rmountain.ShowUI();

}

}

(2)鼠标释放时画出递归山脉

package com.yzd1223.RecurMountain;

import java.awt.Color;

import java.awt.Graphics;

import java.awt.Polygon;

import java.awt.event.MouseEvent;

import java.awt.event.MouseListener;

import java.util.Random;

public class DrawListener implements MouseListener{

Graphics g = null;

int Lx,Ly,Rx,Ry;

int range;

double rate;

@Override //鼠标点击

public void mouseClicked(MouseEvent e) {

}

@Override //鼠标按下

public void mousePressed(MouseEvent e) {

}

@Override //鼠标释放

public void mouseReleased(MouseEvent e) {

Lx=0;Ly=300;Rx=800;Ry=300;//初始左、右两端点坐标

range=150;//生成-range~range的动态取值空间

rate=0.5;//range缩小比例

MyRecurMountain(Lx,Ly,Rx,Ry,range,rate);//调用方法,画递归山脉

}

@Override //鼠标进入

public void mouseEntered(MouseEvent e) {

}

@Override //鼠标退出

public void mouseExited(MouseEvent e) {

}

//自定义画递归山脉图方法

public void MyRecurMountain(int Lx,int Ly,int Rx,int Ry,int range,double rate) {

if(Math.abs(Rx-Lx)<1 | range==0) {

g.drawLine(Lx, Ly, Rx, Ry);

Polygon pon = new Polygon();//利用多边形给画的山脉填充颜色 顺时针和逆时针可以 本次采用顺时针

pon.addPoint(Lx, Ly);

pon.addPoint(Rx, Ry);

pon.addPoint(Rx, 600);

pon.addPoint(Lx, 600);

g.setColor(new Color(0,150,30,20));//设置颜色

g.fillPolygon(pon);//填充

}else {//只做计算

int Mx=(Lx+Rx)/2;//中点坐标

int My=(Ly+Ry)/2;

Random rand = new Random();

int num=rand.nextInt(range*2)-range;//随机生成-150~150的动态范围

range = (int)(range*rate);//range范围不断缩小

MyRecurMountain(Lx,Ly,Mx,My+num,range,rate);//与左端点递归

MyRecurMountain(Mx,My+num,Rx,Ry,range,rate);//与右端点递归

}

}

}

在该段代码中我们在else部分中对坐标进行计算,随机生成num,并按rate比例缩小range,然后调用自己MyRecurMountain,直到满足条件Math.abs(Rx-Lx)<1 | range==0,执行连线g.drawLine(Lx, Ly, Rx, Ry)。

在这里我们还对图像进行了填充,创建一个Polygon对象pon,将连线的两点以及他们对应x坐标位于屏幕底部的点连接,形成一个封闭图像,对该封闭图形进行连接填充颜色。

Polygon pon = new Polygon();//利用多边形给画的山脉填充颜色 顺时针和逆时针可以 本次采用顺时针

pon.addPoint(Lx, Ly);//左端点

pon.addPoint(Rx, Ry);//右端点

pon.addPoint(Rx, 600);//右端点屏幕底部点

pon.addPoint(Lx, 600);//左端点屏幕底部点

g.setColor(new Color(0,150,30,20));//设置颜色

g.fillPolygon(pon);//填充

形成的效果如图:

ededddd4782f034a129579c05fbeec79.png

四,加缓冲提高画图速度

在之前的程序执行过程中,我们发现画图很慢,于是我们想改进画图速度。

BufferedImage是Image的一个子类,BufferedImage的主要作用就是将一副图片加载到内存中。BufferedImage生成的图片在内存里有一个图像缓冲区,利用这个缓冲区我们可以很方便的操作这个图片,通常用来做图片修改操作如大小变换、图片变灰、设置图片透明或不透明等,并且实现速度很快。

public void mouseReleased(MouseEvent e) {

Lx=0;Ly=300;Rx=800;Ry=300;

range=150;//生成-range~range的动态取值空间

rate=0.5;//range缩小比例

//创建缓冲图片 大小和窗体一致 类型为RGB

BufferedImage bufferedimage = new BufferedImage(800, 600, BufferedImage.TYPE_3BYTE_BGR);

//得到缓存图片的画笔

Graphics gr=bufferedimage.getGraphics();

//将缓存图片的画笔一起传入递归山脉画图的方法中

//这样在下一步将缓存图片显示的同时就能将递归山脉一起画出 提高画图速度

MyRecurMountain(Lx,Ly,Rx,Ry,gr,range,rate);

//在画板上将缓存图片显示出来

g.drawImage(bufferedimage, 0, 0,800,600,null);

}

我们在MouseReleased中创建一个和窗体大小一样的RGB类型的bufferedimage对象,得到该对象的画笔gr,将该画笔作为画递归山脉的画笔传入MyRecurMountain()方法中,最后将bufferedimage图像显示出来,这样在显示缓冲图像的同时由于画笔gr传入了递归山脉方法中,递归山脉也能同时画出,大大提高了画图速度,效果如下:

8951be7b7d7099bc15deec64bd7a5e4c.png

本文地址:https://blog.csdn.net/weixin_43722843/article/details/111990719

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

这篇关于java画山_Java第n次入门之递归山脉的实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

基于Redisson实现分布式系统下的接口限流

《基于Redisson实现分布式系统下的接口限流》在高并发场景下,接口限流是保障系统稳定性的重要手段,本文将介绍利用Redisson结合Redis实现分布式环境下的接口限流,具有一定的参考价值,感兴趣... 目录分布式限流的核心挑战基于 Redisson 的分布式限流设计思路实现步骤引入依赖定义限流注解实现

SpringBoot实现虚拟线程的方案

《SpringBoot实现虚拟线程的方案》Java19引入虚拟线程,本文就来介绍一下SpringBoot实现虚拟线程的方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录什么是虚拟线程虚拟线程和普通线程的区别SpringBoot使用虚拟线程配置@Async性能对比H

基于Python实现进阶版PDF合并/拆分工具

《基于Python实现进阶版PDF合并/拆分工具》在数字化时代,PDF文件已成为日常工作和学习中不可或缺的一部分,本文将详细介绍一款简单易用的PDF工具,帮助用户轻松完成PDF文件的合并与拆分操作... 目录工具概述环境准备界面说明合并PDF文件拆分PDF文件高级技巧常见问题完整源代码总结在数字化时代,PD

javaSE类和对象进阶用法举例详解

《javaSE类和对象进阶用法举例详解》JavaSE的面向对象编程是软件开发中的基石,它通过类和对象的概念,实现了代码的模块化、可复用性和灵活性,:本文主要介绍javaSE类和对象进阶用法的相关资... 目录前言一、封装1.访问限定符2.包2.1包的概念2.2导入包2.3自定义包2.4常见的包二、stati

Python实现Word转PDF全攻略(从入门到实战)

《Python实现Word转PDF全攻略(从入门到实战)》在数字化办公场景中,Word文档的跨平台兼容性始终是个难题,而PDF格式凭借所见即所得的特性,已成为文档分发和归档的标准格式,下面小编就来和大... 目录一、为什么需要python处理Word转PDF?二、主流转换方案对比三、五套实战方案详解方案1:

SpringBoot结合Knife4j进行API分组授权管理配置详解

《SpringBoot结合Knife4j进行API分组授权管理配置详解》在现代的微服务架构中,API文档和授权管理是不可或缺的一部分,本文将介绍如何在SpringBoot应用中集成Knife4j,并进... 目录环境准备配置 Swagger配置 Swagger OpenAPI自定义 Swagger UI 底

解决hive启动时java.net.ConnectException:拒绝连接的问题

《解决hive启动时java.net.ConnectException:拒绝连接的问题》Hadoop集群连接被拒,需检查集群是否启动、关闭防火墙/SELinux、确认安全模式退出,若问题仍存,查看日志... 目录错误发生原因解决方式1.关闭防火墙2.关闭selinux3.启动集群4.检查集群是否正常启动5.

SpringBoot集成EasyExcel实现百万级别的数据导入导出实践指南

《SpringBoot集成EasyExcel实现百万级别的数据导入导出实践指南》本文将基于开源项目springboot-easyexcel-batch进行解析与扩展,手把手教大家如何在SpringBo... 目录项目结构概览核心依赖百万级导出实战场景核心代码效果百万级导入实战场景监听器和Service(核心

idea Maven Springboot多模块项目打包时90%的问题及解决方案

《ideaMavenSpringboot多模块项目打包时90%的问题及解决方案》:本文主要介绍ideaMavenSpringboot多模块项目打包时90%的问题及解决方案,具有很好的参考价值,... 目录1. 前言2. 问题3. 解决办法4. jar 包冲突总结1. 前言之所以写这篇文章是因为在使用Mav

C# async await 异步编程实现机制详解

《C#asyncawait异步编程实现机制详解》async/await是C#5.0引入的语法糖,它基于**状态机(StateMachine)**模式实现,将异步方法转换为编译器生成的状态机类,本... 目录一、async/await 异步编程实现机制1.1 核心概念1.2 编译器转换过程1.3 关键组件解析