MPU6050工作原理及STM32控制MPU6050

一·简介:

1.要想知道MPU6050工作原理,得先了解下面俩个传感器:

①陀螺仪传感器:

  陀螺仪的原理就是,一个旋转物体的旋转轴所指的方向在不受外力影响时,是不会改变的。人们根据这个道理,用它来保持方向。然后用多种方法读取轴所指示的方向,并自动将数据信号传给控制系统。我们骑自行车其实也是利用了这个原理。轮子转得越快越不容易倒,因为车轴有一股保持水平的力量。现代陀螺仪可以精确地确定运动物体的方位的仪器,它在现代航空,航海,航天和国防工业中广泛使用的一种惯性导航仪器。传统的惯性陀螺仪主要部分有机械式的陀螺仪,而机械式的陀螺仪对工艺结构的要求很高。70年代提出了现代光纤陀螺仪的基本设想,到八十年代以后,光纤陀螺仪就得到了非常迅速的发展,激光谐振陀螺仪也有了很大的发展。光纤陀螺仪具有结构紧凑,灵敏度高,工作可靠。光纤陀螺仪在很多的领域已经完全取代了机械式的传统的陀螺仪,成为现代导航仪器中的关键部件。光纤陀螺仪同时发展的除了环式激光陀螺仪外。

②加速度传感器:

加速度传感器是一种能够测量加速度的传感器。通常由质量块、阻尼器、弹性元件、敏感元件和适调电路等部分组成。传感器在加速过程中,通过对质量块所受惯性力的测量,利用牛顿第二定律获得加速度值。根据传感器敏感元件的不同,常见的加速度传感器包括电容式、电感式、应变式、压阻式、压电式等。

其实说简单点,在mpu6050中我们用陀螺仪传感器测角度,用加速度传感器测加速度

MPU-60X0 :

MPU-60X0是全球首例9轴运动处理传感器。它集成了3轴MEMS陀螺仪, 3轴MEMS 加速度计,以及一个可扩展的数字运动处理器 DMP(DigitalMotion Processor),可用 I2C 接口连接一个第三方的数字传感器,比如磁力计。扩展之后就可以通过其 I2C 或 SPI 接口 输出一个 9 轴的信号(SPI 接口仅在 MPU-6000 可用)。MPU-60X0 也可以通过其 I2C 接口 连接非惯性的数字传感器,比如压力传感器。 MPU-60X0 对陀螺仪和加速度计分别用了三个 16 位的 ADC,将其测量的模拟量转化 为可输出的数字量。为了精确跟踪快速和慢速的运动,传感器的测量范围都是用户可控的, 陀螺仪可测范围为±250,±500,±1000,±2000°/秒(dps),加速度计可测范围为±2,±4, ±8,±16g。 一个片上 1024 字节的 FIFO,有助于降低系统功耗。 和所有设备寄存器之间的通信采用 400kHz 的 I2C 接口或 1MHz 的 SPI 接口(SPI 仅 MPU-6000 可用)。对于需要高速传输的应用,对寄存器的读取和中断可用 20MHz 的 SPI。 另外,片上还内嵌了一个温度传感器和在工作环境下仅有±1%变动的振荡器。 芯片尺寸 4×4×0.9mm,采用 QFN 封装(无引线方形封装),可承受最大 10000g 的冲 击,并有可编程的低通滤波器。 关于电源,MPU-60X0 可支持 VDD 范围 2.5V±5%,3.0V±5%,或 3.3V±5%。另外 MPU-6050 还有一个 VLOGIC 引脚,用来为 I2C 输出提供逻辑电平。VLOGIC 电压可取 1.8±5%或者 VDD。

数字运动处理器(DMP):

   DMP 从陀螺仪、加速度计以及外接的传感器接收并处理数据,处理结果可以从 DMP 寄存器读出,或通过 FIFO 缓冲。DMP 有权使用 MPU 的一个外部引脚产生中断。

二·数据传输:

1.I2C原理在上一篇博客里有详细讲解,在这里不再赘述。

如果要写 MPU-60X0 寄存器,主设备除了发出开始标志(S)和地址位,还要加一个 R/W 位,0 为写,1 为读。在第 9 个时钟周期(高电平时),MPU-60X0 产生应答信号。然 后主设备开始传送寄存器地址(RA),接到应答后,开始传送寄存器数据,然后仍然要有应 答信号,依次类推。

