虚拟驾驶环境搭建(四)

龙云尧个人博客,转载请注明出处。

CSDN地址:http://blog.csdn.net/michael753951/article/details/75074051

个人blog地址:http://yaoyl.cn/huan-jing-da-jian-si-2/


障碍物移动的改进

在之前的版本中.我们的障碍物移动和场景移动放在同一个for循环中,虽然减少了代码的操作,但是很容易出现边界问题,程序运行中往往会出现障碍物骤现,或者障碍物还未抵达车前方便突然消失的情况,为了修改这出bug,我将重写这部分代码.

需求分析

无脑写代码的多数结果都是bug不断,并且写完后文就忘记前文.为了提高代码的清晰度和准确性,挺尸帮我们捋清思路,我们先通过写伪代码的方式整理一下我们的程序有哪些需求,以及粗略的实现方式.

需求

障碍物能够从左侧楼栋出现,到达路中央之后,将停留一小段时间(等待车撞上).

对这个过程进行分解分析,我们可以大致得到如下流程:

  • 障碍物出现的时机: 左侧有障碍物 & 距离上次消失已经有10s以上
  • 障碍静止时机: 已经达到路中间, 并且停留时间没有达到2s
  • 障碍物消失诗句时机: 停留2s之后,障碍物消失

伪代码

  • 首先判断障碍物是否已经消失,然后判断左侧是否有障碍物,然后检查消失时间是否已经达到10s,如果满足条件,就更新appear标志, 让障碍物开始移动
  • 开始步进障碍物, 直到达到边界, 此时更新appear_tmp, 更新appear标志, 记录下当前时间
  • 如果当前状态为静止, 并且停留时间还未到达2s,保持pos不变
  • 否则如果停留时间超过2s,就将appear更新为消失, 让障碍物消失, 同时用appear_tmp记录下消失的时间, 方便之后判断障碍物的出现时机

代码实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
// 先获取左侧建筑情况
int left_temp = building_left_flag;
// bdw_appear有3个值, 0表示已经消失, 1表示正在进行移动, 2表示移动到边界,已经静止
if(bdw_appear == 0) {
if(left_temp >> 11 & 0b1 == 1) {
time(&appear);
// 如果距离上次出现已经5s,就更新记录,同时更新flag
if(appear - appear_tmp >= 5){
appear_tmp = appear;
bdw_appear = 1;
}
}
}else if(bdw_appear == 1){
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
glTranslatef(child_run_len, -1.0f, -7.0f);
glCallList(children);
if(child_run_len >= -0.5) {
//child_run_len = -3.0f;
//bdw_appear = false;
// 刚移动到边界
if(bdw_appear == 1){
time(&appear_tmp);
child_pos = -7.0f;
bdw_appear = 2;
}
child_run_len = -0.5f;
}else{
// 小孩的速度恒定
child_run_len += 0.1f;
}
}else{
time(&appear);
// 如果距离上次出现已经5s,就更新记录,同时更新flag
if(appear - appear_tmp >= 2){
appear_tmp = appear;
child_run_len = -3.0f;
bdw_appear = 0;
}else{
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
// 障碍物的pos还要加上run_len, 这样能够让障碍物在达到路中央静止之后, 和周围物体一起移动,产生向人靠近的感觉
glTranslatef(child_run_len, -1.0f, child_pos+run_len);
glCallList(children);
}
}

实现障碍物从两侧随机出现

为了和真实驾驶更加相似,我们不应该让障碍物只从单侧出现,而是因该让其两侧都有可能出现.因此,我们需要在原来的代码上进行进一步的修改.

需求分析

需求

首先我们暂时让不倒翁的出现有两种情况, 情况一, 从左侧出现, 情况二, 从两侧随机出现. 通过按键’c’来控制模式的转换(改变appear_mode的值).

因此, 当我们按下’c’键时, 需要对模式进行转换. 接下来在DrawGLScene中将根据appear_mode的值进行一系列操作. 如果appear_mode为默认的0, 就让障碍物从左侧出现, 如果appear_mode为1, 就让障碍物从两侧出现.

如果障碍物从两侧出现的话, 每次障碍物将要出现的时候, 我们需要用一个flag标志保存障碍物本次出现是从左侧出现还是右侧出现. 在绘制障碍物的时候, 我们将通过这个flag来判断物体是如何运动.

  • 障碍物出现的时机: 本身已经消失, 距离上次消失已经有10s以上, 决定好从哪一侧出现, 并且该侧有楼栋
  • 障碍物移动的时机: 本身已经出现, 根据上一步已经得到的flag判定是向左还是向右, 直到到临界点
  • 障碍物静止的时机: 已经达到路中间,并且停留时间并未超过2s
  • 障碍物消失的时机: 已经静止超过2s

