第六届省赛题目,注意矩阵键盘在温度边界设置上的用法即可。

题目要求

简易温度温度采集与控制装置,题目可于此处下载


注意事项

  • 矩阵按键的逐行扫描:
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
//矩阵按键状态与备份
uchar key_stat[4][4]={{1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1}};
uchar key_back[4][4]={{1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1}};

void scan_key()
{
static uchar key_buff[4][4]={{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff}};
static uchar key_out=0; //行索引
uchar i;
key_buff[key_out][0]=(key_buff[key_out][0]<<1)|KEY_IN_0;
key_buff[key_out][1]=(key_buff[key_out][1]<<1)|KEY_IN_1;
key_buff[key_out][2]=(key_buff[key_out][2]<<1)|KEY_IN_2;
key_buff[key_out][3]=(key_buff[key_out][3]<<1)|KEY_IN_3;

for(i=0;i<4;i++)
{
if((key_buff[key_out][i]&0x0f)==0x0f)
{
key_stat[key_out][i]=1;
}
else if((key_buff[key_out][i]&0x0f)==0x00)
{
key_stat[key_out][i]=0;
}
}

key_out++;
key_out&=0x03;

switch(key_out)
{
case 0: KEY_OUT_3=1;KEY_OUT_0=0;break;
case 1: KEY_OUT_0=1;KEY_OUT_1=0;break;
case 2: KEY_OUT_1=1;KEY_OUT_2=0;break;
case 3: KEY_OUT_2=1;KEY_OUT_3=0;break;
default:break;
}
}

void key_press()
{
uchar i,j;
for(i=0;i<4;i++)
{
for(j=0;j<4;j++)
{
if(key_stat[i][j]!=key_back[i][j])
{
if(key_stat[i][j]==0)
{
key_fun(key_map[i][j]);
}
key_back[i][j]=key_stat[i][j];
}
}
}
}
  • 关于矩阵键盘的在温度设置上的使用,可以设置buff数组暂存数据,当S8按下时判定数据是否合法。在buff数组中,11代表熄灭意为关闭当前数码管,按下S7将所有位置为11。
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
uchar buff[4]={11,11,11,11};
void key_fun(uchar value)
{
static uchar pos=0; //按键位置
if(value>=0&&value<=9) //如果按下0~9并且处于设置模式则对buff进行修改
{
if(mode==1)
{
buff[pos]=value;
pos++;
if(pos>=4)
{
pos=4;
}
}
}
if(value==10) //如果按下设置按键则根据情况改变工作模式
{
if(mode==0)
{
mode=1;
}
else if(mode==1)
{
if(tmin=buff[2]*10+buff[3]>=buff[0]*10+buff[1]) //参数设置非法
{
flag_error=1;
mode=1;
}
else
{
mode=0;
tmax=buff[0]*10+buff[1];
tmin=buff[2]*10+buff[3];
buff[0]=buff[1]=buff[2]=buff[3]=11;
pos=0;
flag_error=0;
}
}
}
if(value==11&&mode==1) //如果在设置模式下按下清除
{
pos=0;
buff[0]=buff[1]=buff[2]=buff[3]=11;
}
}
  • 关于LED闪烁的部分,我写了两种但是最后发现只有一种可以正常工作,如下(注释部分工作不正常):
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
void ser_timer0() interrupt 1
{
static uint count_200ms,count_400ms;
count_200ms++;
count_400ms++;
if(count_200ms>=200)
{
count_200ms=0;
flag_200ms=~flag_200ms;
}
<!-- if(count_400ms>=400)
{
count_400ms=0;
flag_400ms=1;
} -->
}

void led_fun()
{
if(flag_200ms)
{
led_on(0xfe);
}
else
{
led_on(0xff);
}
<!-- if(flag_400ms)
{
L1=~L1;
flag_400ms=0;
} -->
}