如果要读取 MPU-60X0 寄存器的值,首先由主设备产生开始信号(S),然后发送从设 备地址位和一个写数据位,然后发送寄存器地址,才能开始读寄存器。紧接着,收到应答信 号后,主设备再发一个开始信号,然后发送从设备地址位和一个读数据位。然后,作为从设 备的 MPU-60X0 产生应答信号并开始发送寄存器数据。通信以主设备产生的拒绝应答信号 (NACK)和结束标志(P)结束。拒绝应答信号(NACK)产生定义为 SDA 数据在第 9 个 时钟周期一直为高。

三·STM32控制MPU6050

1.硬件连接

实验采用正点原子公司的 AN1507 ATK-MPU6050 六轴传感器模块

MPU6050 STM32
VCC <---> VCC
GND <---> GND
SDA <---> PB9
SCL <---> PB8
INT <---> 不接
AD0 <---> 不接

2. 重要寄存器

2.1 电源管理寄存器 1

DEVICE_RESET 位用来控制复位,设置为 1,复位 MPU6050,复位结束后, MPU 
硬件自动清零该位

SLEEEP 位用于控制 MPU6050 的工作模式,复位后,该位为 1,即进 
入了睡眠模式(低功耗),所以我们要清零该位,以进入正常工作模式

TEMP_DIS 用于设置是否使能温度传感器,设置为 0,则使能

CLKSEL[2:0]用于选择系统时钟源,选择关系如表

CLKSEL[2:0]       时钟源
000     内部 8M RC 晶振
001     PLL,使用 X 轴陀螺作为参考
010     PLL,使用 Y 轴陀螺作为参考
011     PLL,使用 Z 轴陀螺作为参考
100     PLL,使用外部 32.768Khz 作为参考
101     PLL,使用外部 19.2Mhz 作为参考
110     保留
111     关闭时钟,保持时序产生电路复位状态

**默认是使用内部 8M RC 晶振的,精度不高,所以我们一般选择 X/Y/Z 轴陀螺作为参考 
的 PLL 作为时钟源,一般设置 CLKSEL=001 即可**

2.2 陀螺仪配置寄存器

FS_SEL[1:0]这两个位,用于设置陀螺仪的满量程范围: 0,±250° 
/S; 1,±500° /S; 2,±1000° /S; 3,±2000° /S;我们一般设置为 3,即±2000° /S,因 
为陀螺仪的 ADC 为 16 位分辨率,所以得到灵敏度为: 65536/4000=16.4LSB/(° /S)

2.3 加速度传感器配置寄存器

AFS_SEL[1:0]这两个位,用于设置加速度传感器的满量程范围: 0, 
±2g; 1,±4g; 2,±8g; 3,±16g;我们一般设置为 0,即±2g,因为加速度传感器的 
ADC 也是 16 位,所以得到灵敏度为: 65536/4=16384LSB/g

2.4 FIFO使能寄存器

该寄存器用于控制 FIFO 使能,在简单读取传感器数据的时候,可以不用 FIFO,设置 
对应位为 0 即可禁止 FIFO,设置为 1,则使能 FIFO

加速度传感器的 3 个轴,全由 1 
个位( ACCEL_FIFO_EN)控制,只要该位置 1,则加速度传感器的三个通道都开启 FIFO

2.5 陀螺仪采样率分频寄存器

该寄存器用于设置 MPU6050 的陀螺仪采样频率,计算公式为:

采样频率 = 陀螺仪输出频率 / (1+SMPLRT_DIV)

这里陀螺仪的输出频率,是 1Khz 或者 8Khz,与数字低通滤波器( DLPF)的设置有关, 
当 DLPF_CFG=0/7 的时候,频率为 8Khz,其他情况是 1Khz。而且 DLPF 滤波频率一般设置 
为采样率的一半。采样率,我们假定设置为 50Hz,那么 SMPLRT_DIV=1000/50-1=19

2.6 配置寄存器

数字低通滤波器( DLPF)的设置位,即: DLPF_CFG[2:0],加速 
度计和陀螺仪,都是根据这三个位的配置进行过滤的。 DLPF_CFG 不同配置对应的过滤情 
况如表:

这里的加速度传感器,输出速率( Fs)固定是 1Khz,而角速度传感器的输出速率( Fs), 
则根据 DLPF_CFG 的配置有所不同。一般我们设置角速度传感器的带宽为其采样率的一半, 
如前面所说的,如果设置采样率为 50Hz,那么带宽就应该设置为 25Hz,取近似值 20Hz, 
就应该设置 DLPF_CFG=100

2.7 电源管理寄存器 2

LP_WAKE_CTRL 用于控制低功耗时的唤醒频率

剩下的 6 位,分别控制加速度和陀螺仪的x/y/z轴是否进入待机模式,这里我们全部都不进入待机模式,所以全部设置为 0 即可

2.8 陀螺仪数据输出寄存器

通过读取这6个寄存器,就可以读到陀螺仪 x/y/z 轴的值,比如 x 轴的数据,可以通过读取 
0X43(高 8 位)和 0X44(低 8 位)寄存器得到,其他轴以此类推

2.9 加速度传感器数据输出寄存器

通过读取这6个寄存器,就可以读到加速度传感器 x/y/z 轴的值,比如读 x 轴的数据,可以通过读取 0X3B(高 8 位)和0X3C(低8位)寄存器得到,其他轴以此类推

2.10 温度传感器数据输出寄存器

温度传感器的值,可以通过读取 0X41(高 8 位)和 0X42(低 8 位)寄存器得到, 
温度换算公式为:

Temperature = 36.53 + regval/340

其中, Temperature 为计算得到的温度值,单位为℃, regval 为从 0X41 和 0X42 读到的 
温度传感器值

2.11 中断使能寄存器

OT_EN 该位置 1,该位使能运动检测(Motiondetection)产生中断。

FIFO_OFLOW_EN该位置1,该位使能FIFO缓冲区溢出产生中断。

I2C_MST_INT_EN该位置1,该位使能I2C主机所有中断源产生中断。

DATA_RDY_EN 该位置 1,该位使能数据就绪中断( Data Ready interrupt),所有的传感器寄存器写操作完成时都会产生

关闭所有中断则给此寄存器赋值0X00

3. 软件驱动

3.1 通过IIC对MPU6050寄存器进行读写

  1 //IIC写一个字节
  2 //reg:      寄存器地址
  3 //data:     数据
  4 //返回值:  0,正常
  5 //          其他,错误代码
  6 u8 IIC_Write_Byte(u8 reg,u8 data)
  7 {
  8     IIC_Start();
  9     IIC_Send_Byte((MPU_ADDR<<1)|0);//发送器件地址+写命令
 10     if(IIC_Wait_Ack())  //等待应答
 11     {
 12         IIC_Stop();
 13         return 1;
 14     }
 15     IIC_Send_Byte(reg); //写寄存器地址
 16     IIC_Wait_Ack();     //等待应答
 17         IIC_Send_Byte(data);//发送数据
 18     if(IIC_Wait_Ack())  //等待ACK
 19     {
 20         IIC_Stop();
 21         return 1;
 22     }
 23     IIC_Stop();
 24     return 0;
 25 }
 26
 27 //IIC读一个字节
 28 //reg:寄存器地址
 29 //返回值:读到的数据
 30
 31 u8 IIC_Read_Byte(u8 reg)
 32 {
 33     u8 res;
 34     IIC_Start();
 35     IIC_Send_Byte((MPU_ADDR<<1)|0);//发送器件地址+写命令
 36     IIC_Wait_Ack();//等待应答
 37     IIC_Send_Byte(reg);//写寄存器地址
 38     IIC_Wait_Ack();//等待应答
 39     IIC_Start();
 40     IIC_Send_Byte((MPU_ADDR<<1)|1);//发送期间地址+读命令
 41     IIC_Wait_Ack();//等待应答
 42     res=IIC_Read_Byte(0);//读取数据,发送nACK
 43     IIC_Stop();//产生一个停止条件
 44     return res;
 45 }
 46
 47 //IIC连续写
 48 //addr:器件地址
 49 //reg: 寄存器地址
 50 //len: 写入长度
 51 //buf: 数据区
 52 //返回值: 0,正常
 53 //              其他,错误代码
 54 u8 IIC_Write_Len(u8 addr,u8 reg,u8 len,u8 *buf)
 55 {
 56     u8 i;
 57     IIC_Start();
 58     IIC_Send_Byte((addr<<1)|0);//发送器件地址+写命令
 59     if(IIC_Wait_Ack())//等待应答
 60     {
 61         IIC_Stop();
 62         return 1;
 63     }
 64     IIC_Send_Byte(reg);//写寄存器地址
 65     IIC_Wait_Ack();//等待应答
 66     for(i=0;i<len;i++)
 67     {
 68         IIC_Send_Byte(buf[i]);//发送数据
 69         if(IIC_Wait_Ack())//等待ACK
 70         {
 71             IIC_Stop();
 72             return 1;
 73         }
 74     }
 75     IIC_Stop();
 76     return 0;
 77 }
 78 //IIC连续读
 79 //addr:器件地址
 80 //reg:要读取的寄存器地址
 81 //len:要读取得长度
 82 //buf:读取到的数据存储区
 83 //返回值: 0,正常
 84 //              其他,错误代码
 85 u8 IIC_Read_Len(u8 addr,u8 reg,u8 len,u8 *buf)
 86 {
 87     IIC_Start();
 88     IIC_Send_Byte((addr<<1)|0);//发送器件地址+写命令
 89     if(IIC_Wait_Ack())//等待应答
 90     {
 91         IIC_Stop();
 92         return 1;
 93     }
 94     IIC_Send_Byte(reg);//写寄存器地址
 95     IIC_Wait_Ack();//等待应答
 96     IIC_Start();
 97     IIC_Send_Byte((addr<<1)|1);//发送器件地址+读命令
 98     IIC_Wait_Ack();//等待应答
 99     while(len)
100     {
101         if(len==1) *buf=IIC_Read_Byte(0);//读数据,发送nACK
102         else *buf=IIC_Read_Byte(1);//读数据,发送ACK
103         len--;
104         buf++;
105     }
106     IIC_Stop();//产生一个停止条件
107     return 0;
108 }