伪代码

  • 首先我们需要在KeyPressed函数中, 添加按键’c’的控制事件, 另其对appear_mode进行修改.
  • 接下来在DrawGLScene中, 我们将对这个appear_mode进行判断. 如果appear_mode为0,就按照之前的代码执行. 如果appear_mode为1, 则进入两侧出现事件.
  • 当appear_mode为1时, 如果当前状态为消失, 如果当前已经消失10s以上, 则随机决定在哪应该出现障碍物, 然后检查该侧是否有楼栋, 如果有楼栋, 则保存裁决位置(左或者右), 同时更新当前状态为移动
  • 当appear_mode为1时, 当前状态为移动时, 根据裁决, 如果裁决决定从左侧出来, 则起点设置为左侧, 同时,不断增加child_run_len,直到到达边界; 如果裁决决定从右侧出来, 则起点设置为1.0f位置, 同时不断减小, 直到到达-0.5f位置停止.
  • 停止时, 将appear_mode设置为2, 记录开始时间, 在原地停留两秒之后, 消失.

代码实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
int left_temp = building_left_flag;
int right_temp = building_right_flag;
if(appear_mode == 0) {
// bdw_appear有3个值, 0表示已经消失, 1表示正在进行移动, 2表示移动到边界,已经静止
if(bdw_appear == 0) {
if(left_temp >> 11 & 0b1 == 1) {
time(&appear);
// 如果距离上次出现已经5s,就更新记录,同时更新flag
if(appear - appear_tmp >= 5) {
appear_tmp = appear;
bdw_appear = 1;
child_run_len = -3.0f;
}
}
} else if(bdw_appear == 1) {
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
glTranslatef(child_run_len, -1.0f, -7.0f);
glCallList(children);
if(child_run_len >= -0.5) {
//child_run_len = -3.0f;
//bdw_appear = false;
// 刚移动到边界
if(bdw_appear == 1) {
time(&appear_tmp);
child_pos = -7.0f;
bdw_appear = 2;
}
child_run_len = -0.5f;
} else {
// 小孩的速度恒定
child_run_len += 0.1f;
}
} else {
time(&appear);
// 如果距离上次出现已经5s,就更新记录,同时更新flag
if(appear - appear_tmp >= 2) {
appear_tmp = appear;
child_run_len = -3.0f;
bdw_appear = 0;
} else {
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
glTranslatef(child_run_len, -1.0f, child_pos+run_len);
glCallList(children);
}
}
}else if(appear_mode == 1) {
if(bdw_appear == 0) {
time(&appear);
if(appear - appear_tmp >= 5){
// 奇数为左侧, 偶数为右侧
if(rand() & 0b1 == 1) {
if(left_temp >> 11 & 0b1 == 1) {
time(&appear);
appear_tmp = appear;
bdw_appear = 1;
left_or_right_flag = 1;
child_run_len = -3.0f;
}
}else{
if(right_temp >> 11 & 0b1 == 1) {
time(&appear);
appear_tmp = appear;
bdw_appear = 1;
left_or_right_flag = 0;
child_run_len = 1.0f;
}
}
}
} else if(bdw_appear == 1) {
if(left_or_right_flag == 1){
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
glTranslatef(child_run_len, -1.0f, -7.0f);
glCallList(children);
if(child_run_len >= -0.5) {
//child_run_len = -3.0f;
//bdw_appear = false;
// 刚移动到边界
if(bdw_appear == 1) {
time(&appear_tmp);
child_pos = -7.0f;
bdw_appear = 2;
}
child_run_len = -0.5f;
} else {
// 小孩的速度恒定
child_run_len += 0.1f;
}
}else{
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
glTranslatef(child_run_len, -1.0f, -7.0f);
glCallList(children);
if(child_run_len <= -0.5f) {
if(bdw_appear == 1) {
time(&appear_tmp);
child_pos = -7.0f;
bdw_appear = 2;
}
} else {
child_run_len -= 0.1f;
}
}
} else {
time(&appear);
// 如果距离上次出现已经5s,就更新记录,同时更新flag
if(appear - appear_tmp >= 2) {
appear_tmp = appear;
child_run_len = -3.0f;
bdw_appear = 0;
} else {
glBindTexture(GL_TEXTURE_2D, texture[4]);
glLoadIdentity();
glTranslatef(child_run_len, -1.0f, child_pos+run_len);
glCallList(children);
}
}
}
Donate
  • Copyright: Copyright is owned by the author. For commercial reprints, please contact the author for authorization. For non-commercial reprints, please indicate the source.

请我喝杯咖啡吧~

支付宝
微信