本帖最后由 hello怪 于 2017-7-25 22:24 编辑 $ Z( G3 w r" R# |' X- A
& j6 _* [4 h, w- X8 X+ W(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html 8 `0 t. `. Z: o: T4 A4 j
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面: R4 e5 P& \+ n- x7 V( v3 f
场地
+ J6 _) q [! q/ e, n& \) d' {3 a 水平度和光滑度 在家里当然瓷砖的效果是最好的; y: a' c, K/ o, a9 ?
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
\- ~: i! j3 ?5 `, ]2 n- x. q" D J# K1 t/ ^3 v" B" v' ^
车子结构8 t- e) X* z/ @: G+ t
轮子间距 w X6 j% `' N
轮子间距越大越好,稍微想一下便知
3 I+ e h& W( j 轮子大小1 t$ o3 d" A" s1 b* y; p3 q/ W
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)% P+ N9 `+ e* p2 S8 `- g# T5 L0 |
随动轮的设计
3 g* u/ ^" N/ K* W4 {8 D 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响4 a4 M- ~; x' `/ F" C/ U
重心位置
, I+ P1 a# D$ n2 ]# { 重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
$ ?& M% M- K, y4 a: B 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大% o5 @0 A3 X& o& `* R* W
轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
' D E1 ?2 x7 i7 \4 Y& X 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动. `; w9 E5 _/ ^2 `9 L* s6 A, _# m- B
; P' v1 w' O( ]
控制程序
% Q% z" u" ^0 f; ?5 @ 运动过程控制% l& s2 c0 r7 J# S5 c
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零), U/ b, d+ |% X7 K0 d2 A; L8 K7 s$ x
4 D. K4 V6 F, D9 k$ o4 H4 v
调试与运行
2 N4 S. B. U6 \ G# p6 f8 U. L1 R 轮子转速调整(车轮大小的设定)
* w4 O% ^3 M! [8 ~- a 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
% b% I8 s+ c! K4 z! \! b 起始位置车子正方向和直线间的夹角. J" T- c7 C+ O$ u: |
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
/ i, p# x- j3 t/ s# T( j- ?; P1 {+ w3 E& `- s, y; [# ?
; L' \$ A& \$ u: r0 T: p
然后上源代码:
" _/ H1 e5 A" G1 J) T0 d- importlejos.robotics.navigation.DifferentialPilot;6 k1 p, K( z" t: w* r6 B/ J
- import lejos.util.Delay;
$ S- l2 a% ^- r/ `8 k2 v I - % g5 c. ^$ ~: r# N; p; u+ O
- /**, B1 Q# p: w" U8 |' p0 G% ~
- *Robot that stops if the buttons are presses before it completes its travel.
' n9 K9 l/ ~# W8 ^9 d8 o$ x" o - */" A5 j1 z* D# @) E0 Q
+ [9 r" [4 a2 u0 Q0 x- 4 O, k) E( u, a+ d* w
- public class Straight {: D m1 u/ Q) v
- DifferentialPilotpilot;8 h$ l1 h* c) A+ B/ `
- doubleleftD = 4.3, rightD = 4.3;
4 E% [1 v4 y4 D
- @) t: W) r& |- w$ ^8 n3 d- publicvoid adjust() {
. s& m# t: i K- g Q8 R1 w0 q - //wait for release
8 h/ U, k- p: X/ B8 H- s - while(Button.readButtons() > 0)' p ~4 z( C; i, a' d/ y
- ;
; J x7 x0 S/ k. t - while(Button.readButtons() == 0) {4 E) r( D& v% T
- //input size by adjusting the wheels
1 U$ r/ m1 v1 ^& X+ c" w - rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
" ~) u$ ?. S! z" X! g# r E$ r - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值8 S* W+ j$ f/ O8 S% o2 @8 j) S
- 0 A. i- o, j5 G* j
- //display the size input3 A4 h) y# a) ]+ ]* y# q' C& d( Z
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);! {) d5 _1 n/ r+ s4 ]. x5 N
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);
* [ R2 W9 C2 C% X - Delay.msDelay(100);
* C6 y) O7 g5 Y: D; ~' F6 y) A - LCD.clear();) Y( i9 P5 o6 s1 j- H/ W5 r
- }
. t) h( e) g& p9 \3 D - / e! D9 _ ]( C( x
- }
$ J( B( M: B8 X7 f7 s8 f
& R) W2 `1 o! A ?& k0 r6 A3 Y' Z- publicvoid go() {
4 x- `3 @/ \! D$ K9 b1 V- [1 N" j - //准备起步
# E: C% q3 j" l% o6 k2 w - //wait for release
( o/ R% H) R F# A! C1 `! G5 k+ M - while(Button.readButtons() > 0)3 w0 [) n) X* u# i2 m" ^3 }
- ;
5 i0 z; C$ d; |6 b# Z - System.out.println("pressto go");& B2 o' }) |* x$ r" _( N
- . ` p' K# t8 R$ s" ~- `* E
- Button.waitForAnyPress();5 K+ [8 m5 t b2 c
- LCD.clear();
; n! {& T8 P- k8 K3 \7 E
. T2 h6 e% Q5 A+ e- //初始化
6 A7 V) ~2 G4 o7 J7 p - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
4 G) P4 Z; e& q; M* [8 j - pilot.reset();* A* k% U0 U: P- F+ f7 l8 N
" p* P @- g: w' N4 K- Delay.msDelay(3000);
. t" ^) d9 {1 I( u P1 p0 l6 B8 t2 U
4 f2 Q- Q3 o% A6 S- doublespeed = 10;
8 _' `& n9 g' [3 c. {+ m% b3 {5 P; p - doubledistance = 0;
& d, s$ Z" h, a% a
6 M) y6 w# T! A5 J/ F- //起步加速
6 p: |2 C) ~% w - pilot.travel(300,true);
9 f; D5 a4 H; {) N - o9 F1 @7 I* G! O
- while(distance < 20) {
m, [9 S# N$ q& W1 H7 X* J# [1 R( @ - distance= pilot.getMovement().getDistanceTraveled();; B, I0 c) U! m6 A" Q5 s
- speed= distance * 5 + 10;" ^6 o$ r2 G( [6 e+ j6 M
- pilot.setRotateSpeed(speed);
x. x1 U5 g# A - Delay.msDelay(100);
1 T5 X* P1 ?( y: T$ Q* e& ~- p- { - }
e2 C6 }) ~- e6 i9 o - for(int i = 0; i < 10; i++) {; y& t) \4 @6 s/ D
- speed+= 5;5 C0 P3 p; G; g0 V8 o
- pilot.setRotateSpeed(speed);2 {2 B% b* O6 c& r0 E0 W* ]4 J
- Delay.msDelay(200);6 S* k4 r# ?2 ]7 I
- }0 s" O) O% U/ \. _ W' w; z
- ; d4 H+ P# X' i, {4 v9 k
- //TODO 减速
) S$ d) ~! \9 t( S2 O- E" Y - , {& J7 _3 a+ ~4 F6 r& e7 t
- //停止机能, J3 C1 t. t' V0 r0 ]
- while(pilot.isMoving()) {+ C; ?+ |, n) X* [4 ^
- if(Button.readButtons() > 0)3 l% i6 S* [% a5 I( J
- pilot.stop();
1 M4 N: Z: M1 B/ d0 Q8 l! N - }
. u- n1 H. X1 e# T8 y# _ - 3 d& s* W* K6 `% r6 L: h# w
- System.out.println("" + pilot.getMovement().getDistanceTraveled());
9 K# `9 ~. L1 s" A0 R, \4 S8 ]( C - System.out.println(-Motor.A.getTachoCount());
, x* V+ K) g# _4 e6 I1 R3 I; R! I - System.out.println(-Motor.B.getTachoCount());' z3 X% d* a3 z$ o f% T& {
- Button.waitForAnyPress();8 Y- k% Y4 M; s1 T6 ?5 ]# T2 B
- }
! o8 a0 z% E! O* h# b8 U
. ~/ V! m$ U# A, D- publicstatic void main(String[] args) {
& ^3 C2 E4 j: `4 Z! l - Straighttraveler = new Straight();
3 L/ s. B6 Y0 k3 }; u - 7 z" H" i( S% |2 A
- traveler.adjust();4 D$ u7 Z, r2 P4 h8 z) P2 D" U
- traveler.go();' I' p: i" p8 L( k/ W) Y; [
- }
p* c l/ c' f- R2 i8 Y - }
复制代码
: H8 X. o5 Z! e! H1 h+ n6 d |