3.2 MPU6050初始化

 1 //初始化MPU6050
 2 //返回值: 0,成功
 3 //        其他,错误代码
 4 u8 MPU_Init(void)
 5 {
 6     u8 res;
 7     IIC_Init();//初始化IIC总线
 8     IIC_Write_Byte(MPU_PWR_MGMT1_REG,0X80);//复位MPU6050
 9     delay_ms(100);
10     IIC_Write_Byte(MPU_PWR_MGMT1_REG,0X00);//唤醒MPU6050
11     MPU_Set_Gyro_Fsr(3); //陀螺仪传感器,±2000dps
12     MPU_Set_Accel_Fsr(0); //加速度传感器 ±2g
13     MPU_Set_Rate(50); //设置采样率50HZ
14     IIC_Write_Byte(MPU_INT_EN_REG,0X00); //关闭所有中断
15     IIC_Write_Byte(MPU_USER_CTRL_REG,0X00);//I2C主模式关闭
16     IIC_Write_Byte(MPU_FIFO_EN_REG,0X00);//关闭FIFO
17     IIC_Write_Byte(MPU_INTBP_CFG_REG,0X80);//INT引脚低电平有效
18     res=IIC_Read_Byte(MPU_DEVICE_ID_REG);
19     if(res==MPU_ADDR)//器件ID正确
20     {
21         IIC_Write_Byte(MPU_PWR_MGMT1_REG,0X01);//设置CLKSEL,PLL X 轴为参考
22         IIC_Write_Byte(MPU_PWR_MGMT2_REG,0X00);//加速度陀螺仪都工作
23         MPU_Set_Rate(50); //设置采样率为50HZ
24     }else return 1;
25     return 0;
26 }
27
28 //设置MPU6050陀螺仪传感器满量程范围
29 //fsr:0,±250dps;1,±500dps;2,±1000dps;3,±2000dps
30 //返回值:0,设置成功
31 //    其他,设置失败
32 u8 MPU_Set_Gyro_Fsr(u8 fsr)
33 {
34     return IIC_Write_Byte(MPU_GYRO_CFG_REG,fsr<<3);//设置陀螺仪满量程范围
35 }
36
37 //设置MPU6050加速度传感器满量程范围
38 //fsr:0,±2g;1,±4g;2,±8g;3,±16g
39 //返回值:0,设置成功
40 //    其他,设置失败
41 u8 MPU_Set_Accel_Fsr(u8 fsr)
42 {
43     return IIC_Write_Byte(MPU_ACCEL_CFG_REG,fsr<<3);//设置加速度传感器满量程范围
44 }
45
46 //设置MPU6050的数字低通滤波器
47 //lpf:数字低通滤波频率(Hz)
48 //返回值:0,设置成功
49 //    其他,设置失败
50 u8 MPU_Set_LPF(u16 lpf)
51 {
52     u8 data=0;
53     if(lpf>=188) data=1;
54     else if(lpf>=98) data=2;
55     else if(lpf>=42) data=2;
56     else if(lpf>=42) data=3;
57     else if(lpf>=20) data=4;
58     else if(lpf>=10) data=5;
59     else data=6;
60     return IIC_Write_Byte(MPU_CFG_REG,data);//设置数字低通滤波器
61 }
62
63 //设置MPU6050的采样率(假定Fs=1KHz)
64 //rate:4~1000(Hz)
65 //返回值:0,设置成功
66 //    其他,设置失败
67 u8 MPU_Set_Rate(u16 rate)
68 {
69     u8 data;
70     if(rate>1000)rate=1000;
71     if(rate<4)rate=4;
72     data=1000/rate-1;
73     data=IIC_Write_Byte(MPU_SAMPLE_RATE_REG,data);  //设置数字低通滤波器
74     return MPU_Set_LPF(rate/2); //自动设置LPF为采样率的一半
75 }

