if ("displayMode".equals(mergeDocType)||"printMode".equals(mergeDocType)) {//如果是打印模板或者是显示模板 writeLog("wyh---mergeDocType"+mergeDocType+"-------mergeDocType"); int modeid = rs.getInt("printMergeMould"); writeLog("wyh----modeid"+modeid); if(modeid > 0) { Map<String, Object> params = new HashMap<String, Object>(); params.put("requestid", requestId); params.put("modeid", modeid); params.put("onlyHtml", 0); params.put("keepsign", 0); LOG.info("modeImageParams=" + params); writeLog("wyh---"+params); Map<String, String> modeImage = wfdocUtil.getFileId(params, user); LOG.info("modeImageResult=" + modeImage); writeLog("wyh---"+modeImage); int printImagefileid = Util.getIntValue(modeImage.get("pdf"), 0); if (printImagefileid > 0) { mergeDocList.add(Util.null2String(printImagefileid)); } } } 给 modeImage 加个try cath

时间: 2024-02-14 07:09:14 浏览: 149
好的,我理解你的意思了。你的建议是在这段代码中加入 try-catch 块,以处理可能出现的异常。那我可以为你提供一些示例代码,帮助你更好地实现这个功能。请注意,这里的代码仅供参考,你需要根据自己的具体需求进行修改和调整: ``` if ("displayMode".equals(mergeDocType) || "printMode".equals(mergeDocType)) { writeLog("wyh---mergeDocType" + mergeDocType + "-------mergeDocType"); int modeid = rs.getInt("printMergeMould"); writeLog("wyh----modeid" + modeid); if (modeid > 0) { Map<String, Object> params = new HashMap<String, Object>(); params.put("requestid", requestId); params.put("modeid", modeid); params.put("onlyHtml", 0); params.put("keepsign", 0); LOG.info("modeImageParams=" + params); writeLog("wyh---" + params); try { Map<String, String> modeImage = wfdocUtil.getFileId(params, user); LOG.info("modeImageResult=" + modeImage); writeLog("wyh---" + modeImage); int printImagefileid = Util.getIntValue(modeImage.get("pdf"), 0); if (printImagefileid > 0) { mergeDocList.add(Util.null2String(printImagefileid)); } } catch (Exception e) { // 处理异常情况 // 这里可以记录日志、发送邮件等操作 // 也可以选择忽略异常,不影响程序的正常运行 writeLog("wyh---modeImage获取异常:" + e.getMessage()); } } } ``` 在上面的代码中,我在获取 modeImage 的过程中加入了 try-catch 块,用来处理可能出现的异常。当代码运行出现异常时,会执行 catch 块中的代码,你可以在这里记录异常信息、发送邮件通知等操作,以便及时发现和解决问题。
阅读全文

相关推荐

#include "key.h" #include "dht11.h" DisplayMode currentMode = DISPLAY_AUTO; // 显示模式 volatile uint8_t keyPressed = 0; volatile uint8_t updateDisplayFlag = 1; /******************************************************************************* * 函 数 名 : key_init * 函数功能 : 外部中断初始化 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void key_init(void) { GPIO_InitTypeDef GPIO_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; EXTI_InitTypeDef EXTI_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO | RCC_APB2Periph_GPIOA, ENABLE); // AFIO的时钟打开 // 配置PA11为上拉输入 GPIO_InitStructure.GPIO_Pin = KEY_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; GPIO_Init(KEY_PORT, &GPIO_InitStructure); // 配置外部中断源 GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource12); // PA12(KEY1)中断源 NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn; //EXTI15中断通道(10~15引脚共用一个中断号) NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2; NVIC_InitStructure.NVIC_IRQChannelSubPriority =1; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); //初始化EXTI外设 EXTI_InitStructure.EXTI_Line =EXTI_Line12; EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; // 按键按下触发中断,1—>0,取下降沿 EXTI_InitStructure.EXTI_LineCmd = ENABLE; EXTI_Init(&EXTI_InitStructure); } /******************************************************************************* * 函 数 名 : EXTI11_IRQHandler * 函数功能 : 外部中断12的中断服务函数 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void EXTI15_10_IRQHandler(void) { static u8 i; if(EXTI_GetITStatus(EXTI_Line12) == SET) { delay_ms(20); // 需要确保delay_ms可以在中断中使用 keyPressed=1; if(++i>2) i=0; } switch(i) { case 0 :printf("Switched to AUTO mode\r\n"); currentMode=DISPLAY_AUTO;dht11.state = WORK;ldr.state = WORK;break; case 1 : printf("Switched to TEMP mode\r\n"); currentMode=DISPLAY_TEMP;dht11.state = WORK;ldr.state = SUSPEND;break; default:printf("Switched to LIGHT mode\r\n"); currentMode=DISPLAY_LIGHT;ldr.state = WORK;dht11.state = SUSPEND;break; } EXTI_ClearITPendingBit(EXTI_Line12); } // 处理按键事件 void HandleKeyPress(void) { if(keyPressed) { keyPressed = 0; // 循环切换模式 // currentMode = (currentMode + 1) % 3; updateDisplayFlag = 1; // 设置更新标志 // 打印当前模式 switch(currentMode) { case DISPLAY_TEMP: dsp.interface = TEMP; // 默认显示温度 break; case DISPLAY_LIGHT: dsp.interface = LIGHT; break; case DISPLAY_AUTO: // 轮询DHT11的状态 if(dht11.state == WORK) { if(dht11.getData(&dht11.temp, &dht11.humi) == SUCCESS) // DHT11温湿度数据更新 { printf("temperature:%d℃ humidity(RH):%d \r\n", dht11.temp, dht11.humi); dsp.interface = TEMP; dht11.onTempChange(dht11.temp_threshold); // 温度超过阈值的处理 } dht11.state = STANDBY; } // 轮询LDR的状态 if(ldr.state == WORK) { ldr.light = ldr.getData(); printf("light intensity:%d \r\n", ldr.light); dsp.interface = LIGHT; ldr.onLightChange(ldr.high_threshold,ldr.low_threshold); // 照度超过阈值的处理 ldr.state = STANDBY; } break; } } } void UpdateDisplay(void) { // // 根据当前模式更新显示 // if(updateDisplayFlag) // { // updateDisplayFlag = 0; switch(currentMode) { case DISPLAY_AUTO: // 轮询DHT11的状态 break; case DISPLAY_TEMP: // 温度模式只更新温度 if(dht11.state == WORK) { if(dht11.getData(&dht11.temp, &dht11.humi) == SUCCESS) { printf("Temperature: %d℃\r\n", dht11.temp); dht11.onTempChange(dht11.temp_threshold); } dht11.state = STANDBY; } break; case DISPLAY_LIGHT: // 光照模式只更新光照 if(ldr.state == WORK) { ldr.light = ldr.getData(); printf("Light intensity: %d\r\n", ldr.light); ldr.onLightChange(ldr.high_threshold, ldr.low_threshold); ldr.state = STANDBY; } break; } // 更新数码管显示 switch(dsp.interface) { case LIGHT: setLightValue(&dsp, 1); break; case TEMP: setTempValue(&dsp, 1); break; case HUMI: setHumiValue(&dsp, 1); break; default: break; } // } } 在数码管显示界面时,当选择AUTO,数码管要轮询温度界面和光度界面,同时更新串口打印,用stm32f103c8t6标准库写,不用HAL

// ui.h typedef enum { MODE_LIST, // 商品列表模式 MODE_PURCHASE // 购买模式 } DisplayMode; typedef struct goods { char name[15]; uint8_t price; uint8_t inventory; uint8_t quantity; // 新增:购买数量 bool used; struct goods *next; } GS, *P_GS; typedef struct { DisplayMode current_mode; // 当前模式(列表/购买) uint8_t current_line; // 当前选中行 uint8_t current_page; // 当前页码 uint8_t page_size; // 每页行数 uint8_t total_pages; // 总页数 DisplayState state; // 显示状态(正常/高亮/刷新) } DisplayHandle; extern DisplayHandle display; // key.h typedef enum { KEY_NONE, KEY_ENTER_PURCHASE, // 按键1:进入/退出购买模式 KEY_UP_ADD, // 按键2:上选择/增加数量 KEY_DOWN_SUB, // 按键3:下选择/减少数量 KEY_CONFIRM_PAY // 按键4:确认支付 } KeyEvent; // key.c KeyEvent Key_Process_Mode(KeyHandle *key, GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) { KeyEvent event = KEY_NONE; key->pin_state = GPIO_ReadInputDataBit(GPIOx, GPIO_Pin); switch (key->state) { case KEY_STATE_IDLE: if (key->pin_state == 0) { key->state = KEY_STATE_DEBOUNCE; key->debounce_timer = 0; } break; case KEY_STATE_DEBOUNCE: key->debounce_timer += 10; if (key->debounce_timer >= 20) { if (key->pin_state == 0) { key->state = KEY_STATE_PRESSED; // 根据引脚确定事件类型 if (GPIO_Pin == KEY1_PIN) event = KEY_ENTER_PURCHASE; else if (GPIO_Pin == KEY2_PIN) event = KEY_UP_ADD; else if (GPIO_Pin == KEY3_PIN) event = KEY_DOWN_SUB; else if (GPIO_Pin == KEY4_PIN) event = KEY_CONFIRM_PAY; } else { key->state = KEY_STATE_IDLE; } } break; case KEY_STATE_PRESSED: if (key->pin_state == 1) key->state = KEY_STATE_RELEASE; brea

#include <Wire.h> #include <MAX30105.h> #include <Adafruit_Sensor.h> #include <Adafruit_BMP280.h> #include <OneWire.h> #include <DallasTemperature.h> #include <MPU6050.h> #include <U8g2lib.h> #include <Blinker.h> // ================== Blinker配置 ================== #define BLINKER_WIFI char auth[] = "f09db7fd07d8"; // 设备密钥 char ssid[] = "sodayo"; // WiFi名称 char pswd[] = "12345678"; // WiFi密码 // 新建Blinker组件 BlinkerButton Button1("btn-led"); // 控制LED的按钮 BlinkerNumber HeartRate("num-hr"); // 心率显示 BlinkerNumber Temperature("num-temp"); // 温度显示 BlinkerNumber Pressure("num-pres"); // 气压显示 BlinkerNumber AccelX("num-accx"); // X轴加速度 BlinkerText TextInfo("tex-info"); // 文本信息提示 // ================== 全局控制标志 ================== bool enableUpload = true; // 是否启用数据上传 int displayMode = 0; // 显示模式 0:轮播 1:心率 2:温度 3:气压 4:加速度 // ================== 传感器配置 ================== #define SCREEN_WIDTH 128 #define SCREEN_HEIGHT 64 U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE, 20, 19); MAX30105 particleSensor; Adafruit_BMP280 bmp; OneWire oneWire(15); DallasTemperature sensors(&oneWire); MPU6050 mpu; // 心率计算相关 const byte RATE_SIZE = 4; byte rates[RATE_SIZE]; byte rateSpot = 0; long lastBeat = 0; float beatsPerMinute; int beatAvg = 0; // ================== 函数声明 ================== void initAllSensors(); void initOLED(); void readSensors(); void updateOLED(); void uploadSensorData(); void handleBlinkerCommands(const String & cmd); void buttonCallback(const String & state); void modeSwitchCallback(const String & state); // ================== 传感器初始化 ================== void initAllSensors() { // 初始化MAX30102 Wire.begin(17, 6); if (!particleSensor.begin(Wire, I2C_SPEED_FAST)) { displayError("MAX30102", "初始化失败"); } particleSensor.setup(); particleSensor.setPulseAmplitudeRed(0x0A); // 初始化BMP280 Wire.begin(9, 10); if (!bmp.begin(0x76)) { displayError("BMP280", "初始化失败"); }

完成一个仓储环境监测的模拟系统。具体要求如下: 1. DHT11模块完成温湿度测量,5s更新,数值显示于数码管,同时打印到串口。温度超过阈值触发蜂鸣器短鸣,同时启动风扇小马达; 2. LDR模块完成环境照度测量,4s更新,数值显示于数码管,同时打印到串口。照度超过阈值启动PWM调光(高、中、低三级调光)。 3. 按键循环切换显示模式,按第1下固定显示温度,按第二下固定显示照度,按第三下恢复默认的自动显示模式。按键用外部中断实现。 #include "systick.h" static u8 fac_us=0; //us延时倍乘数 static u16 fac_ms=0; //ms延时倍乘数 //初始化延迟函数 //SYSTICK的时钟固定为AHB时钟的1/8 //SYSCLK:系统时钟频率 void systick_init(u8 SYSCLK) { SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8); fac_us = SYSCLK/8; fac_ms = (u16)fac_us * 1000; } //延时nus //nus为要延时的us数. void delay_us(u32 nus) { u32 temp; SysTick->LOAD=nus*fac_us; //时间加载 SysTick->VAL=0x00; //清空计数器 SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ; //开始倒数 do { temp=SysTick->CTRL; }while((temp&0x01)&&!(temp&(1<<16))); //等待时间到达 SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; //关闭计数器 SysTick->VAL =0X00; //清空计数器 } //延时nms //注意nms的范围 //SysTick->LOAD为24位寄存器,所以,最大延时为: //nms<=0xffffff*8*1000/SYSCLK //SYSCLK单位为Hz,nms单位为ms //对72M条件下,nms<=1864 void delay_ms(u16 nms) { u32 temp; SysTick->LOAD = (u32)nms*fac_ms; //时间加载(SysTick->LOAD为24bit) SysTick->VAL = 0x00; //清空计数器 SysTick->CTRL |= SysTick_CTRL_ENABLE_Msk ; //开始倒数 do { temp=SysTick->CTRL; } while((temp&0x01) && !(temp&(1<<16))); //等待时间到达 SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk; //关闭计数器 SysTick->VAL = 0x00; //清空计数器 } #include "display.h" DisplayStruct dsp; void segDisplay(void); // 共阴极数码管,阴极位选(0有效),阳极段选(1有效) const u16 segCode[11] = { /* 0 1 2 3 4 5 6 7 */ 0x003F, 0x0006, 0x005B, 0x004F, 0x0066, 0x006D, 0x007D, 0x0007, /* 8 9 off */ 0x007F, 0x006F, 0x0000 }; /******************************************************************************* * @brief 初始化数码管用到的GPIO端口和dsp结构体 * @param None * @retval None *******************************************************************************/ void display_init(void) { GPIO_InitTypeDef GPIO_InitStructure = {0}; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB, ENABLE); // PA0~PA7: 段选(分别连接引脚A~G); // PB12~PB15:位选(分别连接引脚D1~D4) GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOB, &GPIO_InitStructure); dsp.currDigit = 1; // dsp结构体的初始化 dsp.SegDisplay = segDisplay; } /*********************************************************************************** * @brief 设置数码管要显示的4位数值。 * @param dsp:数码管结构体指针;value[4]:4位数值;dotBit:小数点位置(没有小数点则置0) * @retval None ***********************************************************************************/ void setValue(DisplayStruct *dsp, u8 value[4], u8 dotBit) { dsp->digit[0] = value[0]; dsp->digit[1] = value[1]; dsp->digit[2] = value[2]; dsp->digit[3] = value[3]; dsp->dotDigit = dotBit; } void setLightValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 1; // 参数编号 dsp->digit[1] = 10; // 第二位空缺 dsp->digit[2] = ldr.light/10; dsp->digit[3] = ldr.light%10; // dsp->dotDigit = dotBit; // 小数点位置 } void setTempValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 2; dsp->digit[1] = 10; dsp->digit[2] = dht11.temp/10; dsp->digit[3] = dht11.temp%10; dsp->dotDigit = dotBit; } void setHumiValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 3; dsp->digit[1] = 10; dsp->digit[2] = dht11.humi/10; dsp->digit[3] = dht11.humi%10; dsp->dotDigit = dotBit; } /******************************************************************************* * @简 介 段选,让某一位数码管显示指定的数字 * @输入参数 value:段码数组的序号(0~9),代表要显示的数字 * @retval * 备注:只操作PA的低8位,不要影响高8位。 ********************************************************************************/ void seg(uint8_t value) { if(value < sizeof(segCode)) // 避免数组溢出 { GPIOA->ODR &= 0xFF00; GPIOA->ODR |= segCode[value]; } } /******************************************************************************** * 简 介 位选,一次只能选一位。 * 输入参数 com:位数(取值1~4,从左到右) * 返回值 无 * 备注:只操作PB的高4位(置0),不要影响其他位。 ********************************************************************************/ void com(uint8_t currDigit) { // 数码管位选(0有效) GPIOB->ODR |= 0xF000; GPIOB->ODR &= ~(0x1000<<(currDigit-1)); } /******************************************************************************** * 简介 在数码管上逐位显示其段码(由Timer中断服务函数调用) * 参数 无 * 返回值 无 ********************************************************************************/ void segDisplay(void) { seg(10); // 消隐之前的显示内容 com(dsp.currDigit); // 位选,从COM1到COM4逐次移位,实现动态显示 if(dsp.currDigit != dsp.dotDigit) // 当前位不显示小数点 seg(dsp.digit[dsp.currDigit-1]); else // 当前位要显示小数点 { GPIO_Write(GPIOA, segCode[dsp.digit[dsp.currDigit-1]]| 0x0080); //GPIOA->ODR &= 0xFF00; //GPIOA->ODR |= (segCode[dsp.currDigit-1]|0x0080); } if(++dsp.currDigit > DSP_DIGIT) // 从当前位右移到下一位 dsp.currDigit = 1; } #include "display.h" DisplayStruct dsp; void segDisplay(void); // 共阴极数码管,阴极位选(0有效),阳极段选(1有效) const u16 segCode[11] = { /* 0 1 2 3 4 5 6 7 */ 0x003F, 0x0006, 0x005B, 0x004F, 0x0066, 0x006D, 0x007D, 0x0007, /* 8 9 off */ 0x007F, 0x006F, 0x0000 }; /******************************************************************************* * @brief 初始化数码管用到的GPIO端口和dsp结构体 * @param None * @retval None *******************************************************************************/ void display_init(void) { GPIO_InitTypeDef GPIO_InitStructure = {0}; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB, ENABLE); // PA0~PA7: 段选(分别连接引脚A~G); // PB12~PB15:位选(分别连接引脚D1~D4) GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOB, &GPIO_InitStructure); dsp.currDigit = 1; // dsp结构体的初始化 dsp.SegDisplay = segDisplay; } /*********************************************************************************** * @brief 设置数码管要显示的4位数值。 * @param dsp:数码管结构体指针;value[4]:4位数值;dotBit:小数点位置(没有小数点则置0) * @retval None ***********************************************************************************/ void setValue(DisplayStruct *dsp, u8 value[4], u8 dotBit) { dsp->digit[0] = value[0]; dsp->digit[1] = value[1]; dsp->digit[2] = value[2]; dsp->digit[3] = value[3]; dsp->dotDigit = dotBit; } void setLightValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 1; // 参数编号 dsp->digit[1] = 10; // 第二位空缺 dsp->digit[2] = ldr.light/10; dsp->digit[3] = ldr.light%10; // dsp->dotDigit = dotBit; // 小数点位置 } void setTempValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 2; dsp->digit[1] = 10; dsp->digit[2] = dht11.temp/10; dsp->digit[3] = dht11.temp%10; dsp->dotDigit = dotBit; } void setHumiValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 3; dsp->digit[1] = 10; dsp->digit[2] = dht11.humi/10; dsp->digit[3] = dht11.humi%10; dsp->dotDigit = dotBit; } /******************************************************************************* * @简 介 段选,让某一位数码管显示指定的数字 * @输入参数 value:段码数组的序号(0~9),代表要显示的数字 * @retval * 备注:只操作PA的低8位,不要影响高8位。 ********************************************************************************/ void seg(uint8_t value) { if(value < sizeof(segCode)) // 避免数组溢出 { GPIOA->ODR &= 0xFF00; GPIOA->ODR |= segCode[value]; } } /******************************************************************************** * 简 介 位选,一次只能选一位。 * 输入参数 com:位数(取值1~4,从左到右) * 返回值 无 * 备注:只操作PB的高4位(置0),不要影响其他位。 ********************************************************************************/ void com(uint8_t currDigit) { // 数码管位选(0有效) GPIOB->ODR |= 0xF000; GPIOB->ODR &= ~(0x1000<<(currDigit-1)); } /******************************************************************************** * 简介 在数码管上逐位显示其段码(由Timer中断服务函数调用) * 参数 无 * 返回值 无 ********************************************************************************/ void segDisplay(void) { seg(10); // 消隐之前的显示内容 com(dsp.currDigit); // 位选,从COM1到COM4逐次移位,实现动态显示 if(dsp.currDigit != dsp.dotDigit) // 当前位不显示小数点 seg(dsp.digit[dsp.currDigit-1]); else // 当前位要显示小数点 { GPIO_Write(GPIOA, segCode[dsp.digit[dsp.currDigit-1]]| 0x0080); //GPIOA->ODR &= 0xFF00; //GPIOA->ODR |= (segCode[dsp.currDigit-1]|0x0080); } if(++dsp.currDigit > DSP_DIGIT) // 从当前位右移到下一位 dsp.currDigit = 1; } #include "usart.h" int fputc(int ch,FILE *p) //函数默认的,在使用printf函数时自动调用 { USART_SendData(USART1,(u8)ch); while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET); return ch; } //串口1中断服务程序 //注意,读取USARTx->SR能避免莫名其妙的错误 u8 USART1_RX_BUF[USART1_REC_LEN]; //接收缓冲,最大USART_REC_LEN个字节. //接收状态 //bit15, 接收完成标志 //bit14, 接收到0x0d //bit13~0, 接收到的有效字节数目 u16 USART1_RX_STA=0; //接收状态标记 /******************************************************************************* * 函 数 名 : USART1_Init * 函数功能 : USART1初始化函数 * 输 入 : bound:波特率 * 输 出 : 无 *******************************************************************************/ void usart1_init(u32 baud) { //GPIO端口设置 GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE); /* 配置GPIO的模式和IO口 */ GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9; //串口输出TX:PA9 GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP; //复用推挽输出 GPIO_Init(GPIOA,&GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10; //串口输入RX:PA10 GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING; //模拟输入 GPIO_Init(GPIOA,&GPIO_InitStructure); //USART1 初始化设置 USART_InitStructure.USART_BaudRate = baud; //波特率设置 USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式 USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位 USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //收发模式 USART_Init(USART1, &USART_InitStructure); //初始化串口1 USART_Cmd(USART1, ENABLE); //使能串口1 USART_ClearFlag(USART1, USART_FLAG_TC); USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启相关中断 //Usart1 NVIC 配置 NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//串口1中断通道 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;//抢占优先级3 NVIC_InitStructure.NVIC_IRQChannelSubPriority =3; //子优先级3 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能 NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器、 } /******************************************************************************* * 函 数 名 : USART1_IRQHandler * 函数功能 : USART1中断函数 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ void USART1_IRQHandler(void) //串口1中断服务程序 { u8 r; if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断 { r =USART_ReceiveData(USART1);//(USART1->DR); //读取接收到的数据 if((USART1_RX_STA&0x8000)==0)//接收未完成 { if(USART1_RX_STA&0x4000)//接收到了0x0d { if(r!=0x0a)USART1_RX_STA=0;//接收错误,重新开始 else USART1_RX_STA|=0x8000; //接收完成了 } else //还没收到0X0D { if(r==0x0d)USART1_RX_STA|=0x4000; else { USART1_RX_BUF[USART1_RX_STA&0X3FFF]=r; USART1_RX_STA++; if(USART1_RX_STA>(USART1_REC_LEN-1))USART1_RX_STA=0;//接收数据错误,重新开始接收 } } } } } #include "dht11.h" #include "beep.h" Dht11Struct dht11 = {0}; void SET_DHT11_IO_OUT(void); void SET_DHT11_IO_IN(void); void DHT11_RequestData(void); u8 DHT11_RespondRequest(void); u8 DHT11_Read_Data(u8 *temp, u8 *humi); void DHT11_OnTempChange(u8 temp_threshold); // DHT11_PIN初始化 // 返回值: 无 extern void dht11_init() { GPIO_InitTypeDef GPIO_InitStructure; // 数据引脚初始化 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // 数据引脚 GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_Init(DHT11_PORT, &GPIO_InitStructure); GPIO_SetBits(DHT11_PORT, DHT11_PIN); // 初始输出状态:高电平 // 结构体初始化 dht11.period = DHT11_PERIOD; dht11.temp_threshold = TEMP_THRESHOLD; dht11.getData = DHT11_Read_Data; // 获取温湿度数据的函数 dht11.onTempChange = DHT11_OnTempChange; // 温度超过阈值的处理函数 // 做一次数据请求测试,检查设备状态 DHT11_RequestData(); if (DHT11_RespondRequest() == SUCCESS) { dht11.state = STANDBY; printf("DHT11 Init OK!\r\n"); } else { dht11.state = FAIL; printf("DHT11 Check Error!\r\n"); } } // 复位DHT11,单片机向DHT11发起数据采集请求 // 时序图的黑线部分 static void DHT11_RequestData() { SET_DHT11_IO_OUT(); // 数据引脚配置为输出模式 DHT11_DQ_OUT = 0; delay_ms(20); // 低电平持续至少18ms DHT11_DQ_OUT = 1; delay_us(30); // 高电平持续20~40us } // DHT11响应单片机的数据请求 // 返回值:SUCCESS or FAILURE static u8 DHT11_RespondRequest() { u8 retry=0; SET_DHT11_IO_IN(); // 数据引脚设为输入模式,接收DHT11的响应 while (DHT11_DQ_IN && retry<100) // 等待输入由高变低 { retry++; if(retry >= 100) return FAILURE; // 等待时间过长,返回异常。 delay_us(1); } retry=0; while (!DHT11_DQ_IN && retry<100) // 低电平持续时间80us { retry++; if(retry >= 100) return FAILURE; // 低电平持续时间过长,返回异常。 delay_us(1); } return SUCCESS; } //DHT11输出模式配置 static void SET_DHT11_IO_OUT() { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 推挽输出模式 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_Init(DHT11_PORT,&GPIO_InitStructure); } //DHT11输入模式配置 static void SET_DHT11_IO_IN() { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; // 浮空输入模式,最小系统板和DHT11之间并未接上拉电阻 GPIO_Init(DHT11_PORT,&GPIO_InitStructure); } //从DHT11读取一位 //返回值:bit 1或0 //注:此函数处并未做高低电平持续时间异常的处理,因DHT11自身有校验和,即便读错1位也不会造成最终数据的错误。 // 严格来讲,除了正常情况下返回“1”或“0”,还应增加异常情况的返回值(比如“2”)。 static u8 DHT11_Read_Bit(void) { u8 retry=0; while(DHT11_DQ_IN && retry<100) // 等待输入电平由高变低(低电平持续50us) { retry++; delay_us(1); } retry=0; while(!DHT11_DQ_IN && retry<100) // 等待输入电平由低变高 { retry++; delay_us(1); } delay_us(40); // 等待40us(高电平持续26~28us表示0,持续70us表示1) if(DHT11_DQ_IN) return 1; // 40us后如果输入仍为高电平,则表示读入1;否则表示读入0。 else return 0; } //从DHT11读取一个字节 //返回值:读到的8位数据 static u8 DHT11_Read_Byte(void) { u8 i,byte; byte = 0; for (i=0;i<8;i++) { byte <<= 1; // 先前读取的数据(不足8位)全部左移一位 byte |= DHT11_Read_Bit(); // 最低位填入新读取的1位 } return byte; } //从DHT11读取一次完整的数据 //temp:温度值(整数,范围:0~50°) //humi:湿度值(整数,范围:20%~90%) //返回值:0,正常; 1,失败 static u8 DHT11_Read_Data(u8 *temp, u8 *humi) { u8 buf[5]; u8 i; DHT11_RequestData(); if(DHT11_RespondRequest() == SUCCESS) { for(i=0;i<5;i++) // 读取5组共40位数据 { buf[i] = DHT11_Read_Byte(); } if((buf[0]+buf[1]+buf[2]+buf[3]) == buf[4]) // 数据校验 { *humi = buf[0]; // 湿度整数部分 *temp = buf[2]; // 温度整数部分 return SUCCESS; } } return FAILURE; } // 采集DHT11数据并打印至串口 void dht11DataCollect() { u8 temp; u8 humi; if (DHT11_Read_Data(&temp, &humi) == SUCCESS) printf("temperature:%d℃ humidity(RH):%d \r\n", temp, humi); // 输出到串口(重定向) else printf("DHT11 data error! \r\n"); } // 温度阈值处理 void DHT11_OnTempChange(u8 temp_threshold) { if (dht11.temp > temp_threshold) { // 启动风扇(低电平触发) } else { // 关闭风扇 } } #include "ldr.h" LdrStruct ldr; static u8 getLightIntensity(void); static void OnLightChange(u8 hiLight, u8 loLight); /******************************************************************************* * 函 数 名 : adc_init * 函数功能 : ADC外设的初始化 * 输 入 : 无 * 输 出 : 无 *******************************************************************************/ static void adc_init(void) { GPIO_InitTypeDef GPIO_InitStructure; ADC_InitTypeDef ADC_InitStructure; // 1.打开相关外设的总线时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_ADC1,ENABLE); // 2.信号引脚的参数配置 GPIO_InitStructure.GPIO_Pin=LDR_PIN; // 信号引脚:PB0 GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AIN; // 设置模拟输入模式 GPIO_InitStructure.GPIO_Speed=GPIO_Speed_2MHz; // 设置传输速率 GPIO_Init(LDR_PORT,&GPIO_InitStructure); // 3. 输入时钟降频(<14MHz) RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 分频因子6,输入时钟为72M/6=12MHz // 4. 初始化ADC参数 ADC_InitStructure.ADC_Mode=ADC_Mode_Independent; //独立模式 ADC_InitStructure.ADC_ScanConvMode=DISABLE; //单次扫描 ADC_InitStructure.ADC_ContinuousConvMode=DISABLE; //单次转换 ADC_InitStructure.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None; //软件触发 ADC_InitStructure.ADC_DataAlign=ADC_DataAlign_Right; //数据右对齐 ADC_InitStructure.ADC_NbrOfChannel=1; //只有1个通道 ADC_Init(ADC1,&ADC_InitStructure); // 5. 使能ADC ADC_Cmd(ADC1,ENABLE); // 6. ADC校准 ADC_ResetCalibration(ADC1); //复位校准 while(ADC_GetResetCalibrationStatus(ADC1)); ADC_StartCalibration(ADC1); //开启并完成校准 while(ADC_GetCalibrationStatus(ADC1)); } // LDR的设备初始化 extern void ldr_init() { adc_init(); ldr.period = LDR_PERIOD; ldr.high_threshold = HI_THRESHOLD; ldr.low_threshold = LO_THRESHOLD; ldr.getData = getLightIntensity; ldr.onLightChange = OnLightChange; ldr.state = STANDBY; } /******************************************************************************* * 函 数 名 : GET_ADC_Value * 函数功能 : 获取通道ch的转换值,测量times次,取平均值 * 输 入 : ch:通道编号,Rank:规则序列中的第几个转换,取值1~16; times:测量次数 * 输 出 : 通道ch的times次转换结果的平均值 *******************************************************************************/ static u16 Get_ADC_Value(u8 ch, u8 times) //获取ADC1通道ch的转换值 { u8 i; u32 Temp_val=0; ADC_RegularChannelConfig(ADC1,ch,1,ADC_SampleTime_239Cycles5); for(i=0;i<times;i++) { ADC_SoftwareStartConvCmd(ADC1,ENABLE); // 开始转换 while(!ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)); // 等待至单次转换结束 Temp_val += ADC_GetConversionValue(ADC1); delay_ms(10); } return Temp_val/times; } /******************************************************************************* * 函 数 名 : getLightIntensity * 函数功能 : 将ADC转换值解读为光照强度 * 输 入 : 无 * 输 出 : 光照照度值(0~100, 0:照度最低;100:照度最高) *******************************************************************************/ static u8 getLightIntensity(void) //通过ADC1 通道0的值获取亮度值 { u16 value = 0; u8 lightvalue = 0; value = Get_ADC_Value(ADC_Channel_8,20); lightvalue = 100 - (u16)(value/40.95); return lightvalue; } // 添加PWM控制LED功能 static void set_led_brightness(u8 level) { switch(level) { case 0: // 低档 TIM_SetCompare2(TIM3, 125); // 25%占空比 break; case 1: // 中档 TIM_SetCompare2(TIM3, 250); // 50%占空比 break; case 2: // 高档 TIM_SetCompare2(TIM3, 375); // 75%占空比 break; } } /******************************************************************************* * 函 数 名 : OnLightChange * 函数功能 : 照度超过阈值(高阈值和低阈值)的处理措施 * 输 入 : hiLight-高阈值; loLight-低阈值 * 输 出 : 无 *******************************************************************************/ static void OnLightChange(u8 hiLight, u8 loLight) { if (ldr.light > hiLight) { set_led_brightness(0);// 照度过高,降低亮度 } else if (ldr.light < loLight) { set_led_brightness(2); // 照度过低,提升亮度 } else // 正常范围 { set_led_brightness(1); // 中等亮度 } } #include "key.h" #include "pwm.h" /******************************************************************************* * 函 数 名 : TIM3_CH2_PWM_Init * 函数功能 : TIM3通道2 PWM初始化函数 * 输 入 : per:重装载值 * psc:分频系数 * 输 出 : 无 *******************************************************************************/ void TIM3_CH2_PWM_Init(u16 per,u16 psc) { TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure; TIM_OCInitTypeDef TIM_OCInitStructure; GPIO_InitTypeDef GPIO_InitStructure; /* 开启时钟 */ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO,ENABLE); // PB5引脚启用复用模式 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE); // 选择TIM3部分重映射 GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3,ENABLE); /* PB5作为PWM的输出引脚 */ GPIO_InitStructure.GPIO_Pin=GPIO_Pin_5; GPIO_InitStructure.GPIO_Speed=GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP; //复用推挽输出 GPIO_Init(GPIOB,&GPIO_InitStructure); // 初始化TIM3 TIM_TimeBaseInitStructure.TIM_Period=per; //自动装载值 TIM_TimeBaseInitStructure.TIM_Prescaler=psc; //分频系数 TIM_TimeBaseInitStructure.TIM_ClockDivision=TIM_CKD_DIV1; TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up; //设置向上计数模式 TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure); // 设置TIM3_CH2的PWM模式,使能CH2输出,呈现出PPT展示的PWM波形 // PWM1即mode1,先输出有效电平,再输出无效电平;PWM2即mode2则正好相反。 TIM_OCInitStructure.TIM_OCMode=TIM_OCMode_PWM1; // 设置有效电平为低电平(此案例中低电平点亮D2) TIM_OCInitStructure.TIM_OCPolarity=TIM_OCPolarity_Low; TIM_OCInitStructure.TIM_OutputState=TIM_OutputState_Enable; TIM_OC2Init(TIM3,&TIM_OCInitStructure); //输出比较通道2初始化 TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Enable); //使能TIM3在 CCR2 上的预装载寄存器 TIM_Cmd(TIM3,ENABLE); //使能定时器 } 数码管是4位共阴极,段选连PA0~7,位选PB12~15,STM32F103C8T6,用标准库

好的,用户问的是如何在Avalonia中实现控件紧凑显示模式,特别是工具面板的样式。首先,我需要回忆一下Avalonia的控件库和相关文档。用户提到使用MVVM模式、分屏视图控件和紧凑显示模式设置。可能需要用到SplitView控件,因为分屏视图通常用于侧边栏或面板的展开和折叠。 接下来,用户希望在面板关闭时显示一个图标按钮来打开面板。这时候,SplitView的CompactPaneLength属性可能派上用场,这个属性可以设置紧凑模式下的面板宽度,当面板处于紧凑状态时,可能只显示图标而不是完整内容。 然后,我需要考虑如何结合MVVM来实现。可能需要绑定SplitView的IsPaneOpen属性到ViewModel中的一个布尔属性,这样可以通过按钮的点击事件来切换面板的展开和关闭。同时,按钮的图标可能需要根据面板状态变化,比如使用不同的图标表示打开或关闭状态。 另外,用户提到紧凑显示模式,可能需要设置SplitView的DisplayMode属性为CompactInline或CompactOverlay,这样在折叠时面板会以紧凑模式显示,比如只显示图标列,点击后展开。这样在面板关闭时,确实会有更多空间,而图标按钮可以放在紧凑模式的面板区域。 然后,布局方面,SplitView的内容部分可能包含主界面,而Pane部分包含工具面板的内容。当IsPaneOpen为false时,面板折叠,CompactPaneLength决定了折叠后的宽度,这时候可以在Pane中放置一个按钮,点击后打开面板。 还需要注意样式设计,比如按钮的位置、图标的显示,以及面板展开和折叠时的动画效果,可能需要使用Avalonia的动画或过渡效果。同时,数据绑定要正确设置,确保ViewModel中的属性能够正确控制视图的状态。 最后,可能需要处理边缘情况,比如窗口大小变化时如何调整面板的显示模式,或者如何在不同平台上保持一致的外观。可能需要查阅Avalonia的官方文档,确认SplitView的具体用法和属性,确保示例代码的正确性。 总结步骤:使用SplitView控件,设置DisplayMode为CompactInline,绑定IsPaneOpen到ViewModel,在Pane中添加按钮和工具内容,调整CompactPaneLength以适应图标按钮的宽度,并通过命令或事件切换面板状态。 在 Avalonia 中实现工具面板的紧凑显示模式,可以通过以下步骤结合 SplitView 控件和 MVVM 模式实现: xml <Window xmlns="https://github.com/avaloniaui" xmlns:vm="using:YourNamespace.ViewModels"> <Design.DataContext> <vm:MainViewModel /> </Design.DataContext> <SplitView DisplayMode="CompactInline" IsPaneOpen="{Binding IsPanelOpen}" CompactPaneLength="40" OpenPaneLength="200"> <SplitView.Content> <TextBlock Text="主内容区域" VerticalAlignment="Center" HorizontalAlignment="Center"/> </SplitView.Content> <SplitView.Pane> <StackPanel> <StackPanel IsVisible="{Binding !IsPanelOpen}"> <Button Content="🗃️ 工具1" Margin="5"/> <Button Content="📊 工具2" Margin="5"/> <Button Content="⚙️ 设置" Margin="5"/> </StackPanel> <Button Content="▶️" HorizontalAlignment="Right" Command="{Binding TogglePanelCommand}" IsVisible="{Binding !IsPanelOpen}"/> </StackPanel> </SplitView.Pane> </SplitView> </Window> 对应 ViewModel 的实现: csharp // MainViewModel.cs public class MainViewModel : ViewModelBase { private bool _isPanelOpen; public bool IsPanelOpen { get => _isPanelOpen; set => this.RaiseAndSetIfChanged(ref _isPanelOpen, value); } public ICommand TogglePanelCommand { get; } public MainViewModel() { TogglePanelCommand = new RelayCommand(() => { IsPanelOpen = !IsPanelOpen; }); } } 关键实现要点: 1. **SplitView 配置**: - DisplayMode="CompactInline":紧凑模式保留面板空间 - CompactPaneLength="40":紧凑模式下面板宽度 - OpenPaneLength="200":展开后的面板宽度 2. **状态控制**: - 通过绑定 IsPaneOpen 控制面板展开状态 - 使用双向绑定实现按钮与面板状态的联动 3. **视觉切换**: - 面板展开时显示完整工具按钮 - 紧凑模式下显示 ▶️ 展开图标 - 通过 IsVisible 实现不同状态的界面切换 4. **交互逻辑**: - 点击 ▶️ 按钮通过命令切换面板状态 - 面板展开时自动隐藏紧凑模式按钮 扩展功能建议: 1. 添加动画效果: xml <SplitView.Transitions> <Transitions> <DoubleTransition Property="OpenPaneLength" Duration="0:0:0.2"/> </Transitions> </SplitView.Transitions> 2. 响应式布局优化: csharp // 根据窗口宽度自动切换模式 public void UpdateDisplayMode(double windowWidth) { DisplayMode = windowWidth > 600 ? SplitViewDisplayMode.CompactInline : SplitViewDisplayMode.Overlay; } 3. 添加面板关闭按钮: xml <Button Content="❌ 关闭" Command="{Binding TogglePanelCommand}" IsVisible="{Binding IsPanelOpen}" HorizontalAlignment="Right"/> 这种实现方式可以灵活适应不同屏幕尺寸,在保持 MVVM 模式的同时提供良好的用户体验,通过简单的属性绑定即可实现状态控制。更为详细的修改

按键循环切换显示模式,按第1下固定显示温度,按第二下固定显示照度,按第三下恢复自动 * 显示模式。按键用中断实现。#include "display.h" DisplayStruct dsp; void segDisplay(void); // 共阴极数码管,阴极位选(0有效),阳极段选(1有效) const u16 segCode[11] = { /* 0 1 2 3 4 5 6 7 */ 0x003F, 0x0006, 0x005B, 0x004F, 0x0066, 0x006D, 0x007D, 0x0007, /* 8 9 off */ 0x007F, 0x006F, 0x0000 }; /******************************************************************************* * @brief 初始化数码管用到的GPIO端口和dsp结构体 * @param None * @retval None *******************************************************************************/ extern void display_init(void) { GPIO_InitTypeDef GPIO_InitStructure = {0}; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB, ENABLE); // PA0~PA7: 段选(分别连接引脚A~G); // PB12~PB15:位选(分别连接引脚D1~D4) GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(GPIOB, &GPIO_InitStructure); // dsp结构体的初始化 dsp.currDigit = 1; dsp.SegDisplay = segDisplay; } /*********************************************************************************** * @brief 设置数码管要显示的4位数值。 * @param dsp:数码管结构体指针;value[4]:4位数值;dotBit:小数点位置(没有小数点则置0) * @retval None ***********************************************************************************/ //static void setValue(DisplayStruct *dsp, u8 value[4], u8 dotBit) //{ // dsp->digit[0] = value[0]; // dsp->digit[1] = value[1]; // dsp->digit[2] = value[2]; // dsp->digit[3] = value[3]; // dsp->dotDigit = dotBit; //} // 设置光照参数的显示界面 extern void setLightValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 1; // 参数编号 dsp->digit[1] = BLANK_CODE; // 第二位空缺 dsp->digit[2] = ldr.light/10; dsp->digit[3] = ldr.light%10; dsp->dotDigit = dotBit; // 小数点位置 } // 设置温度参数的显示界面 extern void setTempValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 2; dsp->digit[1] = BLANK_CODE; dsp->digit[2] = dht11.temp/10; dsp->digit[3] = dht11.temp%10; dsp->dotDigit = dotBit; } // 设置湿度参数的显示界面 extern void setHumiValue(DisplayStruct *dsp, u8 dotBit) { dsp->digit[0] = 3; dsp->digit[1] = BLANK_CODE; dsp->digit[2] = dht11.humi/10; dsp->digit[3] = dht11.humi%10; dsp->dotDigit = dotBit; } /******************************************************************************* * @简 介 段选,让某一位数码管显示指定的数字 * @输入参数 value:段码数组的序号(0~10),代表要显示的数字0~9,10代表此位黑屏 dotState: 是否带小数点 * @retval * 备注:只操作PA的低8位(置1),不要影响高8位。 ********************************************************************************/ static void seg(uint8_t value, DotState dotState) { if(value < sizeof(segCode)) // 避免数组溢出 { GPIOA->ODR &= 0xFF00; // 带小数点的显示 if (dotState == DOT_OFF) GPIOA->ODR |= segCode[value]; else // 不带小数点的显示 GPIOA->ODR |= (segCode[value]|0x0080); } } /******************************************************************************** * 简 介 位选,一次只能选一位。 * 输入参数 com:位数(取值1~4,从左到右) * 返回值 无 * 备注:只操作PB的高4位(置0),不要影响其他位。 ********************************************************************************/ static void com(uint8_t currDigit) { // 数码管位选(0有效) GPIOB->ODR |= 0xF000; GPIOB->ODR &= ~(0x1000<<(currDigit-1)); } /******************************************************************************** * 简介 在数码管上逐位显示其段码(由Timer中断服务函数调用) * 参数 无 * 返回值 无 ********************************************************************************/ static void segDisplay(void) { seg(BLANK_CODE,DOT_OFF); // 消隐之前的显示内容 com(dsp.currDigit); // 位选,从COM1到COM4逐次移位,实现动态显示 if(dsp.currDigit != dsp.dotDigit) // 当前位不显示小数点 seg(dsp.digit[dsp.currDigit-1],DOT_OFF); else // 当前位要显示小数点 seg(dsp.digit[dsp.currDigit-1],DOT_ON); if(++dsp.currDigit > DSP_DIGIT) // 从当前位右移到下一位 dsp.currDigit = 1; }#include "dht11.h" #include "pwm.h" Dht11Struct dht11 = {0}; u16 PWM_Val = LOW_LIMIT; void SET_DHT11_IO_OUT(void); void SET_DHT11_IO_IN(void); void DHT11_RequestData(void); u8 DHT11_RespondRequest(void); u8 DHT11_Read_Data(u8 *temp, u8 *humi); void DHT11_OnTempChange(u8 temp_threshold); // DHT11设备的初始化 extern void dht11_init() { GPIO_InitTypeDef GPIO_InitStructure; // 数据引脚初始化 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); // 数据引脚 GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_Init(DHT11_PORT, &GPIO_InitStructure); GPIO_SetBits(DHT11_PORT, DHT11_PIN); // 初始输出状态:高电平 // 结构体初始化 dht11.period = DHT11_PERIOD; dht11.temp_threshold = TEMP_THRESHOLD; dht11.getData = DHT11_Read_Data; // 获取温湿度数据的函数 dht11.onTempChange = DHT11_OnTempChange; // 温度超过阈值的处理函数 // 做一次数据请求测试,检查设备状态 DHT11_RequestData(); if (DHT11_RespondRequest() == SUCCESS) { dht11.state = STANDBY; printf("DHT11 Init OK!\r\n"); } else { dht11.state = FAIL; printf("DHT11 Check Error!\r\n"); } } // 复位DHT11,单片机向DHT11发起数据采集请求 // 时序图的黑线部分 static void DHT11_RequestData() { SET_DHT11_IO_OUT(); // 数据引脚配置为输出模式 DHT11_DQ_OUT = 0; delay_ms(20); // 低电平持续至少18ms DHT11_DQ_OUT = 1; delay_us(30); // 高电平持续20~40us } // DHT11响应单片机的数据请求 // 返回值:SUCCESS or FAILURE static u8 DHT11_RespondRequest() { u8 retry=0; SET_DHT11_IO_IN(); // 数据引脚设为输入模式,接收DHT11的响应 while (DHT11_DQ_IN && retry<100) // 等待输入由高变低 { retry++; if(retry >= 100) return FAILURE; // 等待时间过长,返回异常。 delay_us(1); } retry=0; while (!DHT11_DQ_IN && retry<100) // 低电平持续时间80us { retry++; if(retry >= 100) return FAILURE; // 低电平持续时间过长,返回异常。 delay_us(1); } return SUCCESS; } //DHT11输出模式配置 static void SET_DHT11_IO_OUT() { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 推挽输出模式 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; GPIO_Init(DHT11_PORT,&GPIO_InitStructure); } //DHT11输入模式配置 static void SET_DHT11_IO_IN() { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; // 浮空输入模式,最小系统板和DHT11之间并未接上拉电阻 GPIO_Init(DHT11_PORT,&GPIO_InitStructure); } //从DHT11读取一位 //返回值:bit 1或0 //注:此函数处并未做高低电平持续时间异常的处理,因DHT11自身有校验和,即便读错1位也不会造成最终数据的错误。 // 严格来讲,除了正常情况下返回“1”或“0”,还应增加异常情况的返回值(比如“2”)。 static u8 DHT11_Read_Bit(void) { u8 retry=0; while(DHT11_DQ_IN && retry<100) // 等待输入电平由高变低(低电平持续50us) { retry++; delay_us(1); } retry=0; while(!DHT11_DQ_IN && retry<100) // 等待输入电平由低变高 { retry++; delay_us(1); } delay_us(40); // 等待40us(高电平持续26~28us表示0,持续70us表示1) if(DHT11_DQ_IN) return 1; // 40us后如果输入仍为高电平,则表示读入1;否则表示读入0。 else return 0; } //从DHT11读取一个字节 //返回值:读到的8位数据 static u8 DHT11_Read_Byte(void) { u8 i,byte; byte = 0; for (i=0;i<8;i++) { byte <<= 1; // 先前读取的数据(不足8位)全部左移一位 byte |= DHT11_Read_Bit(); // 最低位填入新读取的1位 } return byte; } //从DHT11读取一次完整的数据 //temp:温度值(整数,范围:0~50°) //humi:湿度值(整数,范围:20%~90%) //返回值:0,正常; 1,失败 static u8 DHT11_Read_Data(u8 *temp, u8 *humi) { u8 buf[5]; u8 i; DHT11_RequestData(); if(DHT11_RespondRequest() == SUCCESS) { for(i=0;i<5;i++) // 读取5组共40位数据 { buf[i] = DHT11_Read_Byte(); } if((buf[0]+buf[1]+buf[2]+buf[3]) == buf[4]) // 数据校验 { *humi = buf[0]; // 湿度整数部分 *temp = buf[2]; // 温度整数部分 return SUCCESS; } } return FAILURE; } // 温度超过阈值的处理措施 void DHT11_OnTempChange(u8 temp_threshold) { if (dht11.temp > 29) { // 蜂鸣器短鸣 BEEP_ON(); delay_ms(200); BEEP_OFF(); PWM_Val = HIGH_LIMIT ;// 启动风扇 } else { PWM_Val = LOW_LIMIT;// 关闭风扇 } TIM_SetCompare2(TIM3, PWM_Val); }用C语言写,STM32c8t6使用标准库

<SplitView DisplayMode=“CompactInline” IsPaneOpen=“{Binding classViewModel.IsPanelOpen}” CompactPaneLength=“48” OpenPaneLength=“250” PanePlacement=“Left”> <SplitView.Pane> <StackPanel Background="#F0F0F0"> <Button Command="{Binding classViewModel.TogglePanelCommand}" HorizontalAlignment="Right" Margin="4"> </Button> <StackPanel Margin="10,20" Spacing="10"> <TextBlock Cursor="AppStarting">cs</TextBlock> </StackPanel> </StackPanel> </SplitView.Pane> <Grid> <Grid.RowDefinitions> <RowDefinition Height="Auto"/> <RowDefinition Height="*"/> </Grid.RowDefinitions> <Button Command="{Binding classViewModel.TogglePanelCommand}" Grid.Row="0" Width="40" Height="40" HorizontalAlignment="Right" VerticalAlignment="Top" Margin="10" IsVisible="{Binding !classViewModel.IsPanelOpen}"> </Button> <ScrollViewer Grid.Row="1" Margin="20"> <StackPanel Spacing="15"> <TextBlock Text="报表数据设置" FontSize="16" FontWeight="Bold"/> <StackPanel Spacing="10"> <CheckBox Content="换算符号"/> <CheckBox Content="项目特征"/> <CheckBox Content="特征序号"/> </StackPanel> <Grid> <Grid.ColumnDefinitions> <ColumnDefinition Width="Auto"/> <ColumnDefinition Width="*"/> </Grid.ColumnDefinitions> <TextBlock Text="材料类别" VerticalAlignment="Center"/> <ComboBox Grid.Column="1" Margin="10,0" SelectedIndex="0"> <ComboBoxItem>所有</ComboBoxItem> </ComboBox> </Grid> <Grid> <Grid.ColumnDefinitions> <ColumnDefinition Width="Auto"/> <ColumnDefinition Width="*"/> </Grid.ColumnDefinitions> <TextBlock Text="材料排序" VerticalAlignment="Center"/> <ComboBox Grid.Column="1" Margin="10,0" SelectedIndex="0"> <ComboBoxItem>编号</ComboBoxItem> </ComboBox> </Grid> <CheckBox Content="材料类别排序(从小到大)"/> <Button Content="分析表设置" HorizontalAlignment="Left"/> </StackPanel> </ScrollViewer> </Grid> </SplitView><StackPanel Margin="10,20" Spacing="10"> <TextBlock Cursor="AppStarting">cs</TextBlock> </StackPanel> </StackPanel>修改成是在顶上,下面是然后是上下折叠

/*********************************** 本驱动文件仅适配HAL库版本 ***********************************/ // 在 oled.c 文件顶部添加 // 声明外部定义的句柄 #include "font.h" //字库文件 #include "oled.h" //声明 extern I2C_HandleTypeDef hi2c1; /* 控制宏 */ #define LEFT 0x27 #define RIGHT 0x26 #define UP 0X29 #define DOWM 0x2A #define ON 0xA7 #define OFF 0xA6 /* IIC接口选择 */ #define IICx hi2c1 //oled显示尺寸 uint16_t const displayWidth = 128; uint16_t const displayHeight = 64; /* OLED显存 [0]0 1 2 3 ... 127 [1]0 1 2 3 ... 127 [2]0 1 2 3 ... 127 [3]0 1 2 3 ... 127 [4]0 1 2 3 ... 127 [5]0 1 2 3 ... 127 [6]0 1 2 3 ... 127 [7]0 1 2 3 ... 127 */ static uint8_t OLED_RAM[8][128];//定义GDDRAM缓存区 void HAL_I2C_WriteByte(uint8_t addr,uint8_t data) { uint8_t TxData[2] = {addr,data}; HAL_I2C_Master_Transmit(&IICx,0X78,(uint8_t*)TxData,2,10); } /************************************************************** Prototype : void WriteCmd(uint8_t IIC_Command) Parameters : IIC_Command return : none Description : 写命令 ***************************************************************/ void WriteCmd(uint8_t IIC_Command) { HAL_I2C_WriteByte(0x00, IIC_Command); } /************************************************************** Prototype : void WriteDat(uint8_t IIC_Data) Parameters : IIC_Data return : none Description : 写数据 ***************************************************************/ void WriteDat(uint8_t IIC_Data) { HAL_I2C_WriteByte(0x40, IIC_Data); } /************************************************************** Prototype : void OLED_Init(void) Parameters : none return : none Description : 初始化OLED模块 ***************************************************************/ void OLED_Init(void) { HAL_Delay(500); WriteCmd(0xAE); //开显示 WriteCmd(0x20); //设置内存寻址模式 WriteCmd(0x10); //00,水平寻址模式;01,垂直寻址模式;10,页面寻址模式(重置);11,无效 WriteCmd(0xb0); //为页面寻址模式设置页面开始地址,0-7 WriteCmd(0x00); //---设置低列地址 WriteCmd(0x10); //---设置高列地址 WriteCmd(0xc8); //设置COM输出扫描方向 WriteCmd(0x40); //--设置起始行地址 WriteCmd(0x81); //--set contrast control register WriteCmd(0xff); //亮度调节 0x00~0xff WriteCmd(0xa1); //--设置段重新映射0到127 WriteCmd(0xa6); //--设置正常显示 WriteCmd(0xa8); //--设置复用比(1 ~ 64) WriteCmd(0x3F); // WriteCmd(0xa4); //0xa4,输出遵循RAM内容;0xa5,Output忽略RAM内容 WriteCmd(0xd3); //-设置显示抵消 WriteCmd(0x00); //-not offset WriteCmd(0xd5); //--设置显示时钟分频/振荡器频率 WriteCmd(0xf0); //--设置分率 WriteCmd(0xd9); //--设置pre-charge时期 WriteCmd(0x22); // WriteCmd(0xda); //--设置com大头针硬件配置 WriteCmd(0x12); WriteCmd(0xdb); //--设置vcomh WriteCmd(0x20); //0x20,0.77xVcc WriteCmd(0x8d); //--设置DC-DC WriteCmd(0x14); // WriteCmd(0xaf); //--打开oled面板 OLED_FullyClear();//清屏 } /************************************************************** Prototype : void OLED_ON(void) Parameters : none return : none Description : 将OLED从休眠中唤醒 ***************************************************************/ void OLED_ON(void) { WriteCmd(0X8D); //设置电荷泵 WriteCmd(0X14); //开启电荷泵 WriteCmd(0XAF); //OLED唤醒 } /************************************************************** Prototype : void OLED_OFF(void) Parameters : none return : none Description : 让OLED休眠 -- 休眠模式下,OLED功耗不到10uA ***************************************************************/ void OLED_OFF(void) { WriteCmd(0X8D); //设置电荷泵 WriteCmd(0X10); //关闭电荷泵 WriteCmd(0XAE); //OLED休眠 } /************************************************************** Prototype : void OLED_RefreshRAM(void) Parameters : none return : none Description : 全屏填充 ***************************************************************/ void OLED_RefreshRAM(void) { // 页寻址模式填充 for(uint16_t m = 0; m < displayHeight/8; m++) { WriteCmd(0xb0+m); //设置页地址b0~b7 WriteCmd(0x00); //设置显示位置—列低地址00-0f WriteCmd(0x10); //设置显示位置—列高地址10-1f for(uint16_t n = 0; n < displayWidth; n++) { WriteDat(OLED_RAM[m][n]); } } } /************************************************************** Prototype : void OLED_ClearRAM(void) Parameters : none return : none Description : 清除数据缓冲区 ***************************************************************/ void OLED_ClearRAM(void) { for(uint16_t m = 0; m < displayHeight/8; m++) { for(uint16_t n = 0; n < displayWidth; n++) { OLED_RAM[m][n] = 0x00; } } } /************************************************************** Prototype : void OLED_Fill(uint8_t fill_Data) Parameters : fill_Data 填充的1字节数据 return : none Description : 全屏填充 0x00~0xff ***************************************************************/ void OLED_FullyFill(uint8_t fill_Data) { for(uint16_t m = 0; m < displayHeight/8; m++) { for(uint16_t n = 0; n < displayWidth; n++) { OLED_RAM[m][n] = fill_Data; } } OLED_RefreshRAM(); } /************************************************************** Prototype : void OLED_FullyClear(void) Parameters : none return : none Description : 全屏清除 ***************************************************************/ void OLED_FullyClear(void) { OLED_FullyFill(RESET_PIXEL); } /************************************************************** Prototype : void OLED_SetPixel(int16_t x, int16_t y, uint8_t set_pixel) Parameters : x,y -- 起始点坐标(x:0~127, y:0~63); set_pixel 该点的数据 SET_PIXEL = 1, RESET_PIXEL = 0 return : none Description : 设置坐标像素点数据 ***************************************************************/ void OLED_SetPixel(int16_t x, int16_t y, uint8_t set_pixel) { if (x >= 0 && x < displayWidth && y >= 0 && y < displayHeight) { if(set_pixel){ OLED_RAM[y/8][x] |= (0x01 << (y%8)); } else{ OLED_RAM[y/8][x] &= ~(0x01 << (y%8)); } } } /************************************************************** Prototype : void OLED_GetPixel(int16_t x, int16_t y) Parameters : x,y -- 起始点坐标(x:0~127, y:0~63); return : PixelStatus 像素点状态 SET_PIXEL = 1, RESET_PIXEL = 0 Description : 获得坐标像素点数据 ***************************************************************/ PixelStatus OLED_GetPixel(int16_t x, int16_t y) { if(OLED_RAM[y/8][x] >> (y%8) & 0x01) return SET_PIXEL; return RESET_PIXEL; } /************************************************************** Prototype : void OLED_ShowStr(int16_t x, int16_t y, uint8_t ch[], uint8_t TextSize) Parameters : x,y -- 起始点坐标(x:0~127, y:0~63); ch[] -- 要显示的字符串; TextSize -- 字符大小(1:6*8 ; 2:8*16) return : none Description : 显示codetab.h中的ASCII字符,有6*8和8*16可选择 ***************************************************************/ void OLED_ShowStr(int16_t x, int16_t y, uint8_t ch[], uint8_t TextSize) { if (x >= 0 && x < displayWidth && y >= 0 && y < displayHeight) { int32_t c = 0; uint8_t j = 0; switch(TextSize) { case 1: { while(ch[j] != '\0') { c = ch[j] - 32; if(c < 0) //无效字符 break; if(x >= 125 || (127-x < 6))//一行最大显示字符数:21字节显示,多出两列,不显示 || 剩余列小于6不能显示完整字符,换行显示 { x = 0; y += 8;//换行显示 if(63 - y < 8) // 不足以显示一行时不显示 break; } for(uint8_t m = 0; m < 6; m++) { for(uint8_t n = 0; n < 8; n++) { OLED_SetPixel(x+m, y+n, (F6x8[c][m] >> n) & 0x01); } } x += 6; j++; } }break; case 2: { while(ch[j] != '\0') { c = ch[j] - 32; if(c < 0) //无效字符 break; if(x >= 127 || (127-x < 8))//16字节显示 || 剩余列小于8不能显示完整字符,换行显示 { x = 0; y += 16;//换行显示 if(63 - y < 16) // 不足以显示一行时不显示 break; } for(uint8_t m = 0; m < 2; m++) { for(uint8_t n = 0; n < 8; n++) { for(uint8_t i = 0; i < 8; i++) { OLED_SetPixel(x+n, y+i+m*8, (F8X16[c][n+m*8] >> i) & 0x01); } } } x += 8; j++; } }break; } } OLED_RefreshRAM(); } /************************************************************** Prototype : void OLED_ShowCN(int16_t x, int16_t y, uint8_t* ch) Parameters : x,y -- 起始点坐标(x:0~127, y:0~7); CN[]:汉字在codetab.h中的索引 return : none Description : 显示codetab.h中的汉字,16*16点阵 ***************************************************************/ void OLED_ShowCN(int16_t x, int16_t y, uint8_t* ch) { if (x >= 0 && x < displayWidth && y >= 0 && y < displayHeight) { int32_t len = 0,offset = sizeof(F16x16_CN[0].index); while(ch[len] != '\0') { if(x >= 127 || (127-x < 16))//8个汉字显示||剩余列小于16不能显示完整字符,换行显示 { x = 0; y += 16; if(63 - y < 16) // 不足以显示一行时不显示 break; } //需要处理输入数据大于显示数据的问题 for(uint8_t i = 0; i < sizeof(F16x16_CN)/sizeof(GB2312_CN); i++) { if(((F16x16_CN[i].index[0] == ch[len]) && (F16x16_CN[i].index[1] == ch[len+1]))){ for(uint8_t m = 0; m < 2; m++) //页 { for(uint8_t n = 0; n < 16; n++) // 列 { for(uint8_t j = 0; j < 8; j++) // 行 { OLED_SetPixel(x+n, y+j+m*8, (F16x16_CN[i].encoder[n+m*16] >> j) & 0x01); } } } x += 16; len += offset; break; } else if(F16x16_CN[i].index[0] == ch[len] && ch[len] == 0x20){ for(uint8_t m = 0; m < 2; m++) { for(uint8_t n = 0; n < 16; n++) { for(uint8_t j = 0; j < 8; j++) { OLED_SetPixel(x+n, y+j+m*8, (F16x16_CN[i].encoder[n+m*16] >> j) & 0x01); } } } x += 16; len++; break; } } } } OLED_RefreshRAM(); } /************************************************************** Prototype : void OLED_Show_MixedCH(int16_t x, int16_t y, uint8_t* ch) Parameters : x,y -- 起始点坐标(x:0~127, y:0~7); CN[]:汉字在codetab.h中的索引 return : none Description : 显示codetab.h中的汉字,16*16点阵,英文,8*16点阵 ***************************************************************/ void OLED_ShowMixedCH(int16_t x, int16_t y, uint8_t* ch) { if (x >= 0 && x < displayWidth && y >= 0 && y < displayHeight) { int32_t len = 0, c,offset = sizeof(F16x16_CN[0].index); while(ch[len] != '\0') { if(ch[len] >= 0xa1)//GB2312从0xA1A0开始 { for(uint8_t i = 0; i < sizeof(F16x16_CN)/sizeof(GB2312_CN); i++) { if(((F16x16_CN[i].index[0] == ch[len]) && (F16x16_CN[i].index[1] == ch[len+1]))) { if(x >= 127|| (127-x < 16))//8个汉字显示||剩余列小于16不能显示完整字符,换行显示 { x = 0; y += 16; if(63 - y < 16) // 不足以显示一行时不显示 break; } for(uint8_t m = 0; m < 2; m++) //页 { for(uint8_t n = 0; n < 16; n++) //列 { for(uint8_t j = 0; j < 8; j++) //行 { OLED_SetPixel(x+n, y+j+m*8, (F16x16_CN[i].encoder[n+m*16] >> j) & 0x01); } } } x += 16; len += offset; break; } } } else if(ch[len] <= 127)//ASCII编码范围0-127 { c = ch[len] - 32; if(c < 0) // 无效字符 break; if(x >= 127 || (127-x < 8))//16字节显示 || 剩余列小于8不能显示完整字符,换行显示 { x = 0; y += 16; if(63 - y < 16) // 不足以显示一行时不显示 break; } for(uint8_t m = 0; m < 2; m++) { for(uint8_t n = 0; n < 8; n++) { for(uint8_t i = 0; i < 8; i++) { OLED_SetPixel(x+n, y+i+m*8, (F8X16[c][n+m*8] >> i) & 0x01); } } } x += 8; len++; } } } OLED_RefreshRAM(); } /*************************************************************** Prototype : void OLED_DrawBMP(int16_t x0,int16_t y0,int16_t L,int16_t H,const uint8_t BMP[]) Parameters : (x0,y0)坐标长L宽H区域绘制图像BMP 0<=x0<=127 0<=y0<=63 0<=L+x0<=127 0<=H+y0<= 63 图像取模 纵向取模,字节倒序 return : none Description : 区域图像绘制,显示BMP位图,格式使用二维数组存储 ***************************************************************/ void OLED_DrawBMP(int16_t x0,int16_t y0,int16_t L,int16_t H,const uint8_t BMP[]) { if (x0 >= 0 && x0 < displayWidth && x0+L <= displayWidth &&\ y0 >= 0 && y0 < displayHeight && y0+H <= displayHeight) { uint8_t *p = (uint8_t *)BMP; for(int16_t y = y0; y < y0+H; y+=8) { for(int16_t x = x0; x < x0+L; x++) { for(int16_t i = 0; i < 8; i++) { // OLED_SetPixel(x, y+i, ((*((uint8_t *)BMP+(x-x0)+L*((y-y0)/8))) >> i) & 0x01); OLED_SetPixel(x, y+i, ((*p) >> i) & 0x01); } p++; } } } OLED_RefreshRAM(); } /*************************************************************** Prototype : void OLED_AreaFill(int16_t x0,int16_t y0,int16_t L,int16_t H) Parameters : 区域内容清除,(x0,y0)坐标长L宽H区域 0<=x0<=127 0<=y0<=63 0<=L+x0<=127 0<=H+y0<= 63 图像取模 纵向取模,字节倒序 return : none Description : 规定区域内容填充 ***************************************************************/ void OLED_AreaFill(int16_t x0,int16_t y0,int16_t L,int16_t H, uint8_t fill_data) { if (x0 >= 0 && x0 < displayWidth && x0+L <= displayWidth &&\ y0 >= 0 && y0 < displayHeight && y0+H <= displayHeight) { for(int16_t y = y0; y < y0+H; y++) { for(int16_t x = x0; x < x0+L; x++) { for(int16_t i = 0; i < 8; i++) { OLED_SetPixel(x, y+i, (fill_data >> i) & SET_PIXEL); } } } OLED_RefreshRAM(); } } /*************************************************************** Prototype : void OLED_AreaCLR(int16_t x0,int16_t y0,int16_t L,int16_t H) Parameters : (x0,y0)坐标长L宽H区域 0<=x0<=127 0<=y0<=63 0<=L+x0<=127 0<=H+y0<= 63 图像取模 纵向取模,字节倒序 return : none Description : 规定区域内容清除 ***************************************************************/ void OLED_AreaClear(int16_t x0,int16_t y0,int16_t L,int16_t H) { if (x0 >= 0 && x0 < displayWidth && x0+L <= displayWidth &&\ y0 >= 0 && y0 < displayHeight && y0+H <= displayHeight) { for(int16_t y = y0; y < y0+H; y+=8) { for(int16_t x = x0; x < x0+L; x++) { for(int16_t i = 0; i < 8; i++) { OLED_SetPixel(x, y+i, RESET_PIXEL); } } } OLED_RefreshRAM(); } } /*************************************************************** Prototype : void OLED_FullyToggle(void) Parameters : none return : none Description : 缓冲区数据取反后刷新到GDDRAM ***************************************************************/ void OLED_FullyToggle(void) { for(uint16_t m = 0; m < displayHeight/8; m++) { for(uint16_t n = 0; n < displayWidth; n++) { OLED_RAM[m][n] = ~OLED_RAM[m][n]; } } OLED_RefreshRAM(); } /*************************************************************** Prototype : void OLED_AreaToggle(int16_t x0,int16_t y0,int16_t L,int16_t H) Parameters : (x0,y0)坐标长L宽H区域 0<=x0<=127 0<=y0<=63 0<=L+x0<=127 0<=H+y0<= 63 图像取模 纵向取模,字节倒序 return : none Description : 规定区域内容取反 ***************************************************************/ void OLED_AreaToggle(int16_t x0,int16_t y0,int16_t L,int16_t H) { if (x0 >= 0 && x0 < displayWidth && x0+L <= displayWidth &&\ y0 >= 0 && y0 < displayHeight && y0+H <= displayHeight) { for(int16_t y = y0; y < y0+H; y+=8) { for(int16_t x = x0; x < x0+L; x++) { for(int16_t i = 0; i < 8; i++) { OLED_SetPixel(x, y+i, !OLED_GetPixel(x, y+i)); } } } OLED_RefreshRAM(); } } /**************************************************************** 全屏垂直偏移,0->63方向 方向垂直向上,范围0-63 方向垂直向下,范围63-0 ****************************************************************/ void OLED_VerticalShift(void) { for(uint8_t i = 0; i < displayHeight; i++) { WriteCmd(0xd3);//设置显示偏移,0->63方向 WriteCmd(i);//偏移量 HAL_Delay(40);//延时时间 } } /**************************************************************** 屏幕内容水平全屏滚动播放 左 LEFT 0x27 右 RIGHT 0x26 ****************************************************************/ void OLED_HorizontalShift(uint8_t direction) { WriteCmd(direction);//设置滚动方向 WriteCmd(0x00);//虚拟字节设置,默认为0x00 WriteCmd(0x00);//设置开始页地址 WriteCmd(0x05);//设置每个滚动步骤之间的时间间隔的帧频 WriteCmd(0x07);//设置结束页地址 WriteCmd(0x00);//虚拟字节设置,默认为0x00 WriteCmd(0xff);//虚拟字节设置,默认为0xff WriteCmd(0x2f);//开启滚动-0x2f,禁用滚动-0x2e,禁用需要重写数据 } /**************************************************************** 屏幕内容垂直水平全屏滚动播放 上 UP 0x29 下 DOWN 0x2A ****************************************************************/ void OLED_VerticalAndHorizontalShift(uint8_t direction) { WriteCmd(direction);//设置滚动方向 WriteCmd(0x00);//虚拟字节设置,默认为0x00 WriteCmd(0x00);//设置开始页地址 WriteCmd(0x05);//设置每个滚动步骤之间的时间间隔的帧频 WriteCmd(0x07);//设置结束页地址 WriteCmd(0x01);//垂直滚动偏移量 WriteCmd(0x2f);//开启滚动-0x2f,禁用滚动-0x2e,禁用需要重写数据 } /**************************************************************** 屏幕内容取反显示 开 ON 0xA7 关 OFF 0xA6 默认此模式,设置像素点亮 ****************************************************************/ void OLED_DisplayMode(uint8_t mode) { WriteCmd(mode); } /**************************************************************** 屏幕亮度调节 intensity 0-255 默认为0x7f ****************************************************************/ void OLED_IntensityControl(uint8_t intensity) { WriteCmd(0x81); WriteCmd(intensity); }这些代码怎么在main.c中调用使用

<SplitView DisplayMode="CompactInline" IsPaneOpen="{Binding classViewModel.IsPanelOpen}" CompactPaneLength="48" OpenPaneLength="250" PanePlacement="Left"> <SplitView.Pane> <StackPanel Background="#F0F0F0"> <Button Command="{Binding classViewModel.TogglePanelCommand}" HorizontalAlignment="Right" Margin="4"> </Button> <StackPanel Margin="10,20" Spacing="10"> <TextBlock Cursor="AppStarting">cs</TextBlock> </StackPanel> </StackPanel> </SplitView.Pane> <Grid> <Grid.RowDefinitions> <RowDefinition Height="Auto"/> <RowDefinition Height="*"/> </Grid.RowDefinitions> <Button Command="{Binding classViewModel.TogglePanelCommand}" Grid.Row="0" Width="40" Height="40" HorizontalAlignment="Right" VerticalAlignment="Top" Margin="10" IsVisible="{Binding !classViewModel.IsPanelOpen}"> </Button> <ScrollViewer Grid.Row="1" Margin="20"> <StackPanel Spacing="15"> <TextBlock Text="报表数据设置" FontSize="16" FontWeight="Bold"/> <StackPanel Spacing="10"> <CheckBox Content="换算符号"/> <CheckBox Content="项目特征"/> <CheckBox Content="特征序号"/> </StackPanel> <Grid> <Grid.ColumnDefinitions> <ColumnDefinition Width="Auto"/> <ColumnDefinition Width="*"/> </Grid.ColumnDefinitions> <TextBlock Text="材料类别" VerticalAlignment="Center"/> <ComboBox Grid.Column="1" Margin="10,0" SelectedIndex="0"> <ComboBoxItem>所有</ComboBoxItem> </ComboBox> </Grid> <Grid> <Grid.ColumnDefinitions> <ColumnDefinition Width="Auto"/> <ColumnDefinition Width="*"/> </Grid.ColumnDefinitions> <TextBlock Text="材料排序" VerticalAlignment="Center"/> <ComboBox Grid.Column="1" Margin="10,0" SelectedIndex="0"> <ComboBoxItem>编号</ComboBoxItem> </ComboBox> </Grid> <CheckBox Content="材料类别排序(从小到大)"/> <Button Content="分析表设置" HorizontalAlignment="Left"/> </StackPanel> </ScrollViewer> </Grid> </SplitView><StackPanel Margin="10,20" Spacing="10"> <TextBlock Cursor="AppStarting">cs</TextBlock> </StackPanel> </StackPanel>是在顶上,下面是然后是上下折叠

#include <stdio.h> #include <stdlib.h> #include <time.h> #include <stdbool.h> // ====================== // 全局变量 // ====================== float alarm_high = 30.0; // 高温阈值 float alarm_low = 10.0; // 低温阈值 bool is_fahrenheit = 0; // 温度单位标志(0:℃,1:℉) // 星期数组(索引0-6对应周日-周六) const char* weekdays[7] = { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" }; // 时间结构体 typedef struct { int second; int minute; int hour; int date; int month; int weekday; // 0-6对应周日-周六 int year; // 0-99(实际年份=2000+year) } TIME; // ====================== // 函数声明 // ====================== // 延时函数 void delay_ms(unsigned int ms); // 报警函数 void check_alarm(float temp); // 显示函数 void display_time(void); void display_temp(float temp, bool is_fahrenheit); // 按键处理函数 char get_key(void); void key_handle(void); // 设置函数 void set_alarm_range(void); void set_time(void); void save_settings(void); // 模拟硬件函数 float simulate_temperature(void); TIME get_current_time(void); // ====================== // 主函数 // ====================== int main(void) { float temperature; // 初始化随机数种子 srand(time(NULL)); while(1) { // 模拟读取温度 temperature = simulate_temperature(); // 温度报警检测 check_alarm(temperature); // 显示时间和温度 display_time(); display_temp(temperature, is_fahrenheit); // 按键处理 key_handle(); // 延时避免资源占用过高 delay_ms(1000); } return 0; } // ====================== // 模拟温度传感器函数 // ====================== float simulate_temperature(void) { // 模拟20-30度之间的温度波动 static float base_temp = 25.0; float fluctuation = ((float)rand() / RAND_MAX) * 2.0 - 1.0; // -1到1度的随机波动 base_temp += fluctuation * 0.1; // 确保温度在合理范围内 if (base_temp < 15.0) base_temp = 15.0; if (base_temp > 35.0) base_temp = 35.0; return base_temp; } // ====================== // 获取模拟时间函数 // ====================== TIME get_current_time(void) { TIME t; time_t now = time(NULL); struct tm *tm_info = localtime(&now); t.second = tm_info->tm_sec; t.minute = tm_info->tm_min; t.hour = tm_info->tm_hour; t.date = tm_info->tm_mday; t.month = tm_info->tm_mon + 1; t.weekday = tm_info->tm_wday; t.year = tm_info->tm_year + 1900 - 2000; // 转换为2000+year格式 return t; } // ====================== // 报警函数 // ====================== void check_alarm(float temp) { if(temp > alarm_high || temp < alarm_low) { printf("\a"); // 发出警报声 printf("警告: 温度超出范围!\n"); } } // ====================== // 显示函数 // ====================== void display_time(void) { TIME t = get_current_time(); // 格式:2025-06-07 Sat 12:30:45 printf("\r%04d-%02d-%02d %s %02d:%02d:%02d ", 2000 + t.year, t.month, t.date, weekdays[t.weekday], t.hour, t.minute, t.second); } void display_temp(float temp, bool is_fahrenheit) { char unit = 'C'; if(is_fahrenheit) { temp = temp * 9.0/5.0 + 32.0; // 转换为华氏度 unit = 'F'; } // 格式:25.5°C printf("温度: %4.1f%c%c", temp, 0xDF, unit); // 显示阈值 printf(" 阈值: %.1f-%.1f%c", alarm_low, alarm_high, unit); // 显示单位切换提示 printf(" [A:切换单位]"); } // ====================== // 按键处理函数 // ====================== char get_key(void) { // 在沙盒环境中无法真正检测按键,这里简化处理 // 实际应用中需要替换为真正的按键检测代码 return 0; // 无按键 } void key_handle(void) { char key = get_key(); switch(key) { case 'A': // 切换温度单位 is_fahrenheit = !is_fahrenheit; printf("\n温度单位已切换为 %c\n", is_fahrenheit ? 'F' : 'C'); break; case 'B': // 进入温度设置 set_alarm_range(); break; case 'C': // 进入时间设置 set_time(); break; case '#': // 确认设置 save_settings(); break; } } // ====================== // 延时函数 // ====================== void delay_ms(unsigned int ms) { clock_t start_time = clock(); while (clock() < start_time + ms * CLOCKS_PER_SEC / 1000); } // ====================== // 设置函数 // ====================== void set_alarm_range(void) { printf("\n进入温度阈值设置模式\n"); printf("当前阈值: %.1f-%.1f°C\n", alarm_low, alarm_high); printf("在实际应用中,您可以通过按键调整这些值\n"); // 实际应用中需要通过按键调整 } void set_time(void) { printf("\n进入时间设置模式\n"); printf("在实际应用中,您可以通过按键调整时间\n"); // 实际应用中需要通过按键调整 } void save_settings(void) { printf("\n设置已保存\n"); // 实际应用中需要实现保存功能 }

大家在看

recommend-type

B50610-DS07-RDS(博通千兆以太网手册) - 副本.pdf

B50610C1KMLG datasheet 10/100/1000BASE-T Gigabit Ethernet Transceiver The Broadcom® B50610 is a triple-speed 1000BASE-T/ 100BASE-TX/10BASE-T Gigabit Ethernet (GbE) transceiver integrated into a single monolithic CMOS chip. The device performs all physical-layer functions for 1000BASE-T, 100BASE-TX, and 10BASE-T Ethernet on standard category 5 UTP cable. 10BASE-T can also run on standard category 3, 4, and 5 UTP. The B50610 is a highly integrated solution combining digital adaptive equalizers, ADCs, phase-locked loops, line drivers, encoders, decoders, echo cancellers, crosstalk cancellers, and all required support circuitry. Based on Broadcom’s proven Digital Signal Processor technology, the B50610 is designed to be fully compliant with RGMII, allowing compatibility with industry-standard Ethernet MACs and switch controllers.
recommend-type

尼康D610官方升级固件1.02

尼康D610官方固件1.02版升级包.有需要的来下载吧.将固件复制到SD卡中.打开相机选择到固件点击即可升级!
recommend-type

的表中所-数据结构中文版

是用于对以位单位访问的点数进行指定的数据,在3.3.1项(1)的表中所 示的1次通信中可处理的点数以内进行指定。 a) 通过 ASCII代码进行数据通信时 将点数转换为 ASCII 代码 2位(16 进制数)后使用,从各高位进行 发送。 (示例) 5 点的情况 下 : 变为“05”,从“0”开始按顺序进行发送。 20 点的情 况下 : 变为“14”,从“1”开始按顺序进行发送。 b) 通过二进制代码进行数据通信时 使用表示点数的 1字节的数值进行发送。 (示例) 5 点的情况 下 : 发送 05H。 20 点的情 况下 : 发送 14H。 7) 设置/复位 是用于指定写入到位软元件中的数据的数据,以如下所示的值进行指 定。 写入数据 ON OFF 备注 ASCII 代码 “01” “00” 从“0”开始按顺序发送 2 字符 二进制代码 01H 00H 发送如左所示的 1 字节的数值
recommend-type

