本帖最后由 hello怪 于 2017-7-25 22:24 编辑
8 D% |: i2 N1 `7 K( C) P1 O. f& m% A4 x2 ^% b7 a! f
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html 4 A0 R( u3 Y6 Z; Z( J# P) j
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
4 f) }3 z& A8 v6 X5 r1 n1 \3 ^场地
1 A: a F* k$ `4 f 水平度和光滑度
在家里当然瓷砖的效果是最好的
( P0 _9 |/ n m1 |& i0 q/ ~ 线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直. Q/ U9 q% \5 F. a
G" ^4 {. Y. ]5 V4 s' L车子结构
: s+ D3 }- m4 X 轮子间距
& J( p% H H" o) F" I' D5 Y 轮子间距越大越好,稍微想一下便知
. G" ]* b J/ W! m b9 d( ] 轮子大小( Y* g7 b. Q9 t7 x( g" X
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)+ [5 c# r: t5 j$ S2 `+ t
随动轮的设计
4 L4 k- u2 m8 R& s 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响5 y' b- ]" i7 i) T5 b
重心位置( {6 ^) u; M; p9 H9 \5 `3 g
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小4 s9 y4 G# I) D
结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大" Z- s% z4 S8 ^- F& A0 i4 k' U
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
% e$ P: d% U# U& [( I2 s 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
7 n: b ?) i# ^3 V4 k, A" r/ h7 H, [# f* C
控制程序. f9 Q' Y8 E9 [7 A6 W
运动过程控制1 w+ z% p/ d4 b! l, e! G! W0 R
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)2 N6 v y# k9 h9 S
% s( d8 H4 h4 r8 ]! R) x调试与运行9 S Z' |: i7 i' |$ }( ?4 o
轮子转速调整(车轮大小的设定)
: I' T3 h h! M( N! `; j8 j; C- Q 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法8 u% D- P x$ X5 d; k
起始位置车子正方向和直线间的夹角
8 L+ A" N# q& c2 U& U3 b( F2 r0 t6 | 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点 3 n: \4 `, k4 V# b: p8 w: \9 [, N
0 O0 c) W2 y2 b( I2 X2 E
e' G% R1 @4 H. F) ~" X 然后上源代码:
" B& {0 j1 f$ K- D9 C- importlejos.robotics.navigation.DifferentialPilot;, [2 h! ^. w* j; {" L) {9 U
- import lejos.util.Delay;7 o+ w b) ]. B1 B1 v2 O, d6 H
, w! R. O U5 s- /**
5 u, o! `% d1 H" E; x - *Robot that stops if the buttons are presses before it completes its travel.
* Y( |0 X" `/ q, a- J- f7 L- ~& \ - */
( X* B" B* Z. u, X! Z/ i - / z1 t. a3 D0 _) g
% [+ j0 K& G1 B. x' K1 S: V- public class Straight {" P7 u2 s$ {; `5 K2 S) t2 f
- DifferentialPilotpilot;! k, J/ C, |: u) F7 ]
- doubleleftD = 4.3, rightD = 4.3;4 |. G3 g# y$ {& W- D+ d! M% l
- ; u( @0 \$ M3 S- J0 p
- publicvoid adjust() {
' K; e. R/ i' X2 h: R - //wait for release
7 ]# s+ f5 R9 `# b2 _ - while(Button.readButtons() > 0)
! B* O5 h1 ^" J - ;
3 y5 T+ C+ o7 } ^ - while(Button.readButtons() == 0) {
3 \: k! }6 ^1 | - //input size by adjusting the wheels( c# ~7 O a4 t0 I- I
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
' D4 b/ g% d p' ` - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值" Q; Z+ j2 a3 }
- ) K% y8 i" r8 B
- //display the size input
$ O' ]; j9 l* K2 r2 O! w; p - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0); z( Y U3 V- U5 z
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);
6 W6 A9 l/ ~5 k$ }; H - Delay.msDelay(100);; N& h9 {! @6 r
- LCD.clear();% d% k6 p7 r9 b4 C4 d/ n
- }
+ }0 m5 P$ Y7 k7 f1 p/ L f; f& V - 1 c1 g4 z' A9 N; I9 M
- }
8 r" ^ C* @& w7 X2 v& C( G
- ~0 h1 |0 J/ E& |, c' c) C$ f- publicvoid go() {4 b! G$ S/ n2 l# @) R6 ~
- //准备起步; J' l4 O* ?% j0 W, r" v
- //wait for release9 D, }7 P9 `! R
- while(Button.readButtons() > 0)
6 j! b- \3 d5 M0 g+ f! t) s) j7 W - ;3 n* B `7 k4 `
- System.out.println("pressto go");
& C( m2 F" r; B) n& j - ; ]. N4 D7 v* `- O9 m0 S
- Button.waitForAnyPress();# P1 D2 G1 b) Q7 G( r
- LCD.clear();: y) T0 Q ?7 T- j9 E- e& z4 P- j8 d
- ) F& e6 [1 i: \1 O+ ^
- //初始化9 r) p. w6 h! M1 c
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);7 h& g, v0 ?# y
- pilot.reset();
$ v* ?2 i# {* W5 q9 J: G* B - 2 \1 W& a7 P. F& M8 s' b3 }
- Delay.msDelay(3000);# j- d3 w+ c9 B! j* q
- ; t ~5 O# P4 @: I) R8 O R
- doublespeed = 10;! n. S& E# I( `3 |% Q8 s
- doubledistance = 0;& D' T2 h# Q& L, b8 B
- / V4 @. z8 M# s- X# L) g
- //起步加速3 [2 u& k' M8 |: W, M* [7 s
- pilot.travel(300,true);/ D! P4 B! ^$ A2 F9 _) E( g2 L
- , x/ |0 r, q# i7 w+ O% L3 a
- while(distance < 20) {' Q F! I1 ~) A \6 Y# | z: v
- distance= pilot.getMovement().getDistanceTraveled();9 M5 v1 s6 e6 O& M, H1 Q
- speed= distance * 5 + 10;2 o3 o6 b& B9 Z( d9 L" U
- pilot.setRotateSpeed(speed);* t" E4 b( x1 a; |* P
- Delay.msDelay(100);7 _$ Q2 Z. H: l0 Z
- }( E b# U- l5 S1 w! x/ A5 \3 N6 m
- for(int i = 0; i < 10; i++) {9 }/ \* |% f6 \2 |- @+ O! f
- speed+= 5;
3 r0 p+ h1 ]7 \6 N: L8 W; ?. j - pilot.setRotateSpeed(speed);/ R9 j8 P( b; E( L
- Delay.msDelay(200);
6 w+ b% j @& U( H; R0 G - }: L7 N. b8 ]# Y& y$ I; u
' d: Z/ |- T; i0 e4 q/ i1 ?- //TODO 减速
4 w/ z/ y: Q1 W7 |# y - * u$ S4 H0 J: M" O T) X0 I' X
- //停止机能
" \/ u4 T# b L* m; H - while(pilot.isMoving()) {
5 b; f/ E, {) s8 P' u+ O* { - if(Button.readButtons() > 0)
- G* E" |$ Z2 k, j, E - pilot.stop();% j4 J4 K( X' D4 A6 x+ [
- }# Q/ l6 F" X4 G. f% N
- 6 [: w O# J" F; ?1 P2 B
- System.out.println("" + pilot.getMovement().getDistanceTraveled());$ q/ w5 v- H9 s* J, h$ \, Z
- System.out.println(-Motor.A.getTachoCount());9 k5 c4 j5 o7 d2 m8 \' |
- System.out.println(-Motor.B.getTachoCount());
8 Q+ Q( ~7 u, P; e" | - Button.waitForAnyPress();
) p2 T$ F! }) a5 O( s - }
. y- u7 j' P7 X8 ]
' e0 j) A" j% [, u4 |- publicstatic void main(String[] args) {9 T& y+ T; s& @
- Straighttraveler = new Straight();
& e8 z9 ~" A5 S - ( T# q: n- T1 @7 D1 d+ r# ^
- traveler.adjust();: ?" B. i2 _5 n
- traveler.go();" u& B# i8 G4 ]0 k" P! E4 U( j
- }6 @ C3 |6 u/ _# k7 h3 c9 i* j
- }
复制代码
1 c9 \& k8 m$ K( c6 D; C1 q. l+ k |