3.3 读取MPU6050相关测得原始数据

 1 //得到温度值
 2 //返回值:温度值(扩大了100倍)
 3 short MPU_Get_Temperature(void)
 4 {
 5     u8 buf[2];
 6     short raw;
 7         float temp;
 8         IIC_Read_Len(MPU_ADDR,MPU_TEMP_OUTH_REG,2,buf);
 9     raw=((u16)buf[0]<<8)|buf[1];
10     temp=36.53+((double)raw)/340;
11     return temp*100;;
12 }
13 //得到陀螺仪值(原始值)
14 //gx,gy,gz:陀螺仪x,y,z轴的原始读数(带符号)
15 //返回值:0,成功
16 //    其他,错误代码
17 u8 MPU_Get_Gyroscope(short *gx,short *gy,short *gz)
18 {
19     u8 buf[6],res;
20     res=IIC_Read_Len(MPU_ADDR,MPU_GYRO_XOUTH_REG,6,buf);
21     if(res==0)
22     {
23         *gx=((u16)buf[0]<<8)|buf[1];
24         *gy=((u16)buf[2]<<8)|buf[3];
25         *gz=((u16)buf[4]<<8)|buf[5];
26     }
27     return res;
28 }
29
30 //得到加速度值(原始值)
31 //ax,ay,az:陀螺仪x,y,z轴的原始读数(带符号)
32 //返回值:0,成功
33 //    其他,错误代码
34 u8 MPU_Get_Accelerometer(short *ax,short *ay,short *az)
35 {
36     u8 buf[6],res;
37     res=IIC_Read_Len(MPU_ADDR,MPU_ACCEL_XOUTH_REG,6,buf);
38     if(res==0)
39     {
40         *ax=((u16)buf[0]<<8)|buf[1];
41         *ay=((u16)buf[2]<<8)|buf[3];
42         *az=((u16)buf[4]<<8)|buf[5];
43     }
44     return res;;
45 }

https://blog.csdn.net/he__yuan/article/details/76559569

原文地址:https://www.cnblogs.com/zhb123456/p/10622557.html

时间: 2024-10-12 03:41:59

MPU6050工作原理及STM32控制MPU6050的相关文章

apache2.X 版本工作模式,以及各自工作原理

工作模式为:prefork和worker prefork:多进程,每个进程产生子进程 worker:多进程,每个进程生成多个线程 工作原理: prefork 的工作原理是,控制进程在最初建立―StartServers|个子进程后,为了满足MinSpareServers 设置的需要创建一个进程,等待一秒钟,继续创建两个,再等待一秒钟,继续创建四个--如此按指数级增加创建的进程数,最多达到每秒32 个,直到满足MinSpareServers 设置的值为止.这就是预派生(prefork)的由来.这种模

MVC 模型、视图、控制及其单入口文件的mvc的工作原理

     什么是mvc,mvc就是模型视图控制,模型就是model,在项目中负责数据库相关的操作,视图就是view ,负责页面的展示和数据的展示,控制就是controller ,负责中间的逻辑转换,数据的接收,输出的处理.mvc的好处是让项目开发能够让代码和页面尽可能的分解开,相互之间的关联尽可能的少.这只是一种开发方式,据说是从java过来的,真是坑人的java啊.      常见的php框架基本上都属于mvc的开发模式,例如:Yii(大赞),Ci,Tp(国产),这几个是我用到过的,还有很多p