kfb转换工具(kfb-svs)

kfb转换工具(kfb-svs)
recommend-type

GPS轨迹转换软件 GPSBabel

GPSBabel 是一个非常好用的GPS轨迹转换软件,可以将GPSgate记录的NMEA格式转换为可以在Googleearth中显示的kml格式,也可生成ozi用的plt格式。其中plt格式的轨迹文件可用灵图的电子地图!~~

最新推荐

recommend-type

2005年上半年网络工程师上午试卷(解析)--网络工程师试题及答案.doc

2005年上半年网络工程师上午试卷(解析)--网络工程师试题及答案.doc
recommend-type

构建基于ajax, jsp, Hibernate的博客网站源码解析

根据提供的文件信息,本篇内容将专注于解释和阐述ajax、jsp、Hibernate以及构建博客网站的相关知识点。 ### AJAX AJAX(Asynchronous JavaScript and XML)是一种用于创建快速动态网页的技术,它允许网页在不重新加载整个页面的情况下,与服务器交换数据并更新部分网页内容。AJAX的核心是JavaScript中的XMLHttpRequest对象,通过这个对象,JavaScript可以异步地向服务器请求数据。此外,现代AJAX开发中,常常用到jQuery中的$.ajax()方法,因为其简化了AJAX请求的处理过程。 AJAX的特点主要包括: - 异步性:用户操作与数据传输是异步进行的,不会影响用户体验。 - 局部更新:只更新需要更新的内容,而不是整个页面,提高了数据交互效率。 - 前后端分离:AJAX技术允许前后端分离开发,让前端开发者专注于界面和用户体验,后端开发者专注于业务逻辑和数据处理。 ### JSP JSP(Java Server Pages)是一种动态网页技术标准,它允许开发者将Java代码嵌入到HTML页面中,从而实现动态内容的生成。JSP页面在服务器端执行,并将生成的HTML发送到客户端浏览器。JSP是Java EE(Java Platform, Enterprise Edition)的一部分。 JSP的基本工作原理: - 当客户端首次请求JSP页面时,服务器会将JSP文件转换为Servlet。 - 服务器上的JSP容器(如Apache Tomcat)负责编译并执行转换后的Servlet。 - Servlet生成HTML内容,并发送给客户端浏览器。 JSP页面中常见的元素包括: - 指令(Directives):如page、include、taglib等。 - 脚本元素:脚本声明(Script declarations)、脚本表达式(Scriptlet)和脚本片段(Expression)。 - 标准动作:如jsp:useBean、jsp:setProperty、jsp:getProperty等。 - 注释:在客户端浏览器中不可见的注释。 ### Hibernate Hibernate是一个开源的对象关系映射(ORM)框架,它提供了从Java对象到数据库表的映射,简化了数据库编程。通过Hibernate,开发者可以将Java对象持久化到数据库中,并从数据库中检索它们,而无需直接编写SQL语句或掌握复杂的JDBC编程。 Hibernate的主要优点包括: - ORM映射:将对象模型映射到关系型数据库的表结构。 - 缓存机制:提供了二级缓存,优化数据访问性能。 - 数据查询:提供HQL(Hibernate Query Language)和Criteria API等查询方式。 - 延迟加载:可以配置对象或对象集合的延迟加载,以提高性能。 ### 博客网站开发 构建一个博客网站涉及到前端页面设计、后端逻辑处理、数据库设计等多个方面。使用ajax、jsp、Hibernate技术栈,开发者可以更高效地构建功能完备的博客系统。 #### 前端页面设计 前端主要通过HTML、CSS和JavaScript来实现,其中ajax技术可以用来异步获取文章内容、用户评论等,无需刷新页面即可更新内容。 #### 后端逻辑处理 JSP可以在服务器端动态生成HTML内容,根据用户请求和数据库中的数据渲染页面。Hibernate作为ORM框架,可以处理Java对象与数据库表之间的映射,并提供数据库的CRUD(创建、读取、更新、删除)操作。 #### 数据库设计 博客网站的数据库设计通常包含多个表,如用户表(存储用户信息)、文章表(存储文章信息)、评论表(存储用户评论信息)等。使用Hibernate框架可以简化数据库操作,同时确保数据的一致性和安全性。 #### 安全性和性能优化 安全性是构建网站时需要考虑的重要方面,包括但不限于SQL注入防护、XSS攻击防护、会话管理等。性能优化方面,可以利用Hibernate的缓存机制,以及对JSP页面和ajax请求进行适当的缓存处理。 ### 结论 ajax、jsp、Hibernate技术结合可以构建出高效、动态、易于维护的博客网站。在开发过程中,应当关注前后端分离、用户体验优化、系统性能和安全性等关键要素,确保博客网站的稳定和长期可用性。通过本篇文章,读者应该已经对这些技术有了初步了解,并能够结合文件提供的源码开始进行相关开发实践。
recommend-type

