本帖最后由 hello怪 于 2017-7-25 22:24 编辑 : c) b0 _( o* F; F; ^% A
% J7 L6 ^2 Z8 h/ J2 ^
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html
* x& z4 Q9 k) V8 @8 O1 \
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
$ [9 h4 M3 |4 N, Q场地
* I5 i3 W) ^" I o 水平度和光滑度
在家里当然瓷砖的效果是最好的0 [' A: K3 ]% I* A( f+ e0 p
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
9 L# \1 u7 m W1 e( ?7 K( i( Y2 J6 M2 ?# O- C4 z
车子结构
: q! p' K( P" |: k& h8 C$ h1 g/ N 轮子间距# A" T* C& U) k9 W
轮子间距越大越好,稍微想一下便知
8 p* ~% N1 p# U 轮子大小3 O$ _2 t; E+ D7 [2 U
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子). D0 c$ T" E2 Y' F U/ [7 S
随动轮的设计
4 a+ ^+ U9 w" h: C% G. k, Y 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响
2 T' g2 \5 z) f7 @3 h+ Z 重心位置
" @$ w8 }& k+ Y- S" c g 重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小$ d7 ^0 G9 E1 a) q1 `: C2 J
结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
& x7 V, L; p2 \) h 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
( _* A5 l, s# G 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
+ `9 u' v/ j2 T. F6 B x2 F' G, y" t/ ^
控制程序0 z+ a: ~7 Z# C* D5 p
运动过程控制' m0 O) p" K6 B0 a
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)9 r ~6 L, R; e2 P
4 b; K8 r" c! r- f. a2 x' w调试与运行
8 h6 s3 L# a; _ 轮子转速调整(车轮大小的设定), x5 v5 a: r# n; |* m C3 N0 b
这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
. _/ U6 z5 i% W1 @4 D5 { 起始位置车子正方向和直线间的夹角
& W, l, v6 C* i6 _7 ?) K 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
9 v2 J6 Q0 v; g3 U" Q6 \' h/ r6 v3 W
6 f- M0 q; v. O; B# K 然后上源代码:
% m& T! K* ?8 X0 `- a( |: V- importlejos.robotics.navigation.DifferentialPilot;$ g% _3 C( d% y; Q
- import lejos.util.Delay;' X5 {/ [9 V p- H& j5 J \
# H: U. {- W- T. Q' Y- /**( j9 t: g4 v; R" N3 Q- W" r
- *Robot that stops if the buttons are presses before it completes its travel.
. i Z8 T3 x5 L( @1 L - */* D7 z+ I% I6 |
. T- k' _# p9 `/ z- 2 ^! q9 R1 k( B/ H2 r
- public class Straight {, L V8 Y+ v$ \- s4 w
- DifferentialPilotpilot;, J$ o! W( I# P% S& s0 g6 | U
- doubleleftD = 4.3, rightD = 4.3;
+ p/ b2 `' A5 b2 v
( y! E# y/ f3 V& V& \4 e8 `0 j- publicvoid adjust() { ^! _: Z3 Y4 s: d0 c0 w p
- //wait for release' N4 }1 _( I: D& `
- while(Button.readButtons() > 0)6 Z( M; O# N, i7 \
- ;
2 _% N) m6 V; a1 {8 K& E - while(Button.readButtons() == 0) {- \' a0 I/ e0 j2 U# @6 |
- //input size by adjusting the wheels/ ]) l/ E/ G2 c
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
* ~+ k4 d$ ?* f# V5 f - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
7 n' p) @9 k- G5 a& A5 { - ( r: g' L. ]- N8 k
- //display the size input, ^) L7 P/ \5 s) a
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);4 ?! m# W, K) u* \$ @9 B" M
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);4 o9 Q6 D0 e. M
- Delay.msDelay(100);
( P! S7 s3 o1 p' ^1 ] - LCD.clear();/ s1 u9 i' l# _6 U
- }9 @" K1 M4 N. l2 H4 e$ u
- 6 R4 I! E, t7 \' E z' n
- }
! I% c8 R4 A$ `$ H0 [' A8 L - 3 d# \$ E w( V& [4 \, [$ t
- publicvoid go() {: s4 @( i. D$ s
- //准备起步
! H% P. R% w) b$ ]# |' \ - //wait for release
! ?5 G5 c' Y! E2 w9 B& n3 K8 O - while(Button.readButtons() > 0)" [) h7 {* a& L, z. ^' c' A" ?! l
- ;
: F+ u8 G8 S+ q, F+ G8 ]8 a" T" \ - System.out.println("pressto go");
0 `8 n$ Z+ t. M5 |
" `1 Z. a: q0 c$ W* `- e7 x8 z- Button.waitForAnyPress();2 U+ {7 c' ?6 y
- LCD.clear();" \! n$ g+ N7 K) |! w
- % [5 ~1 Z; S: {
- //初始化
. | J4 _ H" P2 n" L. S - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);7 j5 [; y# L2 p7 l& t
- pilot.reset();
7 z; V. e7 H7 Z& Q8 T+ k: \3 g
5 \( P5 d, o# u2 l- Delay.msDelay(3000);
& o0 K& _: E6 n, G; s3 ]
# Y* ]5 b3 t, F" }5 A3 G0 X2 Z+ b- doublespeed = 10;( H- w! @4 v0 f" \* G' d. u; s1 d; d
- doubledistance = 0;
2 x! P4 S9 M" ]! S3 ` q9 M9 i - / i, \8 x* _ `8 k0 g
- //起步加速
$ d/ Q- j$ c7 i8 h7 x0 x - pilot.travel(300,true);
. {0 m2 @/ T( j8 X& E1 P2 u& y0 E - , j$ @7 t7 t- h8 U' I1 f; O
- while(distance < 20) {
A9 `2 F t: o" A G/ R4 s - distance= pilot.getMovement().getDistanceTraveled();3 q/ y5 {+ ?2 h* K" u1 Y9 N
- speed= distance * 5 + 10;4 `8 V% H: B: i
- pilot.setRotateSpeed(speed);
" m+ {) K8 T! f! d1 {0 X! C2 U - Delay.msDelay(100);
l6 z- [! T& K1 s - }
0 H; L7 N* T) t3 r& } - for(int i = 0; i < 10; i++) {8 D& W# R! C/ E2 M1 |! j& o( U; A, V
- speed+= 5;
. M7 f3 h5 _. Y- f - pilot.setRotateSpeed(speed);: X. H7 ]( v6 |5 j0 w8 X( q
- Delay.msDelay(200);
/ z/ m! `1 d1 n/ b: e - }2 x) w' s% A2 a* u* z
7 l; ^' M2 {6 s/ p" s+ g- //TODO 减速" u7 C4 X# {* A
- ) Q, L8 `8 _) Z9 W. V7 f
- //停止机能( q+ ^+ h( A- J5 M9 n
- while(pilot.isMoving()) {% h+ K" S% O. ?9 Z" K
- if(Button.readButtons() > 0)
3 S! \0 H3 Y5 C, ^8 ]* r - pilot.stop();2 X7 t0 ^& R W; P. P
- }
" D6 h2 [: j, d% n - ( n* r/ ^1 j% Y* F5 e# o- h5 e7 H
- System.out.println("" + pilot.getMovement().getDistanceTraveled());
$ E; _! p. T' N# } - System.out.println(-Motor.A.getTachoCount());
7 t; r1 W: K# O* X7 i - System.out.println(-Motor.B.getTachoCount());
( i4 g. H$ l% @* a8 B, {! `& c - Button.waitForAnyPress();0 z8 c! ^! m( m" j
- }6 s* y' L5 z# d+ O6 u
- 9 F0 T; U& X& W0 ~) Q( p
- publicstatic void main(String[] args) {/ W& z& w/ K5 m; S
- Straighttraveler = new Straight();
# a& j! L* q: r - : [# c; [% d2 k; B9 S
- traveler.adjust();
7 v# P+ E5 a6 |* D/ @+ J - traveler.go();
5 d; ^( N" C; y# P - }) V" ?$ ]7 N& Y
- }
复制代码
1 r+ c7 _% ~& {- S1 m |