特殊功能寄存器sfr
sfr也是一种扩充数据类型,点用一个内存单元,值域为0~255。
利用它可以访问51单片机内部的所有特殊功能寄存器。(sfr 是定义8 位的特殊功能寄存器而sfr16 则是用来定义16 位特殊功能寄存器)
(1) sbit 位变量名=位地址
sbit P1_1 = Ox91;
这样是把位的绝对地址赋给位变量.同sfr 一样sbit 的位地址必须位于80H-FFH 之间.
(2) sbit 位变量名=特殊功能寄存器名^位位置
sft P1= 0x90;
sbit P1_1 = P1 ^ 1;
//先定义一个特殊功能寄存器名再指定位变量名所在的位置,当可
寻址位位于特殊功能寄存器中时可采用这种方法
(3) sbit 位变量名=字节地址^位位置
sbit P1_1 = 0x90 ^ 1;
这种方法其实和2 是一样的,只是把特殊功能寄存器的位址直接用常数表示.
在C51存储器类型中提供有一个bdata 的存储器类型,这个是指可位寻址的数据存储器,位于单片机的可位寻址区中,可以将要求可位录址的数据定义为bdata,
如:
unsigned char bdata ib; //在可位录址区定义ucsigned char 类型的变量ib
sbit ib7=ib^7 //用关键字sbit 定义位变量来独立访问可寻址位对象的其中一位
int bdata ab[2]; //在可位寻址区定义数组ab[2],这些也称为可寻址位对象
sbit ab12=ab[1]^12;
/
sfr 并标准C 语言的关键字,而是Keil 为能直接访问80C51 中的SFR 而提供了一个新的关键词,其用法是:
sfrt 变量名=地址值。
///
二.符号P1_0 来表示P1.0 引脚。
在C 语言里,如果直接写P1.0,C 编译器并不能识别,而且P1.0 也不是一个合法的C
语言变量名,所以得给它另起一个名字,Keil C的关键字sbit 来定义,sbit 的用法有三种:
第一种方法:sbit 位变量名=地址值
第二种方法:sbit 位变量名=SFR 名称^变量位地址值
第三种方法:sbit 位变量名=SFR 地址值^变量位地址值
如定义PSW 中的OV 可以用以下三种方法:
sbit OV=0xd2 (1)说明:0xd2 是OV 的位地址值
sbit OV=PSW^2 (2)说明:其中PSW 必须先用sfr 定义好
sbit OV=0xD0^2 (3)说明:0xD0 就是PSW 的地址值
3).sbit可录址位
sbit同位是C51中的一种扩充数据类型,利用它可以访问芯片内部的RAM中的可寻址位或特殊功能寄存器中的可寻址位。如先前我们定义了
sfr P1 = 0x90; //因P1端口的寄存器是可位寻址的,所以我们可以定义
sbit P1_1 = P1^1; //P1_1为P1中的P1.1引脚
//同样我们可以用P1.1的地址去写,如sbit P1_1 = 0x91;
MCS-51单片机的特殊功能寄存器
|
||
符号
|
地址
|
功能介绍
|
B
|
F0H
|
B寄存器
|
ACC
|
E0H
|
累加器
|
PSW
|
D0H
|
程序状态字
|
TH2*
|
CDH
|
定时器/计数器2(高8位)
|
TL2*
|
CCH
|
定时器/计数器2(低8位)
|
RLDH*
|
CBH
|
外部输入(P1.1)计数器/自动再装入模式时初值寄存器高八位
|
RLDL*
|
CAH
|
外部输入(P1.1)计数器/自动再装入模式时初值寄存器低八位
|
T2CON*
|
C8H
|
T2定时器/计数器控制寄存器
|
IP
|
B8H
|
中断优先级控制寄存器
|
P3
|
B0H
|
P3口锁存器
|
IE
|
A8H
|
中断允许控制寄存器
|
P2
|
A0H
|
P2口锁存器
|
SBUF
|
99H
|
串行口锁存器
|
SCON
|
98H
|
串行口控制寄存器
|
P1
|
90H
|
P1口锁存器
|
TH1
|
8DH
|
定时器/计数器1(高8位)
|
TH0
|
8CH
|
定时器/计数器0(高8位)
|
TL1
|
8BH
|
定时器/计数器1(低8位)
|
TL0
|
8AH
|
定时器/计数器0(低8位)
|
TMOD
|
89H
|
T0、T1定时器/计数器方式控制寄存器
|
TCON
|
88H
|
T0、T1定时器/计数器控制寄存器
|
DPH
|
83H
|
数据地址指针(高8位)
|
DPL
|
82H
|
数据地址指针(低8位)
|
SP
|
81H
|
堆栈指针
|
P0
|
80H
|
P0口锁存器
|
PCON
|
87H
|
电源控制寄存器
|
#include //包含51单片机寄存器定义的头文件
void main(void)
{
P0=0xfe; //P0=1111 1110B,即P1.0输出低电平 直接赋初值
P0=0xfd;
P0=0xff;
P0=0x00;
}
//51单片机低电平有效
/
#include //控制p0连续闪烁
void delay() //延时
{
unsigned int i;
for(i=0;i<20000;i++);
}
void main()
{
while(1)
{
P0=0xfe;
delay();
P0=0xff;
delay();
}
}
///
#include
#define uchar unsigned char
#define uint unsigned int
sbit LED=P0^0;
void DelayMS(uint x)
{
uchar i;
while(x--)
{
for(i=0;i<120;i++);
}
}
void main()
{ while(1)
{
LED=~LED;
DelayMS(150);
}
}
///
#include //包含单片机寄存器的头文件
sfr x=0x80; //P0口在存储器中的地址是b0H, 通过sfr可定义8051内核单片机
//的所有内部8位特殊功能寄存器,对地址x的操作也就是对P1口的操作
//p1:90H
//p2:A0H
//P3:B0H
/****************************************
函数功能:延时一段时间
*****************************************/
void delay(void)
{
unsigned char i,j;
for(i=0;i<250;i++)
for(j=0;j<250;j++)
; //利用循环等待若干机器周期,从而延时一段时间
}
/*****************************************
函数功能:主函数
******************************************/
void main(void)
{
while(1)
{
/
x=0xfe; //第一个灯亮
delay(); //调用延时函数
x=0xfd; //第二个灯亮
delay(); //调用延时函数
x=0xfb; //第三个灯亮
delay(); //调用延时函数
x=0xf7; //第四个灯亮
delay(); //调用延时函数
P0=0xef; //第五个灯亮
delay(); //调用延时函数
P0=0xdf; //第六个灯亮
delay(); //调用延时函数
P0=0xbf; //第七个灯亮
delay(); //调用延时函数
P0=0x7f; //第八个灯亮
delay(); //调用延时函数
}
}
///
函数功能:用整形数据延时一段时间
******************************************************/
void int_delay(void) //延时一段较长的时间
{
unsigned int m; //定义无符号整形变量,双字节数据,值域为0~65535
for(m=0;m<36000;m++)
; //空操作
}
/******************************************************
函数功能:用字符型数据延时一段时间
******************************************************/
void char_delay(void) //延时一段较短的时间
{
unsigned char i,j; //定义无符号字符型变量,单字节数据,值域0~255
for(i=0;i<200;i++)
for(j=0;j<180;j++)
; //空操作
}