本帖最后由 hello怪 于 2017-7-25 22:24 编辑
/ y+ r/ i9 t: C6 ]) c6 E$ n, L. ~% K$ ?
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html
# E2 }# ]* d; w4 f
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
6 g# R! U' o1 t( }" t场地
" Z& ?5 K5 y$ E* o1 v5 w& S 水平度和光滑度
在家里当然瓷砖的效果是最好的: ^2 M6 Q8 ~/ ?- R
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直% T- p: h+ ]2 H" ]8 H. D; {7 B
3 R) [2 a4 U2 |2 B P7 A9 z9 }8 h车子结构
/ B; H# Z2 y2 e1 ^+ T& w1 Y+ [* h 轮子间距
3 {5 K, j0 |3 v. U: _$ ^ 轮子间距越大越好,稍微想一下便知
# R' G2 o( V( q8 d- R 轮子大小4 Y. i7 Y7 n# {* c
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)
9 E( n4 P5 p# L3 D- V 随动轮的设计8 B0 q7 E; L# h. U7 t
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响8 m! s1 m" C$ \5 y# B
重心位置1 x! u( g! V* H; ?; f7 K v- s7 [
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小( l# }: d- w1 _% I$ v
结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
2 t- u+ f) x0 D, k" s3 k+ l! d7 e 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)) \* w1 V! J5 c; D6 i) O
驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动
; F" [8 `+ P3 W' x' J
4 q% X# ~6 {4 P( ]2 x+ _& U" y控制程序 c, ]9 ~7 B; ?% w
运动过程控制8 ~, p7 G+ y( u& N' a% ~7 C7 g
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)9 Y1 e* Z# c. p
( a1 P1 e: P; m; y
调试与运行1 k9 Y: P) I* C6 U# {
轮子转速调整(车轮大小的设定)
1 N. ^/ h1 N r) G4 @ \ 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法4 J, g: {8 K r# O, H! T2 {, N
起始位置车子正方向和直线间的夹角
8 ?+ J$ ^& P, C$ r9 ~6 s0 f 一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点 , z: g6 D. Y: [6 W) ^
* H7 o6 N( W/ l# O5 M
3 a: ~& I& S/ O: q5 I 然后上源代码: - 4 x! }$ M7 O8 q- b
- importlejos.robotics.navigation.DifferentialPilot;
# m( X- i- z, `$ ?" U0 L - import lejos.util.Delay;& [0 d. r L0 t& p" O( z
- / b! j* X A% J+ M6 d( |
- /**
( Z' O# _: i% W. _ - *Robot that stops if the buttons are presses before it completes its travel.& Y+ a! a. d- c- V5 j# F0 w1 L: l2 s
- */( C6 W- O5 N/ f- l5 Y7 B
- ! x/ L$ k5 e: X5 V
% X, ?9 s& ~4 h4 D; f1 K- public class Straight {3 u: b& d# \1 p& Y- W
- DifferentialPilotpilot;
/ g; u0 R' p& {8 g% a# }7 c - doubleleftD = 4.3, rightD = 4.3;1 ~, x0 j: G. n& K2 E
% C4 T6 l# y9 v- publicvoid adjust() {
/ c; a. f1 S8 `! J& @. e - //wait for release
5 P. C( c; ~ q - while(Button.readButtons() > 0)
* n1 E1 {% F3 w" Y: [* B - ;
2 @& r" k4 l/ N6 Q0 r - while(Button.readButtons() == 0) {* N# v5 i+ r# R g
- //input size by adjusting the wheels* @3 `, X2 Y) I {
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;$ ^ T- k- `) Z' I Z* J9 F
- leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
- b, \5 o# I0 f$ ]3 { - 1 z+ R p; L0 U* r
- //display the size input
N4 {, A" s9 \3 @; H - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
) A$ F8 _, L5 \8 D2 i - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);, | u) A' \0 f8 K1 z, R% X6 |
- Delay.msDelay(100);
) ^/ {( k- |: M" ~ - LCD.clear();
0 c$ Z& u$ P0 d: ~6 v8 U: x - }- G' Y& |! q1 n4 t* w
- 0 k2 V/ e2 h3 K5 P) S: O: g
- }
( s. K8 _5 f0 ?: W, c - ( T3 H0 J, y& K6 `4 R# `; }3 ?2 V2 X. ~
- publicvoid go() {: s* e. f# e" e
- //准备起步3 \. y I& W) z8 j" W
- //wait for release
! r8 s( ?# c: l% @0 {0 b$ C - while(Button.readButtons() > 0)
5 K( n, `! ?8 a8 S - ;% {" b) j. G- ~. @6 |& S
- System.out.println("pressto go");3 O/ a8 U; C$ Y1 G; z d
! S5 @) P: j& @ o: O M, O. U- Button.waitForAnyPress();
. \* z+ y% l- J W) G - LCD.clear();# n$ J1 V; p: P5 U9 R2 j! u, U
- & x7 [' a0 u. C% |8 Z* `4 c6 X
- //初始化* @+ j7 N* f* ?( o
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
; L- b0 }* l9 X1 [' _4 M - pilot.reset();
3 [0 \, E* s+ G, y( U2 `3 U, d - $ G( o2 L" D2 ^3 {1 p( K
- Delay.msDelay(3000);. _8 A# t J6 V4 ~4 K# d
5 O% S x% i) E: }& C: ]4 ]- doublespeed = 10;& q( ~9 H& ^" [- x( U
- doubledistance = 0;
* n2 W; o" e! m- n: N - 4 ]! @9 j, K a$ T; o! L+ e1 `
- //起步加速6 L; i6 g* K. P+ w! K
- pilot.travel(300,true);$ ^. M1 G: c3 V6 T
- 3 i) Z: m) F) y; h0 i( F5 s
- while(distance < 20) {
( l5 @5 T2 Y9 L/ L3 S9 m - distance= pilot.getMovement().getDistanceTraveled();
6 T: V4 d, R) z. B - speed= distance * 5 + 10;
2 m U3 O8 G& }: S8 Q& \/ u! b2 S - pilot.setRotateSpeed(speed);
% `0 O5 P1 E E& _! E) V6 q5 r - Delay.msDelay(100);
/ n# z) h2 _$ ?5 V% Q2 n9 Y# u7 K! X3 o - }
+ ?0 d( ~0 x9 ^% i - for(int i = 0; i < 10; i++) {$ [% S4 G4 [: X, ?/ g) N) r, b
- speed+= 5;. ^' \ @$ k* D; t6 s) J
- pilot.setRotateSpeed(speed);
2 A" e! S+ m) W1 m - Delay.msDelay(200);
) U1 R# Q3 C0 g9 t - }
: Z3 K6 `4 E, k) f
0 u$ v! Y6 ?# @4 l" @; u- //TODO 减速
) E9 q& P+ X& I7 F; N1 V! N% @: G - D% P7 d& e K) c" Z( x' Q$ Y& K
- //停止机能, E! B8 p9 H: L: U
- while(pilot.isMoving()) {
% R2 }) M+ H' G' q - if(Button.readButtons() > 0)5 A$ ]1 O* v) B( T Z
- pilot.stop();
, ~% p$ C& Q9 N% D - }/ k* c# q9 w5 ^
- # x* L% H, [' Y6 L& s
- System.out.println("" + pilot.getMovement().getDistanceTraveled());
1 J5 y5 }! i# `$ h" f4 X" y$ B8 e - System.out.println(-Motor.A.getTachoCount());7 R' _; o. w) f1 W# f
- System.out.println(-Motor.B.getTachoCount());
. G/ P% _9 F/ Y3 n( g1 K$ n% j - Button.waitForAnyPress();6 z; h4 @9 `% ~$ g' H# t
- }' e0 X2 S. d8 x$ N) I
- 2 W) @2 I6 a! F m3 y8 ^# t) ]
- publicstatic void main(String[] args) {) o, N1 b' J) F2 t. C
- Straighttraveler = new Straight();
4 n" l7 o3 d: P) c: b
5 w) ?. m$ m& F6 f; L$ J; c- traveler.adjust();
4 v {% T5 c, _* \ - traveler.go();" P3 K, |: R! {% E3 k
- }
! C! g H7 I: P" | - }
复制代码
0 y! H* H5 C* l4 \8 w. b |