本帖最后由 hello怪 于 2017-7-25 22:24 编辑
/ `* Y" V$ {& M3 D8 c
9 l# _7 c4 w s(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html : i( f9 h/ d5 ?0 Y/ ^
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
6 p* Y, Q+ U4 ?% x( }+ f2 M场地5 E5 j0 Q# u5 p- I( j8 p
水平度和光滑度
在家里当然瓷砖的效果是最好的% C) N, J5 I8 ^/ Y
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直5 M' p4 y, ?( l. S( b) o
! b- O5 ]* B- z% \7 `# A% P; n: R车子结构, |3 p/ V0 V. @ K% X& J* O( q
轮子间距
/ L P! P3 Z8 `5 J3 j 轮子间距越大越好,稍微想一下便知
& B* y* B0 V; }9 T$ P1 e 轮子大小9 g' N6 P" M m% T; L
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)
: @( q1 e# [; m2 k 随动轮的设计
' X1 m& F; m+ ^) C" g, @% k 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
' R" _8 A% N- L 重心位置 a* |5 }. @; Y
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
6 g& T. S7 }2 D/ L+ n% a 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大* v; i% m% U/ g. g0 i5 L2 n- J, R! h
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)4 ^0 A% `, t; \* t4 M M
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动# C5 n7 |+ X; j% O1 k
# K6 T9 F( o; ?" Z' ]: s控制程序
3 `& B& C; K+ n D- R 运动过程控制' _( R+ a2 O( b# x+ ]# |
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
3 X3 W3 S) ]2 _7 x9 U/ }% g: H9 s. L
调试与运行* c; s* M; t* J5 K3 @" E
轮子转速调整(车轮大小的设定)
2 O1 y5 w( d$ e1 o8 B1 v+ m 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
' x: A) T" c) `$ a5 E 起始位置车子正方向和直线间的夹角
4 r7 D: ?, x0 e: |9 g 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
B" D0 [, S- X
' \: e0 D1 @0 k* E( x- C7 M) k u; N3 W! s( m6 a! M
然后上源代码:
1 H4 E7 i* }9 t- importlejos.robotics.navigation.DifferentialPilot;- _# c8 D& Q7 J
- import lejos.util.Delay;
& \8 s& I! u6 N - # J8 h0 [" ?: ^4 a. |/ g8 \
- /**
2 C( S2 d( e' L2 S5 f" B - *Robot that stops if the buttons are presses before it completes its travel.
& x9 }7 `& K9 `: }, a - */
$ u8 L4 a" J" @( P: t' R* k0 G, m
- h0 w8 F& k2 G) Q9 r
: C8 L" T7 ] O" e' i5 z9 ~9 p6 i# q. P- public class Straight {2 L1 B( R7 M# U
- DifferentialPilotpilot;
4 t7 A9 O+ E+ A2 w3 _" Z! g/ I - doubleleftD = 4.3, rightD = 4.3;* w) A5 a, w" S
2 ^, \1 E0 `; a, H- publicvoid adjust() {" D2 \% S+ G) r: {
- //wait for release
( ]+ O5 h& y, R: ]! U - while(Button.readButtons() > 0)
; i/ \3 W+ t3 L3 M - ;
0 ?" G+ u3 v$ [" L9 A/ p$ `3 W: L - while(Button.readButtons() == 0) {/ T: J" C& J5 k4 ]2 R- u
- //input size by adjusting the wheels7 Y! d# _, J' T L" C4 A
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
1 q# y7 p P* R - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
) c+ u6 c& u) e5 Y
) ], I ~, ]% G* l8 K9 U- //display the size input
: W! `, `% T! M0 T9 u - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
8 q, X1 L" x" d: t& H6 a9 f/ z! B - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);; U Y! [) @; g# g3 H; q
- Delay.msDelay(100);
" z) K$ E' B8 w" H2 i: c - LCD.clear();
" l' u- L1 X' K6 _& a/ T; J% p - }, T/ t3 P9 g& ?+ ^
- 0 s6 {2 H! S( v+ L R! g& r* V9 [. A
- }
' f+ z* ?: H5 ]4 [' C - $ K" o: r0 T4 W/ N/ q" L
- publicvoid go() {
" B; v q0 i I# P' b6 M - //准备起步
7 @) W- I5 G+ q - //wait for release2 o! q* e4 ], S: |' {1 _, ], q
- while(Button.readButtons() > 0) z' r: n! e+ {4 _
- ;
8 l# v$ R- g1 Y0 R2 W - System.out.println("pressto go");5 J( _0 B6 y8 L3 U# p" ^
- ' P) N: T. M# ` b
- Button.waitForAnyPress();
+ I# k! ]! L( ~" T! R) b& w5 ] - LCD.clear();
+ O# o( K- l% r, }( |0 n; J$ Z
$ @ Y( ?3 Q6 M9 R; p/ d% R2 y- //初始化* H# l4 @) I4 O. r- {9 E, n
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);4 A$ H r! B/ z4 `' L8 P/ X* P4 M
- pilot.reset();
T! c y C/ s - X. U8 ~. x" n# ]2 h
- Delay.msDelay(3000);
4 i, S( T& P) r g$ V - 4 F1 u" P+ P" P/ G% w+ O
- doublespeed = 10;
$ q# \1 f5 _! g. U# W! n - doubledistance = 0;
. R' y8 F7 J7 K9 d V% ?: z - 9 I: _8 j8 j9 ?6 `' f9 `% d+ }
- //起步加速
@" }) T/ n* u0 W! [ - pilot.travel(300,true);. {. t; v$ i/ X' p$ y0 n
. K0 h) V' ^. K' Q- while(distance < 20) {, H# ~3 q. `. ]* C& x
- distance= pilot.getMovement().getDistanceTraveled();
- Z$ ^8 s `4 h3 ` - speed= distance * 5 + 10;
5 L* Q6 C" j" c* C% h1 ]- r: I - pilot.setRotateSpeed(speed);
, A; c, E! r) B: N: @) J - Delay.msDelay(100);
7 u: S0 V+ u2 K; u9 u& W - }
4 `, K0 K2 x+ a - for(int i = 0; i < 10; i++) {5 |* v& T3 \! I& L
- speed+= 5;+ `! H6 q: F2 f* M0 e: E
- pilot.setRotateSpeed(speed);1 B& t- U7 U! q3 J/ A
- Delay.msDelay(200);6 l1 R( Y. |8 u9 G& D2 `* E
- }
& k, x0 w8 ~# L' P% c
8 s' L# s0 a7 h a- //TODO 减速 C9 g! d: I: D
% o% _ Q1 @3 }9 z$ o4 Q& ?- //停止机能( i; M, {" |/ t* F* ?
- while(pilot.isMoving()) {0 ?, A( h* Y/ o# Z* c% E6 a
- if(Button.readButtons() > 0)
$ [. L0 Z5 }9 Z" f U0 L B, U' w - pilot.stop();
( T- S( a9 P$ l( f - }
* p3 [' o. k7 ^/ m - # S" P7 B; o% c+ z9 l
- System.out.println("" + pilot.getMovement().getDistanceTraveled());; T" U7 I- O. n7 D! Q
- System.out.println(-Motor.A.getTachoCount());
% I* P' }) ~* u2 e. i1 J$ @ - System.out.println(-Motor.B.getTachoCount());6 ?2 | Q' {' A. I3 X1 x
- Button.waitForAnyPress();6 T% t+ ^* P# V c9 q8 s! F; H
- }
, R' ]& g- }& [1 W# A) n; a - 4 U ?7 O: g# L- d/ l: B6 n/ x3 d
- publicstatic void main(String[] args) {. h) D) W2 w. ~( E9 T
- Straighttraveler = new Straight();
c) d5 I3 q$ P- I" C, p$ A) W - . |* x2 @$ Y) X3 z* J, J
- traveler.adjust();( ]" t P9 ~3 J% ]
- traveler.go();# x2 O& ]) J' ?
- }
Z4 Y& G( x! y. `7 {$ R5 n6 c6 C - }
复制代码
9 P! `8 Z+ F' Z! ^8 E! [ |