【Unity Sunny Land关卡设计高级指南】:打造完美关卡的8大技巧

# 1. Unity关卡设计概述 ## 1.1 关卡设计的核心要素 关卡设计作为游戏开发中不可或缺的一环,承担着构建游戏世界、呈现故事背景、引导玩家体验和激发玩家兴趣的责任。一个优秀的关卡不仅需要有吸引人的视觉效果和有趣的挑战,还需平衡玩家体验,提供连贯的游戏故事。 ## 1.2 关卡设计师的角色 在Unity引擎的环境下,关卡设计师需要具备对场景布局、角色互动、剧情
recommend-type

C++ 模版

<think>我们正在讨论C++模板的使用和常见问题。根据用户要求,所有行内数学表达式使用$...$,独立公式使用$$...$$,用中文回答,并生成相关问题。同时,回答要结构清晰,帮助用户逐步解决问题。 首先,回顾用户的问题:用户想了解C++模板的使用和常见问题。我们将从以下几个方面展开: 1. 模板的基本概念和用途 2. 函数模板和类模板 3. 模板特化和偏特化 4. 模板元编程简介 5. 常见问题及解决方案 注意:在回答中,我们会引用站内提供的引用内容(引用[1]和引用[2])作为参考,但主要围绕模板展开。 ### 1. 模板的基本概念和用途 C++模板是一种支持泛型编程的特性,允许
recommend-type

