unsigned int类型的数据进行按位调换,0bit调换到31bit,31bit调换到0bit

本文介绍了一种位操作技巧,用于将unsigned int类型数据的位进行反转。通过具体的C语言实现,展示了如何将32位整数的最高位与最低位交换,并提供了测试案例。
/*
函数目的,将unsigned int类型的数据进行按位调换,0bit调换到31bit,31bit调换到0bit
*/
#include <stdio.h>


typedef unsigned int u32;
u32 rever(u32 num)
{
u32 index = 1;
u32 ver = 0,loop = 0,result=0;
for (loop = 0;loop < 32;loop++)
{
index = 1 << loop;
ver = num & index;
if (ver != 0)
{
result |= (1 << (31 - loop));
}
}


return result;
}


int main()
{
u32 n = 0;
printf("input your num:");
scanf("%x",&n);
printf("\n rever(n) = 0x%x\n",rever(n));
return 0;

}


经过实际测试,对于usigned int类型完全符合预期。但是对于 < 0的int类型,暂时还没有实现

#include "reg52.h" typedef unsigned int u16; //对系统默认数据类型进行重定义 typedef unsigned char u8; //定义ULN2003控制步进电机管脚 sbit IN1_A=P1^0; sbit IN2_B=P1^1; sbit IN3_C=P1^2; sbit IN4_D=P1^3; //定义独立按键控制脚 sbit KEY1=P3^1; sbit KEY2=P3^0; sbit KEY3=P3^2; sbit KEY4=P3^3; //使用宏定义独立按键按下的键值 #define KEY1_PRESS 1 #define KEY2_PRESS 2 #define KEY3_PRESS 3 #define KEY4_PRESS 4 #define KEY_UNPRESS 0 // 定义步进电机速度,值越小,速度越快 // 最小不能小于1 #define STEPMOTOR_MAXSPEED 1 #define STEPMOTOR_MINSPEED 5 /******************************************************************************* * 函 数 名 : delay_10us * 函数功能 : 延时函数,ten_us=1时,大约延时10us * 输 入 : ten_us * 输 出 : 无 *******************************************************************************/ void delay_10us(u16 ten_us) { while(ten_us--); } /******************************************************************************* * 函 数 名 : delay_ms * 函数功能 : ms延时函数,ms=1时,大约延时1ms * 输 入 : ten_us * 输 出 : 无 *******************************************************************************/ void delay_ms(u16 ms) { u16 i,j; for(i=ms;i>0;i--) for(j=110;j>0;j--); } /******************************************************************************* * 函 数 名 : step_motor_28BYJ48_send_pulse * 函数功能 : 输出一个数据给ULN2003从而实现向步进电机发送一个脉冲 * 输 入 : step:指定步进序号,可选值0~7 dir:方向选择,1:顺时针,0:逆时针 * 输 出 : 无 *******************************************************************************/ void step_motor_28BYJ48_send_pulse(u8 step,u8 dir) { u8 temp=step; if(dir==0) //如果为逆时针旋 temp=7-step;//调换节拍信号 switch(temp)//8个节拍控制:A->AB->B->BC->C->CD->D->DA { case 0: IN1_A=1;IN2_B=0;IN3_C=0;IN4_D=0;break; case 1: IN1_A=1;IN2_B=1;IN3_C=0;IN4_D=0;break; case 2: IN1_A=0;IN2_B=1;IN3_C=0;IN4_D=0;break; case 3: IN1_A=0;IN2_B=1;IN3_C=1;IN4_D=0;break; case 4: IN1_A=0;IN2_B=0;IN3_C=1;IN4_D=0;break; case 5: IN1_A=0;IN2_B=0;IN3_C=1;IN4_D=1;break; case 6: IN1_A=0;IN2_B=0;IN3_C=0;IN4_D=1;break; case 7: IN1_A=1;IN2_B=0;IN3_C=0;IN4_D=1;break; default: IN1_A=0;IN2_B=0;IN3_C=0;IN4_D=0;break;//停止相序 } } /******************************************************************************* * 函 数 名 : key_scan * 函数功能 : 检测独立按键是否按下,按下则返回对应键值 * 输 入 : mode=0:单次扫描按键 mode=1:连续扫描按键 * 输 出 : KEY1_PRESS:K1按下 KEY2_PRESS:K2按下 KEY3_PRESS:K3按下 KEY4_PRESS:K4按下 KEY_UNPRESS:未有按键按下 *******************************************************************************/ u8 key_scan(u8 mode) { static u8 key=1; if(mode)key=1;//连续扫描按键 if(key==1&&(KEY1==0||KEY2==0||KEY3==0||KEY4==0))//任意按键按下 { delay_10us(1000);//消抖 key=0; if(KEY1==0) return KEY1_PRESS; else if(KEY2==0) return KEY2_PRESS; else if(KEY3==0) return KEY3_PRESS; else if(KEY4==0) return KEY4_PRESS; } else if(KEY1==1&&KEY2==1&&KEY3==1&&KEY4==1) //无按键按下 { key=1; } return KEY_UNPRESS; } /******************************************************************************* * 函 数 名 : main * 函数功能 : 主函数 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void main() { u8 key=0; u8 dir=0;//默认逆时针方向 u8 speed=STEPMOTOR_MAXSPEED;//默认最大速度旋 u8 step=0; while(1) { key=key_scan(0); if(key==KEY1_PRESS)//换向 { dir=!dir; } else if(key==KEY2_PRESS)//加速 { if(speed>STEPMOTOR_MAXSPEED) speed-=1; } else if(key==KEY3_PRESS)//减速 { if(speed<STEPMOTOR_MINSPEED) speed+=1; } step_motor_28BYJ48_send_pulse(step++,dir); if(step==8)step=0; delay_ms(speed); } }将该代码与以下代码结合,去掉按键控制,使dsb20的温度控制步进电机选择,若温度大于25步进电机顺时针旋,温度小于25就逆时针旋,#include "reg52.h" #include "intrins.h" // 定义数据类型 typedef unsigned char u8; typedef unsigned int u16; // 定义端口 sbit LSA = P2^2; sbit LSB = P2^3; sbit LSC = P2^4; sbit DS18B20_PORT = P3^7; #define SMG_A_DP_PORT P0 // 共阳极数码管显示0~F的段码数据 u8 gsmg_code[17] = {0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x07, 0x7f, 0x6f, 0x77, 0x7c, 0x39, 0x5e, 0x79, 0x71}; // 函数声明(添加分号) void smg_display(u8 dat[], u8 pos); void ds18b20_reset(void); u8 ds18b20_check(void); u8 ds18b20_read_bit(void); u8 ds18b20_read_byte(void); void ds18b20_write_byte(u8 dat); void ds18b20_start(void); u8 ds18b20_init(void); float ds18b20_read_temperture(void); void delay_10us(u16 ten_us); void delay_ms(u16 ms); // 主函数 void main() { u8 i = 0; int temp_value; u8 temp_buf[5]; ds18b20_init(); // 初始化DS18B20 while(1) { i++; if(i%50 == 0) // 间隔一段时间读取温度值,间隔时间要大于温度传感器换温度时间 temp_value = ds18b20_read_temperture() * 10; // 保留温度值小数点后一 if(temp_value < 0) // 负温度 { temp_value = -temp_value; temp_buf[0] = 0x40; // 显示负号 } else temp_buf[0] = 0x00; // 不显示 temp_buf[1] = gsmg_code[temp_value/1000]; // 百 temp_buf[2] = gsmg_code[temp_value%1000/100]; // 十 temp_buf[3] = gsmg_code[temp_value%1000%100/10]; // 个 temp_buf[4] = gsmg_code[temp_value%10] | 0x80; // 显示小数点 smg_display(temp_buf, 1); } } // 动态数码管显示 void smg_display(u8 dat[], u8 pos) { u8 i = 0; u8 pos_temp = pos - 1; for(i = pos_temp; i < pos_temp + 5; i++) // 只显示5数码管 { switch(i) // 选 { case 0: LSC = 1; LSB = 1; LSA = 1; break; case 1: LSC = 1; LSB = 1; LSA = 0; break; case 2: LSC = 1; LSB = 0; LSA = 1; break; case 3: LSC = 1; LSB = 0; LSA = 0; break; case 4: LSC = 0; LSB = 1; LSA = 1; break; case 5: LSC = 0; LSB = 1; LSA = 0; break; case 6: LSC = 0; LSB = 0; LSA = 1; break; case 7: LSC = 0; LSB = 0; LSA = 0; break; } SMG_A_DP_PORT = dat[i - pos_temp]; // 传输段选数据 delay_10us(100); // 延时一段时间,等待显示稳定 SMG_A_DP_PORT = 0x00; // 消隐 } } // 延时函数,ten_us=1时,大约延时10us void delay_10us(u16 ten_us) { while(ten_us--); } // ms延时函数,ms=1时,大约延时1ms void delay_ms(u16 ms) { u16 i, j; for(i = ms; i > 0; i--) for(j = 110; j > 0; j--); } // DS18B20相关函数 // 复DS18B20 void ds18b20_reset(void) { DS18B20_PORT = 0; // 拉低DQ delay_10us(75); // 拉低750us DS18B20_PORT = 1; // DQ=1 delay_10us(2); // 20US } // 检查DS18B20是否存在 u8 ds18b20_check(void) { u8 time_temp = 0; while(DS18B20_PORT && time_temp < 20) // 等待DQ为低电平 { time_temp++; delay_10us(1); } if(time_temp >= 20) return 1; // 如果超时则强制返回1 else time_temp = 0; while((!DS18B20_PORT) && time_temp < 20) // 等待DQ为高电平 { time_temp++; delay_10us(1); } if(time_temp >= 20) return 1; // 如果超时则强制返回1 return 0; } // 从DS18B20读取一个 u8 ds18b20_read_bit(void) { u8 dat = 0; DS18B20_PORT = 0; _nop_(); _nop_(); DS18B20_PORT = 1; _nop_(); _nop_(); // 该段时间不能过长,必须在15us内读取数据 if(DS18B20_PORT) dat = 1; // 如果总线上为1则数据dat为1,否则为0 else dat = 0; delay_10us(5); return dat; } // 从DS18B20读取一个字节 u8 ds18b20_read_byte(void) { u8 i = 0; u8 dat = 0; u8 temp = 0; for(i = 0; i < 8; i++) // 循环8次,每次读取一,且先读低再读高 { temp = ds18b20_read_bit(); dat = (temp << 7) | (dat >> 1); } return dat; } // 写一个字节到DS18B20 void ds18b20_write_byte(u8 dat) { u8 i = 0; u8 temp = 0; for(i = 0; i < 8; i++) // 循环8次,每次写一,且先写低再写高 { temp = dat & 0x01; // 选择低准备写入 dat >>= 1; // 将次高移到低 if(temp) { DS18B20_PORT = 0; _nop_(); _nop_(); DS18B20_PORT = 1; delay_10us(6); } else { DS18B20_PORT = 0; delay_10us(6); DS18B20_PORT = 1; _nop_(); _nop_(); } } } // 开始温度换 void ds18b20_start(void) { ds18b20_reset(); // 复 ds18b20_check(); // 检查DS18B20 ds18b20_write_byte(0xcc); // SKIP ROM ds18b20_write_byte(0x44); // 换命令 } // 初始化DS18B20的IO口DQ同时检查DS的存在 u8 ds18b20_init(void) { ds18b20_reset(); return ds18b20_check(); } // 从ds18b20得到温度值 float ds18b20_read_temperture(void) { float temp; u8 dath = 0; u8 datl = 0; u16 value = 0; ds18b20_start(); // 开始换 ds18b20_reset(); // 复 ds18b20_check(); ds18b20_write_byte(0xcc); // SKIP ROM ds18b20_write_byte(0xbe); // 读存储器 datl = ds18b20_read_byte(); // 低字节 dath = ds18b20_read_byte(); // 高字节 value = (dath << 8) + datl; // 合并为16数据 if((value & 0xf800) == 0xf800) // 判断符号,负温度 { value = (~value) + 1; // 数据取反再加1 temp = value * (-0.0625); // 乘以精度 } else // 正温度 { temp = value * 0.0625; } return temp; }
最新发布
05-29
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值