实例

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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
#include<stc15f2k60s2.h>
#include"onewire.h"
typedef unsigned char uchar;
typedef unsigned int uint;
//数码管段码与按键分布
uchar code dig_code[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xbf,0xff,0xc6};
uchar code key_map[4][4]={{0,1,2,19},{3,4,5,18},{6,7,8,17},{9,10,11,16}};
//矩阵按键状态与备份
uchar key_stat[4][4]={{1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1}};
uchar key_back[4][4]={{1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1}};
//数码管与按键缓冲区
uchar buff_tube[8]={0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff};
uchar buff[4]={11,11,11,11};
//逐行扫描
sbit KEY_IN_0=P4^4;
sbit KEY_IN_1=P4^2;
sbit KEY_IN_2=P3^5;
sbit KEY_IN_3=P3^4;
sbit KEY_OUT_0=P3^0;
sbit KEY_OUT_1=P3^1;
sbit KEY_OUT_2=P3^2;
sbit KEY_OUT_3=P3^3;
sbit L1=P0^0;
sbit L2=P0^1;
sbit relay=P0^4;
//变量:最低温度、最高温度、当前工作模式、当前的温度模式
uchar tmin=20,tmax=30,mode=0,mode_temp=0;
//一堆标志变量
uchar flag_1ms,flag_200ms,flag_400ms,flag_800ms,flag_error,flag_1s;
uint temp=0;

void select(uchar channel)
{
switch(channel)
{
case 4:
P2=(P2&0x1f)|0x80;
break;
case 5:
P2=(P2&0x1f)|0xa0;
break;
case 6:
P2=(P2&0x1f)|0xc0;
break;
case 7:
P2=(P2&0x1f)|0xe0;
break;
case 0:
P2=(P2&0x1f);
break;
}
}

void init_sys()
{
select(5);
P0=0x00;
select(4);
P0=0xff;
select(0);
}

void display()
{
static uchar index=0;
select(7);
P0=0xff;
select(6);
P0=0x01<<index;
select(7);
P0=buff_tube[index];
select(0);

index++;
index&=0x07;
}

void scan_key()
{
static uchar key_buff[4][4]={{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff}};
static uchar key_out=0; //行索引
uchar i;
key_buff[key_out][0]=(key_buff[key_out][0]<<1)|KEY_IN_0;
key_buff[key_out][1]=(key_buff[key_out][1]<<1)|KEY_IN_1;
key_buff[key_out][2]=(key_buff[key_out][2]<<1)|KEY_IN_2;
key_buff[key_out][3]=(key_buff[key_out][3]<<1)|KEY_IN_3;

for(i=0;i<4;i++)
{
if((key_buff[key_out][i]&0x0f)==0x0f)
{
key_stat[key_out][i]=1;
}
else if((key_buff[key_out][i]&0x0f)==0x00)
{
key_stat[key_out][i]=0;
}
}

key_out++;
key_out&=0x03;

switch(key_out)
{
case 0: KEY_OUT_3=1;KEY_OUT_0=0;break;
case 1: KEY_OUT_0=1;KEY_OUT_1=0;break;
case 2: KEY_OUT_1=1;KEY_OUT_2=0;break;
case 3: KEY_OUT_2=1;KEY_OUT_3=0;break;
default:break;
}
}

void key_fun(uchar value)
{
static uchar pos=0; //按键位置
if(value>=0&&value<=9) //如果按下0~9并且处于设置模式则对buff进行修改
{
if(mode==1)
{
buff[pos]=value;
pos++;
if(pos>=4)
{
pos=4;
}
}
}
if(value==10) //如果按下设置按键则根据情况改变工作模式
{
if(mode==0)
{
mode=1;
}
else if(mode==1)
{
if(tmin=buff[2]*10+buff[3]>=buff[0]*10+buff[1]) //参数设置非法
{
flag_error=1;
mode=1;
}
else
{
mode=0;
tmax=buff[0]*10+buff[1];
tmin=buff[2]*10+buff[3];
buff[0]=buff[1]=buff[2]=buff[3]=11;
pos=0;
flag_error=0;
}
}
}
if(value==11&&mode==1) //如果在设置模式下按下清除
{
pos=0;
buff[0]=buff[1]=buff[2]=buff[3]=11;
}
}

