: ?# D7 w" s' j/ W2 t$ g' L f如果只是这样的话,那和爱心1.0没有多大区别,所以我打算让它动起来,画出不同的颜色. i+ Q' u' o+ _$ X5 D: Q8 I
+ Y, _9 m' E1 C* q实现要点' I/ E* d. J$ F4 T( R
$ ]- q7 H" W6 q( S# h怎么画出图形: 使用java的图形化界面,通过画笔画很多连续的矩形。使用到的是重写的public void paintComponent(Graphics g)方法。 要使该方法能够使用,就要在一个JPanel类中重写,并且把这个类添加到一个JFrame类中,之后在里面使用g.fillRect()方法画矩形就好了- N' Q2 L% u3 @# u# A. H# a# J
怎么修改画出矩形的颜色: 在使用g.fillRect()方法画图之前可以先通过g.setColor()方法设置颜色,在此方法中要放入一个Color对象作为参数,那么就可以通过RGB三原色设置要画的颜色,之后不断改变RBG数值就可以达到不断改变颜色的效果 : e" y6 {* h' h+ Q怎么画爱心: 根据上面说的公式来判断当前坐标的值是否在爱心外或爱心内来判断该点是否要画,如果要画的话就调用repaint()方法,会直接执行paintComponent(Graphics g)方法# r! y& T Q. \. B( v
怎么实现动画效果: 最开始使用的是时间事件,每隔一定时间就画一行。看似很理想但是会有一个问题,如果画一行的操作是连续的,那么在这一行中是需要循环判断每一个点是否要画,如果要画就使用repaint(),否则就继续循环。结果画出来的只会有一个点。原因是Java有一个GUI (AWT) Thread来负责GUI事件的分发,这个线程一般是和主程序的线程是绑定的,如果当前线程休眠那么画图的这个事件也会丢失。如果循环调用repaint()的话就会合并为最后一个repaint(),所以永远只会画出一个点。那么就需要使用到多线程,为面板创建一个新的线程,每次调用repaint()时就让面板线程休眠一段时间,那么就会轮到GUI (AWT) Thread,就可以执行repaint(),从而就不会丢失事件。既然使用了线程休眠,那么就不需要再额外增加一个时间事件了,只需要通过线程休眠达到时间间隔的效果就可以了* n& F: v+ {' g1 X& I% S& K0 d
完整代码7 ` c5 }% ?& a$ P/ j% q7 q" S7 q1 {
" j* {) j9 }+ f- y
爱心1.0 ! Y; k* Z P9 g; C N% C7 V# u# }( _8 y# C
, P9 Q( Z. l# r( C- S2 I: d' R" Y
#include<stdio.h> 3 {# C! `( O0 b, \+ V3 s* g- ?#include<iostream> 9 _4 f% V$ K/ g d* Husing namespace std ; ( V4 N! N$ a0 |int main() { 3 u- K! P" ?+ P8 B# X# `) N. O for(float y = 1.5f; y >= -1.5f; y -= 0.1f){ % i: [. N6 q) A" H( I for(float x = -1.5f; x <= 1.5f; x += 0.05f){; ?4 q! T1 {2 J5 P- C4 e
float a = x*x + y*y - 1;# y/ i; \- V. ^1 g
if(a*a*a - x*x*y*y*y <= 0.0f) 8 ]: z( x+ ?' b" Y9 D/ l: C printf("*");# G4 D: q! O* u1 {; ?
else' a9 k3 O( L- V5 J
printf(" ");4 J! w' w& |% e+ l, D
} " o( n- }- w% D$ Y. ?; N% T printf("\n");# e0 V! ~* E' K$ i9 g; ]
}9 |. O9 J2 H" b5 G7 G
return 0 ;" x7 G; ^; K+ A A2 w6 q, X& U
} $ o6 c. ?) U$ U/ z6 Z 3 R: } E& V3 {/ @2 V- M# R爱心2.0 - ]7 T4 R8 } n- xpublic class Main {" U6 u- l t3 v8 x( R/ N
public static void main(String[] args) { ; K7 z: s+ X) J8 O9 Q# k Window window = new Window() ; ' G8 K3 f) [( i8 t9 k } % E3 S8 L8 l( D" J}; `* M' W" |( Y& `
- d3 s; h f A- p. S1 [) u* a - m" z' D0 {; F' x/ [public class Window extends JFrame { 2 l6 Q4 _( Z# B) L7 V5 o) b* ? Panel panel;/ h+ u) s. t# N; ]3 f0 p5 i4 w
; w& w4 d8 f( X# W7 ?0 d, L5 r
public Window() { ; i5 {1 A0 s+ M7 a! n( J panel = new Panel() ;8 P. q7 U+ q4 _2 O0 X
add(panel) ; //只有将画板添加到窗口才能画图% U* i8 m: a; x& K5 |$ B
panel.setBounds(0, 0, 860, 750); # l2 `; \1 w) l Z; ]* I+ c Thread t = new Thread(panel) ; //要使用线程才能实现动画效果 9 M$ i/ A2 s5 `$ ?+ K( F t.start();9 t1 D f$ q* g8 L- d X
setLayout(null); //画板要能调节大小,则窗口不能使用默认排版方式 * J" ^! c' } P) Y1 H setBounds(400, 50, 860, 750); * t5 \% Z* U! N v& Y) f8 f setVisible(true);# E* m" X- ^) Q9 h6 v3 Z
validate();5 l0 q; G! _! t+ @9 K: q! _. R
setDefaultCloseOperation(Window.EXIT_ON_CLOSE);( R' H% ?1 W, u3 i4 p' b9 k
}4 A4 s) _5 ?6 s9 g0 t# I
}, ~! E, `6 B: g( m
" I; k* J% i9 \/ _3 L9 Y. D/ ^
1 Y& E7 ]( e0 w ^public class Panel extends JPanel implements Runnable {, C0 s) l: D1 R7 D1 P' M
int R ; //三原色red" V1 [. E; d$ t5 w6 f% ]8 T9 C
int G ; //三原色green % U/ \: z T4 ], P& Q/ S* ~ int B ; //三原色blue* P7 C( {) M9 R. B
int tx; //画图坐标 z! W7 e! ]3 ~, J int ty ; //画图坐标 ) C- ?; a' C: ]) L& Y& m( O* K float y ; //循环画图行数. r0 V- j- K; w& `7 \
boolean flag ; //画边框爱心还是实体爱心 2 L0 h; X: a/ z$ c* H, [5 k0 K boolean increaseOrDecrease = false ; //G、B增大或减小 ! B# c1 z2 p8 X; y boolean backRed = false ; //从黑色变回红色 5 \ i6 J+ V2 P: B# h5 g- i File file = new File("my lonely soul.wav") ; //背景音乐0 H! t) e7 A8 d
URL url = null;: O7 y- t9 q& R
URI uri = null ; : o. p1 G7 a2 \ u AudioClip clip = null; / ~3 j m! Z( U x% Q
public Panel(){ 4 n5 U" ?8 K& T7 L; M3 J try {$ ^" `# H+ N/ ]7 T; O2 p' ^3 M
uri=file.toURI(); 9 M: r5 N$ c [8 I* e8 I url = uri.toURL() ;, F- q- F# @% s9 z$ I5 K
} 7 c! f' L J! R& N! m, j) P catch (MalformedURLException e1) {}* \3 R3 O- D" c& v& m8 ?
clip= Applet.newAudioClip(url); 6 E9 `" i9 l* x4 a# _, } clip.loop(); //播放背景音乐 8 o6 e* S6 Y P; J4 I y R = 255 ; //初始三原色为红色" D+ ?3 s4 s3 V; V
G = 0 ;" z% ^7 \0 L) B3 h$ l3 \1 X! y" g
B = 0 ;0 a0 Q0 U; F7 u
y = 1.5f ; //初始循环位置8 b( A) F& R$ j0 j* M4 F
tx = 30 ; //每一行画图的位置7 L6 P8 A, p2 f4 K' J
ty = 10 ; //初始画图的列的位置! K6 {7 F; f* X
flag = false ; //最开始画边框爱心3 [6 C8 }" q- a4 K$ Q2 Q
setVisible(true); 0 w! s6 S$ V4 D5 j/ Z } 5 j3 ]; A8 ?' s" ]4 _- A- m public void paintComponent(Graphics g) { . d* p* ?5 D. W: S" l& r if(!flag) { //画边框' @3 e+ ?3 B& K( A& w+ g4 o5 n C' O
Color color = new Color(R,G,B) ; //根据当前的RGB画相应颜色的图形 1 ?- a/ c) p+ E$ { g.setColor(color); $ ]/ K2 U& ]* W: [' q g.fillRect(tx, ty, 13, 13);. m+ Z! G% V1 B2 }" @
g.fillRect(tx, ty+11, 13, 10); //多往下画一点减小每行的间隔. H, F) @ @8 ~8 G0 G2 D: k
}1 l6 H* J) T- g
else { //画实体爱心 , P# w' l( u# G" j) H* Z super.paintComponent(g); //将之前所有的边框先清空! q! K! l* v/ J, J0 H! T, C
Color color = new Color(206, 40, 34) ; //最终的颜色% ^. m- u+ a& g1 Y; J
g.setColor(color); & d- F! _; j( G9 O8 s$ o for(float i = 1.5f; i >= -1.5f; i -= 0.1f){ 9 C' I [1 ?! m1 \ h for(float x = -1.5f; x <= 1.5f; x += 0.05f){5 l2 P9 R3 n- @8 F0 Y
float a = x*x + i*i - 1; ; s4 l7 X/ A. B2 J9 p3 R) u8 J if(a*a*a - x*x*i*i*i <= 0.0f) {( O E3 K' p6 V) ?" w* ~$ _( f
g.fillRect(tx, ty, 13, 13); 9 | _! V6 M. D3 u g.fillRect(tx, ty+11, 13, 16);8 h* T6 x! f, M F6 R/ m) N
} 6 G1 O5 g; T0 E tx += 13 ; 8 f0 O( V7 z" _" ? }, x1 X$ c6 p% {1 D' u- e
tx = 30 ; - l5 r7 F @" j5 N" l ty += 25 ; : Q7 K2 G" H2 `+ F: j0 l }6 Y! _: R& [8 }( S. r8 l9 o" q
} / k5 J2 Q+ `: ] } 0 M9 H7 o, i3 Q6 g public void run() { % V/ }9 p2 t/ q: p, Q O while (true) { $ L4 }; L6 g2 l- L8 C2 L try {. Y2 s: G: f/ [: |; ?
Thread.sleep(70);" I& P1 H: h0 F- M
} 5 ]+ N9 u" _6 [( T$ n/ }( W8 v catch(Exception e) {}0 L; d3 U1 h. e n2 A
if(y>=-1.2f && !flag) { //画边框爱心 3 t. n7 g7 e; _) @ //根据公式(x^2+y^2-1)^3-x^2y^3=0画3 W- ?% k v% H4 {; E( x7 L/ A$ n
for(float x = -1.5f; x <= 1.5f; x += 0.05f) { ! u/ s4 Q- E: a4 ` H float a = x*x + y*y - 1; 3 N$ D1 T( w7 S4 M. b$ J, I0 L if(a*a*a-x*x*y*y*y>0.0f && y!=1.5f) { //大于0是爱心外侧,小于0是内侧4 b; L% d6 J5 M# \
this.repaint(); ' Y# K6 u& p- K# E$ l try { //要把线程休眠一会才会轮到repaint()的线程 ; {( A2 D: ? ^+ r8 K Thread.sleep(4); //可自定义事件,事件越小画的速度越快但太小的话可能会漏画 ) h9 ~) X: w* h' Q0 [. d4 { } 8 z* V, d1 L6 z catch(InterruptedException e){} E- S5 _: c8 E
} f9 K8 x! M0 ~! P. R% R3 e' C5 | tx += 13 ; //每行往右走一点1 @& c- P8 n" i3 x5 K' R* Q- V( y
}9 @' d: \( F# r7 ]
tx = 30 ; //画完一行后要回到最左边 9 `, I) Y& j# l0 Q ty += 25 ; //画下一行7 o* C+ r# D$ I: Y+ z" M
y -= 0.1f ; //循环次数减少6 L! K9 k" I$ K& U1 |8 R; Q
} 5 H7 A! E, y' c0 b: H3 q else { //画完一个边框后判断继续画下一个边框或画实体+ n4 [8 \2 Q- e
if(!increaseOrDecrease) { //G、B数值增加2 l; p, r1 l2 a% e8 [) [; _
G += 4 ; " L! U) X+ r0 \5 d1 V) g( O- m B += 4 ; 3 s( L3 V% o- d# t8 e, g/ v$ E5 u } / B: q& V" |3 g' L1 S. F$ z else { //G、B数值减小* F4 r% x& u$ G/ y+ _2 w
G -= 4 ; ' V3 G4 s4 A3 W) t- d. e" V B -= 4 ; ' m9 A( Q9 G6 c7 f. ~ } - j4 ` z8 n3 E4 |; W. K% A if(G >= 70) //G、B数值在0~70范围内* \4 f# Q8 o) x# a1 H/ R! l
increaseOrDecrease = true ;8 I1 d& ?0 ]; C% k) p
if(!backRed) //红变黑0 H& }1 J3 Y$ k2 h) t' V. C4 s
R -= 10 ; ) }0 j+ R6 {6 X1 L
else //黑变红 % y' M3 w& G; K' ]3 I# z R += 10 ;- t; }% r, ^1 I& B
if(R <= 0) { //到黑色了准备变回红色0 v2 C }& j. k5 Z; S( E7 R
R = 1 ; //重新初始化R - Q( F( X8 D* } P" r9 c backRed = true ; ' z' d) T# y' ~4 q; c: [5 k } , o2 N! i: ^& o2 O9 r! |4 u; b if(G<=0 && B<=0) {5 {2 A" R/ e. g4 _: a5 q
G = 1 ; 2 y$ E5 e1 u& V: j8 b B = 1 ;9 p8 o+ [/ P. m; h) m& ?6 y
backRed = true ;: O9 K9 T! w# j2 t
}$ P6 P+ Y) D% h7 {* i3 Z
if(R < 255) //R没有再次变回255说明还在画边框 $ D$ l" N. C) l7 F: o- b3 J y = 1.5f ;. T; Q5 s6 B+ h# O$ W( `. o
else { //画实体爱心- W1 {! l$ B8 ?# g4 l, x, }
flag = true ;/ k; d! O) M3 Z* p
try { : n* M6 q6 ~* y, c; S# @ Thread.sleep(500); ) C5 l; I8 Q2 L" m u" p: f }% s3 j6 l. D- y! ?* q
catch(Exception e) {}+ n) H( S, K. T
this.repaint();6 n! T d9 ~" r6 n" J. D9 ?
}! _2 G- q: c7 q; Z) I' [- u* ^) p
tx = 30 ; //每画完一次都要重新初始化画图坐标! b- i! u* f6 O: k
ty = 10 ;, D, [& s- J4 }0 `# t X
} 7 ^- I9 B5 {! ? }/ q1 M& I' s# t, q' s5 L& n
}3 F, p- t2 B) ?# _" @6 F
} P- d/ ?$ J+ U) P1 C( h. N7 Y/ @
1 S' V3 N# o" |/ M8 }' T) R
总结4 {% f, `7 Z: ]% G
0 e' v/ X7 _/ C5 [# a
这是继圣诞树之后第二个突发奇想做的东西,还是挺有意思的,也能学到一些新的东西,比如多线程实现循环repaint()。以后还会有更多突发奇想的东西做出来吧. Q2 U; T$ S+ k6 d$ _* y* o7 ?
做出来的只是一个模板,如果对你有用,完全可以在这个基础上实现更多有趣浪漫的操作。不准觉得这个表白程序太直男!把“硬核”打在公屏上!7 @: u) k- K1 l/ U$ ^
5 K0 B- R" S X) y1 l* W- e1 _( a7 P! e' w5 Q- f$ O
2 C8 o4 E! L! b C+ s" X6 T4 q& L5 m& Y8 }
j$ Y, a1 I8 a, g6 s版权声明:本文为CSDN博主「打代码的小明」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。. F, Y i2 F, I. j. H3 H
原文链接:https://blog.csdn.net/weixin_44689154/article/details/106172622 ; _' f1 G4 T; N ( M! j) r2 x6 y; ~9 Z' v7 c9 d ^/ \ 4 m0 A- A+ L; W5 [; K0 ? I9 [- K% @& X0 n9 c. w