C#随机数摇奖系统功能及隐藏开关揭秘

### C#摇奖系统知识点梳理 #### 1. C#语言基础 C#(发音为“看井”)是由微软开发的一种面向对象的、类型安全的编程语言。它是.NET框架的核心语言之一,广泛用于开发Windows应用程序、ASP.NET网站、Web服务等。C#提供丰富的数据类型、控制结构和异常处理机制,这使得它在构建复杂应用程序时具有很强的表达能力。 #### 2. 随机数的生成 在编程中,随机数生成是常见的需求之一,尤其在需要模拟抽奖、游戏等场景时。C#提供了System.Random类来生成随机数。Random类的实例可以生成一个伪随机数序列,这些数在统计学上被认为是随机的,但它们是由确定的算法生成,因此每次运行程序时产生的随机数序列相同,除非改变种子值。 ```csharp using System; class Program { static void Main() { Random rand = new Random(); for(int i = 0; i < 10; i++) { Console.WriteLine(rand.Next(1, 101)); // 生成1到100之间的随机数 } } } ``` #### 3. 摇奖系统设计 摇奖系统通常需要以下功能: - 用户界面:显示摇奖结果的界面。 - 随机数生成:用于确定摇奖结果的随机数。 - 动画效果:模拟摇奖的视觉效果。 - 奖项管理:定义摇奖中可能获得的奖品。 - 规则设置:定义摇奖规则,比如中奖概率等。 在C#中,可以使用Windows Forms或WPF技术构建用户界面,并集成上述功能以创建一个完整的摇奖系统。 #### 4. 暗藏的开关(隐藏控制) 标题中提到的“暗藏的开关”通常是指在程序中实现的一个不易被察觉的控制逻辑,用于在特定条件下改变程序的行为。在摇奖系统中,这样的开关可能用于控制中奖的概率、启动或停止摇奖、强制显示特定的结果等。 #### 5. 测试 对于摇奖系统来说,测试是一个非常重要的环节。测试可以确保程序按照预期工作,随机数生成器的随机性符合要求,用户界面友好,以及隐藏的控制逻辑不会被轻易发现或利用。测试可能包括单元测试、集成测试、压力测试等多个方面。 #### 6. System.Random类的局限性 System.Random虽然方便使用,但也有其局限性。其生成的随机数序列具有一定的周期性,并且如果使用不当(例如使用相同的种子创建多个实例),可能会导致生成相同的随机数序列。在安全性要求较高的场合,如密码学应用,推荐使用更加安全的随机数生成方式,比如RNGCryptoServiceProvider。 #### 7. Windows Forms技术 Windows Forms是.NET框架中用于创建图形用户界面应用程序的库。它提供了一套丰富的控件,如按钮、文本框、标签等,以及它们的事件处理机制,允许开发者设计出视觉效果良好且功能丰富的桌面应用程序。 #### 8. WPF技术 WPF(Windows Presentation Foundation)是.NET框架中用于构建桌面应用程序用户界面的另一种技术。与Windows Forms相比,WPF提供了更现代化的控件集,支持更复杂的布局和样式,以及3D图形和动画效果。WPF的XAML标记语言允许开发者以声明性的方式设计用户界面,与C#代码分离,易于维护和更新。 #### 9. 压缩包子文件TransBallDemo分析 从文件名“TransBallDemo”可以推测,这可能是一个C#的示例程序或者演示程序,其中“TransBall”可能表示旋转的球体,暗示该程序包含了动画效果,可能是用来模拟转动的球体(如转盘或摇奖球)。该文件可能是用来展示如何实现一个带有视觉动画效果的摇奖系统的C#程序。 总结以上内容,我们可以得出构建一个C#摇奖系统需要深入理解C#语言及其随机数生成机制,设计用户界面,集成动画效果,确保隐藏控制逻辑的安全性,以及全面测试系统以保证其正确性和公平性。通过掌握Windows Forms或WPF技术,可以进一步增强系统的视觉和交互体验。
recommend-type