void key_press()
{
uchar i,j;
for(i=0;i<4;i++)
{
for(j=0;j<4;j++)
{
if(key_stat[i][j]!=key_back[i][j])
{
if(key_stat[i][j]==0)
{
key_fun(key_map[i][j]);
}
key_back[i][j]=key_stat[i][j];
}
}
}
}

void set_display()
{
if(mode==0)
{
buff_tube[0]=dig_code[10];
buff_tube[1]=dig_code[mode_temp];
buff_tube[2]=dig_code[10];
buff_tube[3]=dig_code[11];
buff_tube[4]=dig_code[11];
buff_tube[5]=dig_code[11];
buff_tube[6]=dig_code[temp/10];
buff_tube[7]=dig_code[temp%10];
}
else
{
buff_tube[0]=dig_code[10];
buff_tube[1]=dig_code[buff[0]];
buff_tube[2]=dig_code[buff[1]];
buff_tube[3]=dig_code[11];
buff_tube[4]=dig_code[11];
buff_tube[5]=dig_code[10];
buff_tube[6]=dig_code[buff[2]];
buff_tube[7]=dig_code[buff[3]];
}
}

void init_timer0(void) //1毫秒@11.0592MHz
{
AUXR |= 0x80; //定时器时钟1T模式
TMOD &= 0xF0; //设置定时器模式
TL0 = 0xCD; //设置定时初值
TH0 = 0xD4; //设置定时初值
TF0 = 0; //清除TF0标志
TR0 = 1; //定时器0开始计时
ET0 = 1;
EA = 1;
}

void ser_timer0()interrupt 1
{
static uint count_2ms,count_200ms,count_400ms,count_800ms,count_1s;
count_2ms++;
count_200ms++;
count_400ms++;
count_800ms++;
count_1s++;
flag_1ms=1;
if(count_2ms>=2)
{
count_2ms=0;
set_display();
display();
}
if(count_200ms>=200)
{
count_200ms=0;
flag_200ms=~flag_200ms;
}
if(count_400ms>=400)
{
count_400ms=0;
flag_400ms=1;
}
if(count_800ms>=800)
{
count_800ms=0;
flag_800ms=~flag_800ms;
}
if(count_1s>=1000)
{
count_1s=0;
flag_1s=1;
}
}

void led_on(uchar dat)
{
P0=0xff;
select(4);
P0=dat;
select(0);
}

void led_fun()
{
if(flag_error)
{
P0=0xff;
select(4);
L2=0;
select(0);
}
else if(flag_error==0)
{
P0=0xff;
select(4);
L2=1;
select(0);
}

if(temp<tmin)
{
mode_temp=0;
P0=0x00;
select(5);
relay=0;
select(0);
if(flag_800ms)
{
led_on(0xfe);
}
else
{
led_on(0xff);
}
}
else if(temp<tmax)
{
mode_temp=1;
P0=0x00;
select(5);
relay=0;
select(0);
if(flag_400ms)
{
led_on(0xfe);
}
else
{
led_on(0xff);
}
}
else
{
mode_temp=2;
P0=0x00;
select(5);
relay=1;
select(0);
if(flag_200ms)
{
led_on(0xfe);
}
else
{
led_on(0xff);
}
}
}

void get_temp()
{
uchar LSB,MSB;
EA=0;
init_ds18b20();
Write_DS18B20(0xcc);
Write_DS18B20(0x44);
init_ds18b20();
Write_DS18B20(0xcc);
Write_DS18B20(0xbe);
LSB=Read_DS18B20();
MSB=Read_DS18B20();
init_ds18b20();
temp=(MSB<<8)|LSB;
temp=temp>>4;
EA=1;
}

void main()
{
init_sys();
init_timer0();
while(1)
{
if(flag_1ms)
{
flag_1ms=0;
led_fun();
scan_key();
key_press();
}
if(flag_1s)
{
flag_1s=0;
get_temp();
}
}
}