本帖最后由 hello怪 于 2017-7-25 22:24 编辑 % `, E* U5 E' N; x0 Y
5 c3 Q* h3 L" x' ~(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html . A# a: Z; k6 B
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面: , a% x9 }! Y0 \* T( P
场地* K! R5 j* [6 H* L
水平度和光滑度 在家里当然瓷砖的效果是最好的% T% X, o, z4 |$ {1 j$ ` s
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
4 q0 }0 r. v& W% A4 i1 X: M" F/ q! V$ p7 R# P
车子结构
) f3 I' ]7 i/ u8 M: j0 R& m 轮子间距$ w( p5 y8 g: L$ |( F* z
轮子间距越大越好,稍微想一下便知
& F8 I/ F$ u- m3 U9 @' ` G# Y+ N 轮子大小. e4 M; ?0 q/ O, g- k6 t
轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子)
" I1 p& J# z+ O. a& M# v: B 随动轮的设计9 @7 U2 k9 U: f, F- E. s9 E
随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响3 f/ c4 F- R/ g6 \
重心位置4 C4 G/ c8 W5 G
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小
: E. E# Z% W* ^ 结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
6 x; G7 V; R; } 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
+ a+ ]# G, _- S 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动1 f- U$ X' H4 K2 l$ z& i/ W1 f
+ n+ H n8 t6 Q1 A6 F3 F+ \) d
控制程序6 f) o4 P$ V4 g" ^
运动过程控制
3 e8 O& Y& l- A/ m; J! q 好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)% j% Z9 R+ ^( i
X! I* L) }! t% P& e调试与运行
1 Y& \# w! s; j' U1 [- e 轮子转速调整(车轮大小的设定)
9 X. U. Z0 V8 Y% A 这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
& Q/ {3 B# X% k8 J1 j 起始位置车子正方向和直线间的夹角. I6 h0 E- J) Z# |3 Q" W
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
/ o+ K4 C- x: `5 @, {# q1 D+ q) Z* @, c& V1 G
1 i2 s% q" T+ Y3 n, x- c
然后上源代码:
0 W' K9 a. }( Z( l1 U, n. D# f* ?- importlejos.robotics.navigation.DifferentialPilot;
6 N6 e9 u/ p1 H& h; e - import lejos.util.Delay;4 D8 {- b% b: |$ X1 J) G
- ) ]; S5 f0 r7 w$ V9 f) K4 D
- /**
. \! c1 x- ^+ y5 `: F4 T; I$ k7 M0 K6 W - *Robot that stops if the buttons are presses before it completes its travel.
2 _: C3 Y7 m8 d1 L( r# J% D - */$ N) S# h" y6 [# C* J
- 1 J8 K( e5 E. O+ ^# U9 x* J
- L' b8 b: y3 R& E: Z* v
- public class Straight {- i8 f, u% X" V2 ?9 c9 Y
- DifferentialPilotpilot;2 Q# [9 s8 k% j/ f. G+ f1 ^
- doubleleftD = 4.3, rightD = 4.3;6 r' y5 {. M+ r, z
4 w, H% m+ f# \+ B. X* e: w5 A- publicvoid adjust() {
4 a) E9 K8 i/ {( O3 z) F1 Q* q; q - //wait for release
6 S9 F2 e9 u" b, s( C, }5 d - while(Button.readButtons() > 0)
) H9 T4 ^. n* d& } - ;8 X3 C: f9 A: i0 e
- while(Button.readButtons() == 0) {
9 T- e' V% U s4 j1 Q; h2 T5 y& e - //input size by adjusting the wheels( w {- e7 U0 y5 v& r
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;# i' M4 c4 F, e5 u
- leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值
3 |7 E2 d, Z' s: L
' W+ @1 x' o7 e, i$ E9 p- //display the size input$ E3 W; V: b0 ]2 |( ^ K
- LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);
) x) @1 c+ o8 [% r$ o- S% i - LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);8 W' s: |2 a" k
- Delay.msDelay(100);
% _7 w" k6 o- x9 [: g - LCD.clear();
3 o& v) Z/ b8 w5 o! n4 x3 c* U' u - }
% J0 @8 x# Q2 @7 S! R- u - 7 U( c4 f+ V5 n1 a
- }6 w. |) V+ X0 @/ M6 D, \* g; j
- ; v9 r* f5 c: r" ~- B8 m7 j
- publicvoid go() {
/ m: ^- R; w, z, S8 [7 Y$ G - //准备起步
1 k! f8 `5 O$ W, \0 s - //wait for release
9 [0 ?! _/ c, X; T1 X$ [' _ - while(Button.readButtons() > 0)
& E' y) }* E# i* _# m - ;
& z+ S( u- K4 H1 h - System.out.println("pressto go");
. W& t) m K5 _5 L. @, w5 W - O' w2 f/ J3 p) T5 E) \2 E
- Button.waitForAnyPress();
- ?8 e7 _7 c2 M9 Z - LCD.clear(); g- z1 l' f5 O
/ c( r0 |6 f/ v! `7 s& K3 `2 a' p3 H- //初始化4 E( s$ T9 l& `
- pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
2 S0 j- D* v4 Q+ E3 N - pilot.reset();
8 d# ?5 O" W( ~3 D
( U+ O+ h% N5 g( n$ t+ H! G- Delay.msDelay(3000);
! f' M3 q6 `7 ?& p6 ]" x2 ] - . l# D4 U4 k i
- doublespeed = 10;
' h' r+ C, u3 A, {6 C, \1 v - doubledistance = 0;1 a9 w/ }/ c$ ]
) W9 w6 L4 P1 [( I2 U- //起步加速
# m; c2 ^7 x# S$ N* ? - pilot.travel(300,true);# m5 z. Z5 N x: s) J
- " g8 q! r. n& e. s7 t7 f
- while(distance < 20) {
, U8 H# f) R: ]0 J0 b. Y - distance= pilot.getMovement().getDistanceTraveled();5 E6 A. J0 @: |5 ~; H) z
- speed= distance * 5 + 10;) ]1 x1 U5 z% b& [+ T
- pilot.setRotateSpeed(speed);' r* y3 i2 K- g* j6 i* d! h' G! T
- Delay.msDelay(100);* D* ^2 M+ O1 Q9 ~6 q. l: R4 Y
- }
* m1 l9 Z) r" n3 l* d6 k; s7 w) r - for(int i = 0; i < 10; i++) {
4 U9 F1 s& ^* E) |. k' u" R - speed+= 5;; @* E) P; w5 c/ x9 ?* A+ p" _$ i
- pilot.setRotateSpeed(speed);
2 e7 D7 R7 o; @& U5 S - Delay.msDelay(200);
' B8 }! D5 b, d, S" n* e - }/ d; j5 H* \# `3 s' [
! C. c* |, n) K h6 E5 T+ s/ C- //TODO 减速
Z! h4 j. k5 g
& s( c2 ~; r# T9 c2 O, L- //停止机能
& R8 Y D1 P n" Z/ t - while(pilot.isMoving()) {* h9 V7 h6 y1 K0 E
- if(Button.readButtons() > 0)
O- O) @! Z6 a6 ]- U) e - pilot.stop();5 ?: |& e2 H- B: r
- }
8 G7 P- w: q3 [
# U" W" t# |; L: p) p5 M- z \- System.out.println("" + pilot.getMovement().getDistanceTraveled());! J: t3 P" d7 e1 _8 F
- System.out.println(-Motor.A.getTachoCount());
1 O: K$ f0 r" f$ U7 r- d O6 | - System.out.println(-Motor.B.getTachoCount());. r! e& u( t8 _* @' c
- Button.waitForAnyPress();& G& f% W5 Q6 U7 }% R: o
- }' _* ^4 c5 Z' n5 F1 T4 A3 C7 I; [
/ n* U% J, b$ T% K# d5 B Q- publicstatic void main(String[] args) {2 D1 W" {, g6 k0 D$ S, y O& S8 R
- Straighttraveler = new Straight();; x9 k( Z: y! k6 U! j
- 0 `& F' U8 C2 ?0 ~% [
- traveler.adjust();
! k6 S; j% U! o: ]; B# |$ D* I - traveler.go();
4 m! Z: g" o+ u+ I( U5 C - }! P" D0 j2 U/ x
- }
复制代码 3 V, [0 [4 ?. B8 x0 y0 f
|