【数据驱动的力量】:管道缺陷判别方法论与实践经验

# 摘要 数据驱动技术在管道缺陷检测领域展现出强大的力量,本文首先概述了数据驱动的力量和管道缺陷判别的基础理论。接着,重点探讨了管道缺陷的类型与特征、数据采集与预处理、数据驱动模型的选择与构建。在实践技巧章节,本文详述了实战数据集的准备与处理、缺陷识别算法的应用、以及性能评估与模型优化。此外,高级应用章节深入讨论了实时数据处理分析、多模态数据融合技术、以及深度学习在缺
recommend-type

FatalError: `Segmentation fault` is detected by the operating system. [TimeInfo: *** Aborted at 1752568501 (unix time) try "date -d @1752568501" if you are using GNU date ***] [SignalInfo: *** SIGSEGV (@0x0) received by PID 3203 (TID 0x7f8a04143340) from PID 0 ***],训练模型中出现上述错误,怎么解决

<think>我们面对的是一个模型训练过程中出现的Segmentation fault错误。根据引用[1],Segmentation Fault通常是由非法内存访问引起的,常见原因包括空指针解引用、数组越界、使用未初始化指针等。在深度学习训练场景中,还可能涉及显存溢出、CUDA内核错误等。 引用[2]展示了一个具体案例:在PaddlePaddle框架中遇到Segmentation fault,并提示了C++ Traceback。这通常表明底层C++代码出现了问题。而引用[3]则提到Python环境下的Segmentation fault,可能涉及Python扩展模块的错误。 解决步骤: 1
recommend-type

