本帖最后由 hello怪 于 2017-7-25 22:24 编辑
$ {( ]+ `5 @) [# N5 O7 w7 q* q$ c; n, A7 L7 T# ]& K+ j
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html
3 y j! X* X8 \% w, m
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面: 9 j" Y1 a2 h$ L0 H1 H
场地
V3 b1 a; f+ } 水平度和光滑度 在家里当然瓷砖的效果是最好的! f/ b5 E* U8 ?9 ~" j$ b, P
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直7 `: d% k* n. q$ N2 M& `
3 ?. x. N6 q5 e车子结构 V& m0 m+ G0 m0 b/ c
轮子间距
F+ `, s( B+ ^+ z+ ?: r& Q1 m% i3 } 轮子间距越大越好,稍微想一下便知
+ s( g& H4 `7 ]5 x6 h 轮子大小
! j% M. h, \# x$ O6 { 轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)3 S6 g; V4 V G, E$ U% h
随动轮的设计
* E f. S$ b4 T& ~2 J9 A 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响1 Z, U( b" V$ v' F% ]. A/ T1 v
重心位置5 [+ J5 J; E8 L `
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
4 D7 S: P4 t7 @ 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大4 G9 z1 h' a) Y! p I0 s
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
. p' s6 I, c8 h5 C7 J: C8 I: }8 v 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动) m# T, z" a2 s. B* U2 J; x$ U
( o6 N, R3 q' s
控制程序/ k9 p1 V: M* b
运动过程控制
5 A& H n% \+ ?: A- Z. Q7 I 好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)* _8 w) G3 @7 c, ?2 v2 B3 a
% ]+ E# ^' \/ @" U2 u/ }
调试与运行
# J3 _! u0 [$ k' Y1 ~3 B 轮子转速调整(车轮大小的设定)
; y# W4 W6 y4 O/ J# n 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法! `% l8 ^. [& U/ ]
起始位置车子正方向和直线间的夹角 b! z: }& E" ?2 `
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点 8 {3 p. Q# s7 P8 Z+ t
4 {- s1 X. H j+ ?
. W# T* J0 H! i1 a 然后上源代码:
! r l; Y0 X. E: u6 t- importlejos.robotics.navigation.DifferentialPilot;
0 }, E) \4 h' _! Y! T/ g( d - import lejos.util.Delay;
6 E, E5 Y- U# f" N( R0 t. E3 e2 T4 d
/ W7 I" c, Y& j. B3 q- /**
5 e8 Z1 p2 _5 j - *Robot that stops if the buttons are presses before it completes its travel.
4 u, {) Y5 V# R/ b - */
* `5 s K% k6 z4 W6 V6 t! W - |- Z' {' Y. l& y8 @: g
, O ? `1 h/ c! z" }) m- n1 y- public class Straight {
8 \/ e8 c% Y) K- [4 | - DifferentialPilotpilot;
+ f' y% ^; R2 e - doubleleftD = 4.3, rightD = 4.3;
0 L# ~: \% z$ r. Y3 z% W: k/ i
/ s$ ?. [, J- m! n- P8 B& A- publicvoid adjust() {
3 E1 W* e9 R% w+ W3 m7 | - //wait for release' L! {' m/ n' I" O6 \5 g* J
- while(Button.readButtons() > 0)
7 ~' w8 B* q) J3 G - ;
1 W0 u$ ^' h$ R, U, {# s - while(Button.readButtons() == 0) {
# p, K3 B1 S/ [ - //input size by adjusting the wheels; Y o+ K6 }1 h; J* J F( p5 h
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
+ j9 t9 B r4 Y4 m3 [7 X - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
" | j( N( y2 d# ? y' c( h - 9 K% x0 L& A5 W3 G
- //display the size input
, e0 C7 l- q u - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
/ A8 R h& ]/ s8 X R/ D; A - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);$ k3 r- z3 C/ H4 k- m
- Delay.msDelay(100);
! {9 B! K$ m4 L( u* l* n - LCD.clear();
, k U7 p; S' I+ x% ] - }$ \: W8 j0 k, \! V0 c8 Y) @3 k2 x3 h
$ H+ Q/ [0 ^9 j4 m- }* x/ u# k1 l! L7 _% e/ ]
{5 H. N' X/ F: C- publicvoid go() {$ q) n. e; Y. R+ u! B) S) u- V
- //准备起步; w) E# N% g, k8 [7 ] \
- //wait for release
' @) [& W1 o2 k. Q: Z0 [; O2 V4 C& ~ - while(Button.readButtons() > 0)- R4 c" w; o! s* o- f/ K0 Z
- ;
, P8 A3 M, G: G( X8 {! Z - System.out.println("pressto go");3 P. Q5 [1 E, e( u
- ! a( }% P6 V1 R) F# H! @! \$ a
- Button.waitForAnyPress();
! M( r$ _/ f/ J - LCD.clear();
5 g9 K( N/ U! G( j! G
% y; ?- ?: B+ j0 s' y# I- //初始化( _7 A: ]1 V7 N, j4 @3 j0 I+ k
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
6 W# W* z9 t! J - pilot.reset();) m2 t$ r h' j4 e7 P+ |" @" h: s
% q1 I5 L2 h* C- Delay.msDelay(3000);
6 p/ U" E$ ]% O) T) T- A4 X - , \/ n6 ?/ Z; ]2 u
- doublespeed = 10;9 s6 x. G* i/ q9 `
- doubledistance = 0;+ }) s, u D6 {8 P3 `! g+ T2 r
- @& i- E4 e0 W- //起步加速
+ k* G* Z% D2 L8 y% a - pilot.travel(300,true);7 b: u0 f3 Q3 f6 K! U5 ` m
. N% ]6 }2 e3 {! b q7 b2 [- while(distance < 20) {
3 v* e+ l4 y& q - distance= pilot.getMovement().getDistanceTraveled();6 F" D( J" v3 V8 G C4 J# t% U) U
- speed= distance * 5 + 10;
# p6 N. ]) o0 Q9 f2 y - pilot.setRotateSpeed(speed);" q0 ~$ V# {" C
- Delay.msDelay(100);7 c7 n# Q" c, l: A7 A4 A; u
- }
2 a6 w. Q, e: T9 l( `/ F - for(int i = 0; i < 10; i++) {7 C# L9 z- t0 {& j
- speed+= 5;* M8 g: Y: \4 u6 [& G8 g9 D
- pilot.setRotateSpeed(speed);! n) l P/ j" e! x# Z* D7 p" z
- Delay.msDelay(200);
: N# O! q- o! s4 Y* p0 B - }1 w; K1 \2 z$ S. ~% c
- : Z5 g1 d5 ]( u* r
- //TODO 减速/ s. F' A6 w1 I, ~
- + m1 \, m. P( S, i- V
- //停止机能
/ h1 D& ?. \/ ~. O; D0 F- j3 y - while(pilot.isMoving()) {& J8 X: o) t6 y' A
- if(Button.readButtons() > 0)7 l8 m+ _8 t- p. ? x5 t B: Y
- pilot.stop();
# h: t* J' C9 }& \5 D - }2 E0 \/ q; X% |. ]: d& K
# M' L/ L2 X( \0 ?$ @+ y9 P1 z5 Q5 B- System.out.println("" + pilot.getMovement().getDistanceTraveled());
6 v4 u h9 i( q7 h# K - System.out.println(-Motor.A.getTachoCount());) `- H3 ?3 e3 t b) L3 J8 o( _6 S
- System.out.println(-Motor.B.getTachoCount());
1 k5 O' Y# _% m; k9 m- ~+ ^ - Button.waitForAnyPress();5 ?$ p! O( ^* V: E) B; N
- }# o. e; T- \/ l8 K2 Z; w, |) V& S
* l; l" q4 f1 s' O5 C- N- publicstatic void main(String[] args) {
" X2 h+ e I! ~& _ - Straighttraveler = new Straight();$ w+ H2 O0 L' m* J4 e
- 4 T; \7 C8 P, P. d' {7 U5 m) k9 m8 S5 u
- traveler.adjust();
% B1 S& _, `5 G. m8 p5 h( t9 P - traveler.go();; p2 O; ~8 X) d- l
- }
/ G) ?3 e" t. Z - }
复制代码
5 Z6 F% s$ P6 C$ Z/ b |