本帖最后由 hello怪 于 2017-7-25 22:24 编辑
# r+ v) [# U& b3 {" P" C! Y% H. \ f4 s' x0 |; `
(视频里面最终轮子与黑线的距离,都不超过6mm,最小距离只有2mm。虽然也有运气成分在里面,不过就结果来讲已经很令人满意了^_^)
优酷视频网址: http://v.youku.com/v_show/id_XMjkxOTU1MjEyNA==.html . j) i5 L4 h$ Y. X* @
首先说说车子的设计思路,希望也能对大家有用。我想到的影响车子走直线的因素有以下几个方面:
; V/ w. R3 B3 _/ C场地
) W7 h. ~- G9 y. T# O7 q 水平度和光滑度
在家里当然瓷砖的效果是最好的8 d: N, t' |$ X3 B3 z3 m
线的直度 虽然规则里说起点终点之间不能有黑线,但是因为没用光感,完全是开环控制,所以也无所谓了。这里就直接用了瓷砖上的黑线来说明车子走的直不直
0 v+ a: \9 r! P Y3 h3 \ m8 @9 M# f9 o9 r. U9 V3 {( a* T- Z
车子结构
* Y- }- `, T! b 轮子间距/ T. R( H% z. P% }& L4 O( n
轮子间距越大越好,稍微想一下便知; L$ V7 k. D1 f4 h- [
轮子大小
. r ? Z& I) N& C% g 轮子直径越小越好,而且轮皮要尽量薄(我这里用的是nxt自带的轮子) _" R4 n2 ^6 \
随动轮的设计
$ i& U" L* b$ x$ P* v 随动轮转动时对车子的影响应该尽量小。这里我自己想出了一个设计方案(当然我觉得别人也用过),让车子几乎不受随动轮的影响1 k! v: p+ ~$ z
重心位置" R" D0 `6 \1 u5 u y/ |
重心应该稍稍靠近主动轮以增大主动轮的抓地能力。但是太靠近的话就会降低车子的稳定性,并且升级空间较小* C5 `) Y4 I) Z0 J3 Z: y
结构稳定性 不用说,肯定是越结实越好。但是不能造成重量过大
' ]. E+ w# n, M% i% z2 k/ c" \" [. V 轮子个数 有三种方案:平衡车(独轮或双轮),三轮车,四轮或以上。虽然很想做平衡车,但是无奈暂时不会用陀螺仪。四轮的话转角不好把控,而且结构复杂,所以就用了最简单的三轮车。(另外,我个人认为不能拐弯的车子就不能叫车子~叫火车或者轨道炮比较合适。。。)
* c& z* w/ B$ o 驱动方式 这里我选择了直接由马达驱动轮子,不经过任何齿轮传动,一种简单准确的方式驱动- U, H, \8 m' s( I8 S
% E( Z( h0 `- E8 @# u8 p$ J
控制程序
4 e- T+ F* b1 i! ^- w. `; R+ w. x 运动过程控制3 u) J# e3 L3 J( j
好的程序应该使轮子不打滑,车子运行平稳。由此采取平滑运动的控制方法,让车子开始时(近似)匀加速,直到速度足够(160度每秒)。在开始时,加速度甚至更小,可以参考我的程序(加速度增加->加速度不变->加速度为零)
8 \& d( [7 P0 u! ^7 ?! b9 K' F& h0 X$ I* i# E9 X8 c; ]
调试与运行2 ^( g" {. s6 I2 w
轮子转速调整(车轮大小的设定)5 G5 f+ _. x1 L8 U. D4 R
这个就没什么办法,只能硬着头皮调了。调整轮子的大小设置,实际上就是调马达转速,没有什么更有效的解决办法
7 H: q' U T- v( b; R" L8 x 起始位置车子正方向和直线间的夹角5 J4 _$ [" @* \7 i4 \) v* c
一个非常重要的因素,可谓差之毫厘谬以千里。所以尽可能减小两者的方向差。顺带一提,开环控制的小车,如果最终结果差距在一厘米只内,就已经没有什么可比性了。一厘米之内运气成分占比很大,因为0.01rad(0.5度以内)的夹角足以造成差距一厘米的偏差。这就是开环控制的特点
$ o! e6 W* ^) }' ^, ]
1 H* M7 @% n0 T: [! _- _) F/ ~( [( X' Z, q0 n% F) g& C! U; U% L/ P
然后上源代码:
: D; C: I" P" Y; q& N4 l* v. A" G; v- importlejos.robotics.navigation.DifferentialPilot;
8 p5 ]& k( @/ Z2 {2 ]4 Y6 o$ V, T$ Y - import lejos.util.Delay;4 _$ Y* D4 }( j
- . x* g) ], _% J- G" S1 d, W C
- /**
' l$ e$ n& f- ? v* Z - *Robot that stops if the buttons are presses before it completes its travel.
) q* r; d* ?6 Y1 w - */
5 w; k- O* |. a( }# N" a0 v) J - ( k1 j3 _4 ^/ z6 v
- 0 V: P% B2 R. J6 P4 M1 K( n: W
- public class Straight {: V; X4 z6 R2 E9 r
- DifferentialPilotpilot;7 I; _) ?8 V3 ~0 U' r: b
- doubleleftD = 4.3, rightD = 4.3;
, X* l( }% ]" C! A/ k
) P9 h5 b0 n. C# s4 c- publicvoid adjust() {8 j3 K9 Q6 U/ X; C& `/ C; P
- //wait for release% v8 c' M& r% R0 f
- while(Button.readButtons() > 0)
h9 y% M! A0 ^- L& t% B/ U. a - ;
) \) C' Q) X2 C# I" ]( h2 n - while(Button.readButtons() == 0) {5 H( ~, o" H1 D
- //input size by adjusting the wheels- N% S t# g& P
- rightD= (double) (Motor.A.getTachoCount()) / 2160 + 4.3;
9 ]. r2 h! ^6 ^& f - leftD= (double) (Motor.B.getTachoCount()) / 2160 + 4.3148;// 调试所得值% ^7 s. x" M+ E% h h& p4 G) e( B
- + a7 M, C1 X4 Q0 {: Y
- //display the size input
9 V; b8 S2 R( {& |! R& o/ k - LCD.drawString("L:"+ String.valueOf(leftD), 0, 0);9 a! j$ t( y3 i) @* e
- LCD.drawString("R:"+ String.valueOf(rightD), 0, 1);
. [# T0 u. S8 u" g+ X4 Y/ p - Delay.msDelay(100); n+ r6 [2 q. p. k: w1 M4 e' u
- LCD.clear();
0 q; r8 G+ A5 r - }
% | Z: B. t/ B6 j- {. O& F/ l - ! m0 I: x1 w- Q1 l4 G: z7 p
- }
R1 P1 X* }9 w: k - R1 M2 _3 b: N- Z9 l# w+ O
- publicvoid go() {* F3 m4 p3 W% J T5 a% E, _
- //准备起步% z" P l) t" H8 C2 o- S, R
- //wait for release' X0 |1 y( R1 T2 {
- while(Button.readButtons() > 0)1 j9 v: p3 Q6 y
- ;8 Y+ |" a! P3 ]8 O
- System.out.println("pressto go");, t% n$ Z# X6 R
6 j6 C5 H% o( _) _3 Y" I0 ~$ Y- Button.waitForAnyPress();$ r+ C \4 d$ T+ N: L+ `4 [
- LCD.clear();
. E4 L3 e8 ^( r4 k* Y - # C8 b4 V' {4 G, Y( H
- //初始化
% a9 X1 M1 d* C% J; p2 c - pilot= new DifferentialPilot(rightD, leftD, 13.5, Motor.A, Motor.B, true);
" w2 S, b y+ d8 W - pilot.reset();
. f2 H; F/ }7 _
9 p% K1 _4 Y6 w' E# |- Delay.msDelay(3000);
' @6 u+ H/ ?- N% X+ l0 T
& \! H6 C0 t% F+ G- doublespeed = 10;: L9 \2 r/ ?2 f/ u4 Y
- doubledistance = 0;
/ _8 G- I b7 p1 |/ J1 S - / ?0 J3 d9 U; b$ d0 K5 D
- //起步加速
0 S0 d t: t/ W! l$ P - pilot.travel(300,true);
6 S2 y( H8 M0 D) } - ( C8 M& ~) p6 X1 F
- while(distance < 20) {3 O) b" v v7 z) W
- distance= pilot.getMovement().getDistanceTraveled();
, ?7 s" A" w3 S3 j5 y+ r7 j - speed= distance * 5 + 10;
9 `2 E2 E1 S* L6 d$ f5 v7 D* A - pilot.setRotateSpeed(speed);% P6 U% `9 \- @; m. U! I. @* K/ H
- Delay.msDelay(100);' }8 p, z+ J+ S3 H
- }( i3 h1 H6 Y7 v0 U
- for(int i = 0; i < 10; i++) {
' L) b8 n: i1 {$ F - speed+= 5;
8 K+ r5 S# ~; K3 O - pilot.setRotateSpeed(speed);
) S" J7 I9 q* n Q" X& N, C/ g- |6 c - Delay.msDelay(200);
+ o$ }4 q+ m7 R( `4 o' e - }
; T2 ?. j6 I0 R: V+ y
6 c4 \! z* S! z# t/ P" G( F8 @3 R- //TODO 减速+ F' @/ B- Y. v" v' L+ f
- : |( \ M- N9 j
- //停止机能
# N g* B. q q& n! `4 T4 J% y# p | - while(pilot.isMoving()) {
% Q s5 `( R( }* o6 ~ - if(Button.readButtons() > 0)9 A6 [2 }. y5 H5 D" s/ Q
- pilot.stop();
! S& o9 d r$ O; h - }
+ p% Y* T, A/ m1 \
- ?/ D2 V/ X7 T$ A i4 ?- System.out.println("" + pilot.getMovement().getDistanceTraveled());4 z0 H( q, }# M) t3 d' E
- System.out.println(-Motor.A.getTachoCount());
/ c6 M: k- J* S+ ^. H - System.out.println(-Motor.B.getTachoCount());3 t+ t3 F9 b9 ]& [
- Button.waitForAnyPress();
. M6 L" z7 r# u. M - }. j: v" J5 {. d0 T6 V# z/ d- p
) g. V$ ?# m1 p% j5 j- publicstatic void main(String[] args) {8 x4 f" v* N+ F6 \" O
- Straighttraveler = new Straight();2 I7 k) {! r+ b1 d9 a( d
; Z4 u* Y) Y/ o6 A- traveler.adjust();0 i/ l( z9 k1 N. L
- traveler.go();
. T0 E; ^- i, D9 x - }/ E% _- i- _( |
- }
复制代码 8 v# b. p! W1 A
|