EditPlus中实现COBOL语言语法高亮的设置

标题中的“editplus”指的是一个轻量级的代码编辑器,特别受到程序员和软件开发者的欢迎,因为它支持多种编程语言。标题中的“mfcobol”指的是一种特定的编程语言,即“Micro Focus COBOL”。COBOL语言全称为“Common Business-Oriented Language”,是一种高级编程语言,主要用于商业、金融和行政管理领域的数据处理。它最初开发于1959年,是历史上最早的高级编程语言之一。 描述中的“cobol语言颜色显示”指的是在EditPlus这款编辑器中为COBOL代码提供语法高亮功能。语法高亮是一种编辑器功能,它可以将代码中的不同部分(如关键字、变量、字符串、注释等)用不同的颜色和样式显示,以便于编程者阅读和理解代码结构,提高代码的可读性和编辑的效率。在EditPlus中,要实现这一功能通常需要用户安装相应的语言语法文件。 标签“cobol”是与描述中提到的COBOL语言直接相关的一个词汇,它是对描述中提到的功能或者内容的分类或者指代。标签在互联网内容管理系统中用来帮助组织内容和便于检索。 在提供的“压缩包子文件的文件名称列表”中只有一个文件名:“Java.stx”。这个文件名可能是指一个语法高亮的模板文件(Syntax Template eXtension),通常以“.stx”为文件扩展名。这样的文件包含了特定语言语法高亮的规则定义,可用于EditPlus等支持自定义语法高亮的编辑器中。不过,Java.stx文件是为Java语言设计的语法高亮文件,与COBOL语言颜色显示并不直接相关。这可能意味着在文件列表中实际上缺少了为COBOL语言定义的相应.stx文件。对于EditPlus编辑器,要实现COBOL语言的颜色显示,需要的是一个COBOL.stx文件,或者需要在EditPlus中进行相应的语法高亮设置以支持COBOL。 为了在EditPlus中使用COBOL语法高亮,用户通常需要做以下几步操作: 1. 确保已经安装了支持COBOL的EditPlus版本。 2. 从Micro Focus或者第三方资源下载COBOL的语法高亮文件(COBOL.stx)。 3. 打开EditPlus,进入到“工具”菜单中的“配置用户工具”选项。 4. 在用户工具配置中,选择“语法高亮”选项卡,然后选择“添加”来载入下载的COBOL.stx文件。 5. 根据需要选择其他语法高亮的选项,比如是否开启自动完成、代码折叠等。 6. 确认并保存设置。 完成上述步骤后,在EditPlus中打开COBOL代码文件时,应该就能看到语法高亮显示了。语法高亮不仅仅是颜色的区分,它还可以包括字体加粗、斜体、下划线等样式,以及在某些情况下,语法错误的高亮显示。这对于提高编码效率和准确性有着重要意义。
recommend-type