[Java] SSH框架笔记_SSH三大框架的工作原理及流程

Hibernate工作原理及为什么要用? 原理:1.通过Configuration().configure();读取并解析hibernate.cfg.xml配置文件2.由hibernate.cfg.xml中的<mapping resource="com/xx/User.hbm.xml"/>读取并解析映射信息3.通过config.buildSessionFactory();//创建SessionFactory4.sessionFactory.openSession();//打

[转]ZK工作原理

ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,它包含一个简单的原语集,分布式应用程序可以基于它实现同步服务,配置维护和命名服务等.Zookeeper是hadoop的一个子项目,其发展历程无需赘述.在分布式应用中,由于工程师不能很好地使用锁机制,以及基于消息的协调机制不适合在某些应用中使用,因此需要有一种可靠的.可扩展的.分布式的.可配置的协调机制来统一系统的状态.Zookeeper的目的就在于此.本文简单分析zookeeper的工作原理,对于如何使用zookeeper不是本

???Struts2框架03 session的使用、登录逻辑【session工作原理】

1 登录逻辑 1.1 获取登录数据(例如:用户民.密码) 1.2 在控制层调用业务层来验证数据信息 1.3 登录成功:保存用户信息(服务器用session.浏览器用cookie),跳转到主页满 1.4 登录失败:留在登录页面,并给出错误提示信息 2 什么是session 就是一个会话对象,有服务器创建,目的是为了共享数据(注意:每一个浏览器对应一个session对象) 3 session的工作原理 待更新... 2017年7月3日21:53:09 4 在struts2中获取session对象的方

计算机的基本组成及工作原理

计算机的基本组成及工作原理 1.3.1 计算机系统的组成 计算机系统是由硬件系统和软件系统两大部分组成,这一节将分别介绍计算机硬件系统和软件系统. 计算机硬件是构成计算机系统各功能部件的集合.是由电子.机械和光电元件组成的各种计算机部件和设备的总称,是计算机完成各项工作的物质基础.计算机硬件是看得见.摸得着的,实实在在存在的物理实体. 计算机软件是指与计算机系统操作有关的各种程序以及任何与之相关的文档和数据的集合.其中程序是用程序设计语言描述的适合计算机执行的语句指令序列. 没有安装任何软件的计

Struts2(二):工作原理

struts可查看源码:https://github.com/apache/struts 在学习struts2之前,我先看了一些比较早版本对struts2的工作原理相关的介绍,顺便抄写过来,用来帮助自己要弄清这个框架的工作原理. struts2.1.3之前使用的FilterDispatcher,之后的版本使用StrutsPrepareAndExecuteFilter,而我这里还是以Dispatcher为例来记录的. 依据: Since Struts 2.1.3, use org.apache.s

系统栈的工作原理(转)

1.开篇 本篇文章着重写的是系统中栈的工作原理,以及函数调用过程中栈帧的产生与释放的过程,有可能名字过大,如果不合适我可以换一个名字,希望大家能够指正,小丁虚心求教!如果有哪里写的不清楚的或者错误的地方请及时更正,小丁再次谢过了.文章里面有错别字,也可能会有好友说寄存器的32.16位的区别其实我感觉这里主要讲的还是些原理性的东西,后续会将文章图片错别字进行调整.(图片里面的posh改为push) 2.内存的不同用途 根据不同的操作系统,一个进程可能被分配到不同的内存区域去执行.但是不管什么样的操

谈谈Angular关于$watch,$apply 以及 $digest的工作原理

这篇文章主要是面向那些刚开始学AngularJs和想要了解数据绑定(data-binding)是怎么工作的, 如果你已经熟悉如何使用angularjs了,我强烈建议你不用阅读了. angularjs使用者想要知道data-binding是如何工作的,就会遇到很多的关的术语 比如$wacth,$apply,$digest,dirty-checking(脏值检测)...等等,这些又是做什么的呢? 在这篇文章里我会解决所有的疑问,通过结合这些术语在一起来学习. 但是我会尽量用简单的方式来说明. 现在开