影子系统(windows)问题排查:常见故障诊断与修复

# 摘要 本文旨在深入探讨影子系统的概念、工作原理以及故障诊断基础。首先,介绍影子系统的定义及其运作机制,并分析其故障诊断的理论基础,包括系统故障的分类和特征。接着,详细探讨各种故障诊断工具和方法,并提供实际操作中的故障排查步骤。文中还深入分析了影子系统常见故障案例,涵盖系统启动问题、软件兼容性和网络连通性问题,并提供相应的诊断与解决方案。高级故障诊断与修复
recommend-type

nt!DbgBreakPointWithStatus: fffff805`7affd0b0 cc int 3 kd> g KDTARGET: Refreshing KD connection *** Fatal System Error: 0x0000001a (0x0000000000061941,0xFFFFF8057B20E1C0,0x0000000000000019,0xFFFFFC89CACA7190) Break instruction exception - code 80000003 (first chance) A fatal system error has occurred. Debugger entered on first try; Bugcheck callbacks have not been invoked. A fatal system error has occurred. For analysis of this file, run !analyze -v nt!DbgBreakPointWithStatus: fffff805`7affd0b0 cc int 3 kd> !analyze -v Connected to Windows 10 19041 x64 target at (Tue Jul 15 23:02:04.588 2025 (UTC + 8:00)), ptr64 TRUE Loading Kernel Symbols ............. Press ctrl-c (cdb, kd, ntsd) or ctrl-break (windbg) to abort symbol loads that take too long. Run !sym noisy before .reload to track down problems loading symbols. .................................................. ................................................................ ..................................................... Loading User Symbols Loading unloaded module list ...... ERROR: FindPlugIns 8007007b ******************************************************************************* * * * Bugcheck Analysis * * * ******************************************************************************* MEMORY_MANAGEMENT (1a) # Any other values for parameter 1 must be individually examined. Arguments: Arg1: 0000000000061941, The subtype of the BugCheck. Arg2: fffff8057b20e1c0 Arg3: 0000000000000019 Arg4: fffffc89caca7190 Debugging Details: ------------------ Press ctrl-c (cdb, kd, ntsd) or ctrl-break (windbg) to abort symbol loads that take too long. Run !sym noisy before .reload to track down problems loading symbols. 蓝屏了,但原帖子代码能正常执行,帖子作者说:我win10是专门针对2mb大页,win7是专门针对4kb小页的,蓝屏的原因我只能想到是因为没有做大页和小页的处理了。你做了2mb和4kb之类的处理吗,还有2mb的大页的物理地址要2mb对齐,不然电脑会卡死,我win10蓝屏的原因就是因为没有物理地址2mb对齐,我这代码是专门针对我那个虚拟机的环境,其他环境蓝屏你要是懂分页的话自己就会处理了。最重要的细节就是物理地址要对齐,2mb就2mb的对齐,1gb就1gb的对齐。

<think>我们之前讨论的代码中已经包含了对大页(2MB)和小页(4KB)的处理。但是,根据蓝屏信息(MEMORY_MANAGEMENT, 0x1a)和参数,问题可能出在内存对齐或页表项设置上。 原帖作者提到:2MB大页的物理地址必须2MB对齐,否则会导致系统不稳定甚至蓝屏。在之前的代码中,我们确实处理了大页的情况,但在分配2MB内存时,我们使用了`MmAllocateContiguousMemorySpecifyCache`,这个函数分配的内存不一定满足2MB对齐(尽管它分配的是连续内存)。因此,我们需要确保分配的内存地址是2MB对齐的。 解决方案: 1. 在分配2MB内存时,要