开源MicroPython飞控

原创 云深之无迹 2021-11-21 17:47


起点亦是终点,世间有轮回,相似又不相同。就像书里写的那样。江湖因一声“小二上酒”开始,最后也因一杯绿蚁戛然而止。生活,也是如此吧?


飞控不是一下做出来的,下面是一些重要传感器的mpy驱动代码,为飞控项目添砖加瓦。


from pyb import Pinfrom time import sleep_us,ticks_us
class HC(): def __init__(self,trig='C5',echo='C4'): self.trig = Pin(trig, Pin.OUT_PP) self.echo = Pin(echo, Pin.IN)
def trigUp(self): self.trig.value(1) sleep_us(25) self.trig.value(0)
def getlen(self): distance = 0 self.trigUp() while self.echo.value() == 0: pass ts = ticks_us() # 开始时间 while self.echo.value() == 1: pass te = ticks_us() # 结束时间 tc = te - ts # 回响时间(单位us) distance = (tc * 170) / 10000 # 距离计算(单位为:cm) return distance


超声波传感器


'''参考:https://blog.csdn.net/qq_38721302/article/details/83095545'''
MPU_ADDR=0X68WHO_AM_I_VAL = MPU_ADDR

MPU_PWR_MGMT1_REG = 0x6B # 电源管理寄存器1MPU_GYRO_CFG_REG = 0X1B # 陀螺仪配置寄存器MPU_ACCEL_CFG_REG = 0X1C # 加速度传感器配置寄存器MPU_SAMPLE_RATE_REG=0X19 # 设置MPU6050的采样率MPU_CFG_REG=0X1A # 配置寄存器
MPU_INT_EN_REG=0X38MPU_USER_CTRL_REG=0X6AMPU_FIFO_EN_REG=0X23MPU_INTBP_CFG_REG=0X37MPU_DEVICE_ID_REG=0X75
MPU_GYRO_XOUTH_REG=0X43MPU_GYRO_XOUTL_REG=0X44MPU_GYRO_YOUTH_REG=0X45MPU_GYRO_YOUTL_REG=0X46MPU_GYRO_ZOUTH_REG=0X47MPU_GYRO_ZOUTL_REG=0X48
MPU_ACCEL_XOUTH_REG=0X3BMPU_ACCEL_XOUTL_REG=0X3CMPU_ACCEL_YOUTH_REG=0X3DMPU_ACCEL_YOUTL_REG=0X3EMPU_ACCEL_ZOUTH_REG=0X3FMPU_ACCEL_ZOUTL_REG=0X40MPU_TEMP_OUTH_REG=0X41MPU_TEMP_OUTL_REG=0X42

config_gyro_range = 3 # 0,±250°/S;1,±500°/S;2,±1000°/S;3,±2000°/Sconfig_accel_range = 0# 0,±2g;1,±4g;2,±8g;3,±16g
class MPU6050(): def __init__(self,iicbus,address=WHO_AM_I_VAL): self._address = address self._bus = iicbus self.reset() def _write_byte(self,cmd,val): self._bus.mem_write(val,self._address,cmd) def _read_byte(self,cmd): buf = self._bus.mem_read(1,self._address,cmd,addr_size=8) return int(buf[0]) def reset(self): self._write_byte(MPU_PWR_MGMT1_REG, 0x00) # 配置电源管理寄存器 启动MPU6050 self._write_byte(MPU_GYRO_CFG_REG, config_gyro_range<<3) # 陀螺仪传感器,±2000dps self._write_byte(MPU_ACCEL_CFG_REG, config_accel_range<<3)# 加速度传感器,±2g self._write_byte(MPU_SAMPLE_RATE_REG, 0x07) # 采样频率 100 self._write_byte(MPU_CFG_REG, 0x06) # 设置数字低通滤波器 self._write_byte(MPU_INT_EN_REG,0X00) #关闭所有中断 self._write_byte(MPU_USER_CTRL_REG,0X00) #I2C主模式关闭 self._write_byte(MPU_FIFO_EN_REG,0X00) #关闭FIFO self._write_byte(MPU_INTBP_CFG_REG,0X80) #INT引脚低电平有效 buf = self._read_byte(MPU_DEVICE_ID_REG) if buf != self._address: print("NPU6050 not found!") else: pass def _read_byte(self,cmd): buf = self._bus.mem_read(1,self._address,cmd,addr_size=8) return int(buf[0]) def _read_u16(self,reg): MSB = self._read_byte(reg) LSB = self._read_byte(reg) return (MSB<< 8) + LSB def _read_s16(self,reg): result = self._read_u16(reg) if result > 32767:result -= 65536 return result
def read_Gyro_x(self): x = self._read_s16(MPU_GYRO_XOUTH_REG) return x def read_Gyro_y(self): y = self._read_s16(MPU_GYRO_YOUTH_REG) return y def read_Gyro_z(self): z = self._read_s16(MPU_GYRO_ZOUTH_REG) return z
def read_Accel_x(self): x = self._read_s16(MPU_ACCEL_XOUTH_REG) return x def read_Accel_y(self): y = self._read_s16(MPU_ACCEL_YOUTH_REG) return y def read_Accel_z(self): z = self._read_s16(MPU_ACCEL_ZOUTH_REG) return z
def read_Temp(self): temp = self._read_s16(MPU_TEMP_OUTH_REG) return temp

from pyb import I2C
i2c = I2C(2, I2C.MASTER)mpu = MPU6050(i2c)
def GyroToDegree(num): return num / 32768 * 2000
def AccelToGram(num): return num / 32768 * 2


MPU6050


from imu import InvenSenseMPU, bytes_toint, MPUExceptionfrom imu import Vector3d

class MPU9250(InvenSenseMPU): ''' MPU9250 constructor arguments 1. side_str 'X' or 'Y' depending on the Pyboard I2C interface being used 2. optional device_addr 0, 1 depending on the voltage applied to pin AD0 (Drotek default is 1) if None driver will scan for a device (if one device only is on bus) 3, 4. transposition, scaling optional 3-tuples allowing for outputs to be based on vehicle coordinates rather than those of the sensor itself. See readme. '''
_mpu_addr = (104, 105) # addresses of MPU9250 determined by voltage on pin AD0 _mag_addr = 12 # Magnetometer address _chip_id = 113
def __init__(self, side_str, device_addr=None, transposition=(0, 1, 2), scaling=(1, 1, 1)):
super().__init__(side_str, device_addr, transposition, scaling) self._mag = Vector3d(transposition, scaling, self._mag_callback) self.accel_filter_range = 0 # fast filtered response self.gyro_filter_range = 0 self._mag_stale_count = 0 # MPU9250 count of consecutive reads where old data was returned self.mag_correction = self._magsetup() # 16 bit, 100Hz update.Return correction factors. self._mag_callback() # Seems neccessary to kick the mag off else 1st reading is zero (?)
@property def sensors(self): ''' returns sensor objects accel, gyro and mag ''' return self._accel, self._gyro, self._mag
# get temperature @property def temperature(self): ''' Returns the temperature in degree C. ''' try: self._read(self.buf2, 0x41, self.mpu_addr) except OSError: raise MPUException(self._I2Cerror) return bytes_toint(self.buf2[0], self.buf2[1])/333.87 + 21 # I think
# Low pass filters @property def gyro_filter_range(self): ''' Returns the gyro and temperature sensor low pass filter cutoff frequency Pass: 0 1 2 3 4 5 6 7 Cutoff (Hz): 250 184 92 41 20 10 5 3600 Sample rate (KHz): 8 1 1 1 1 1 1 8 ''' try: self._read(self.buf1, 0x1A, self.mpu_addr) res = self.buf1[0] & 7 except OSError: raise MPUException(self._I2Cerror) return res
@gyro_filter_range.setter def gyro_filter_range(self, filt): ''' Sets the gyro and temperature sensor low pass filter cutoff frequency Pass: 0 1 2 3 4 5 6 7 Cutoff (Hz): 250 184 92 41 20 10 5 3600 Sample rate (KHz): 8 1 1 1 1 1 1 8 ''' if filt in range(8): try: self._write(filt, 0x1A, self.mpu_addr) except OSError: raise MPUException(self._I2Cerror) else: raise ValueError('Filter coefficient must be between 0 and 7')
@property def accel_filter_range(self): ''' Returns the accel low pass filter cutoff frequency Pass: 0 1 2 3 4 5 6 7 BEWARE 7 doesn't work on device I tried. Cutoff (Hz): 460 184 92 41 20 10 5 460 Sample rate (KHz): 1 1 1 1 1 1 1 1 ''' try: self._read(self.buf1, 0x1D, self.mpu_addr) res = self.buf1[0] & 7 except OSError: raise MPUException(self._I2Cerror) return res
@accel_filter_range.setter def accel_filter_range(self, filt): ''' Sets the accel low pass filter cutoff frequency Pass: 0 1 2 3 4 5 6 7 BEWARE 7 doesn't work on device I tried. Cutoff (Hz): 460 184 92 41 20 10 5 460 Sample rate (KHz): 1 1 1 1 1 1 1 1 ''' if filt in range(8): try: self._write(filt, 0x1D, self.mpu_addr) except OSError: raise MPUException(self._I2Cerror) else: raise ValueError('Filter coefficient must be between 0 and 7')
def _magsetup(self): # mode 2 100Hz continuous reads, 16 bit ''' Magnetometer initialisation: use 16 bit continuous mode. Mode 1 is 8Hz mode 2 is 100Hz repetition returns correction values ''' try: self._write(0x0F, 0x0A, self._mag_addr) # fuse ROM access mode self._read(self.buf3, 0x10, self._mag_addr) # Correction values self._write(0, 0x0A, self._mag_addr) # Power down mode (AK8963 manual 6.4.6) self._write(0x16, 0x0A, self._mag_addr) # 16 bit (0.15uT/LSB not 0.015), mode 2 except OSError: raise MPUException(self._I2Cerror) mag_x = (0.5*(self.buf3[0] - 128))/128 + 1 mag_y = (0.5*(self.buf3[1] - 128))/128 + 1 mag_z = (0.5*(self.buf3[2] - 128))/128 + 1 return (mag_x, mag_y, mag_z)
@property def mag(self): ''' Magnetomerte object ''' return self._mag
def _mag_callback(self): ''' Update magnetometer Vector3d object (if data available) ''' try: # If read fails, returns last valid data and self._read(self.buf1, 0x02, self._mag_addr) # increments mag_stale_count if self.buf1[0] & 1 == 0: return self._mag # Data not ready: return last value self._read(self.buf6, 0x03, self._mag_addr) self._read(self.buf1, 0x09, self._mag_addr) except OSError: raise MPUException(self._I2Cerror) if self.buf1[0] & 0x08 > 0: # An overflow has occurred self._mag_stale_count += 1 # Error conditions retain last good value return # user should check for ever increasing stale_counts self._mag._ivector[1] = bytes_toint(self.buf6[1], self.buf6[0]) # Note axis twiddling and little endian self._mag._ivector[0] = bytes_toint(self.buf6[3], self.buf6[2]) self._mag._ivector[2] = -bytes_toint(self.buf6[5], self.buf6[4]) scale = 0.15 # scale is 0.15uT/LSB self._mag._vector[0] = self._mag._ivector[0]*self.mag_correction[0]*scale self._mag._vector[1] = self._mag._ivector[1]*self.mag_correction[1]*scale self._mag._vector[2] = self._mag._ivector[2]*self.mag_correction[2]*scale self._mag_stale_count = 0
@property def mag_stale_count(self): ''' Number of consecutive times where old data was returned ''' return self._mag_stale_count
def get_mag_irq(self): ''' Uncorrected values because floating point uses heap ''' self._read(self.buf1, 0x02, self._mag_addr) if self.buf1[0] == 1: # Data is ready self._read(self.buf6, 0x03, self._mag_addr) self._read(self.buf1, 0x09, self._mag_addr) # Mandatory status2 read self._mag._ivector[1] = 0 if self.buf1[0] & 0x08 == 0: # No overflow has occurred self._mag._ivector[1] = bytes_toint(self.buf6[1], self.buf6[0]) self._mag._ivector[0] = bytes_toint(self.buf6[3], self.buf6[2]) self._mag._ivector[2] = -bytes_toint(self.buf6[5], self.buf6[4])


MPU9250



I2C接线图


使用MPU6050 Yaw轴会不可避免的飘动,MPU9250自带磁力计,可以减少飘动。


# 控制电机函数
from pyb import Timer,Pin,ADCimport time
class Motor(): # 电机pwm初始化 def __init__(self,isInit=False): timerMotor_1 = Timer(3, freq=50) timerMotor_2 = Timer(4, freq=50) self.motor1 = timerMotor_1.channel(1, Timer.PWM, pin=Pin('B4')) self.motor2 = timerMotor_1.channel(2, Timer.PWM, pin=Pin('B5')) self.motor3 = timerMotor_2.channel(3, Timer.PWM, pin=Pin('B8')) self.motor4 = timerMotor_2.channel(4, Timer.PWM, pin=Pin('B9')) self.motors = [self.motor1,self.motor2,self.motor3,self.motor4] # self.x = ADC(Pin('X2')) # self.btn_stop = Pin('X4',Pin.IN) if not isInit: for moto in self.motors: self.MotoSet(moto) time.sleep(1) self.MotosPwmUpdate([0,0,0,0]) # 电机初始化 设置最高油门和最低油门 def MotoSet(self,moto): moto.pulse_width_percent(10) time.sleep(2) moto.pulse_width_percent(5)
# pwm 更新函数 1 # 可以用于调试单个电机 def MotoPwmUpdate(self,n,pwm): if pwm < 0 or pwm > 100: return None self.motors[n].pulse_width_percent(5 + pwm*5/100)
# pwm 更新函数 2 # 用于实际飞行 def MotosPwmUpdate(self,pwms): for moto,pwm in zip(self.motors,pwms): moto.pulse_width_percent(5 + pwm*5/100)
# 电机停止转动 # 用于紧急制动和测试 def MotoStop(self): for moto in self.motors: moto.pulse_width_percent(5)


电调控制,PWM实现


然后就是PWM的接收机太费引脚,换PPM,问题在于转换~


上面是PPM

下面是PWM



标准的PPM信号,以0.4ms的低电平为起始标识。后边以电平的上升沿的间隔时间来表达各个通道的控制量。一般排列10个上升沿后,电平保持高电平,直到重复下一个PPM信号。


PPM信号可以看做是一帧数据,它包含了8个通道的信息。每个上升沿间隔时间刚好等于PWM信号的高电平持续时间,也就1000us~2000us之间。

PPM的重复周期也为20ms,也是50hz的刷新频率。


import pybimport micropythonmicropython.alloc_emergency_exception_buf(100)
# Futaba PPM decoder# http://diydrones.com/profiles/blogs/705844:BlogPost:38393class Decoder():
def __init__(self, pin: str): self.pin = pin self.current_channel = -1 self.channels = [0] * 20 # up to 10 channels self.timer = pyb.Timer(2, prescaler=83, period=0x3fffffff) self.timer.counter(0) # clear any previously set interrupt pyb.ExtInt(pin, pyb.ExtInt.IRQ_RISING, pyb.Pin.PULL_NONE, None) self.ext_int = pyb.ExtInt(pin, pyb.ExtInt.IRQ_RISING, pyb.Pin.PULL_NONE, self._callback)
def _callback(self, t): ticks = self.timer.counter() if ticks > 5000: self.current_channel = 0 elif self.current_channel > -1: self.channels[self.current_channel] = ticks self.current_channel += 1 self.timer.counter(0)
def get_channel_value(self, channel: int) -> int: return self.channels[channel]
def enable(self): self.ext_int.enable()
def disable(self):        self.ext_int.disable()


PPM驱动代码



# 匿名上位机 移植上位机、协议版本v4.34# 移植部分功能# 关闭了数据的校验功能,提高发送速度class ANO():    def  __init__(self,uart):        self.writechar = uart.writechar        self.BYTE0 = lambda x : (x>>0)&0xff        self.BYTE1 = lambda x : (x>>8)&0xff        self.BYTE2 = lambda x : (x>>16)&0xff        self.BYTE3 = lambda x : (x>>24)&0xff        pass
# 发送数据 def ANO_Send_Data(self,data): for v in data: self.writechar(v)
#send 数据 def cs(self,data_to_send): _cnt = len(data_to_send) data_to_send[3] = _cnt - 4 data_to_send.append(0) # 需要校验功能的将这一行改了就行 self.ANO_Send_Data(data_to_send)
# 飞机姿态 高度 飞行模式 是否解锁 def ANO_DT_Send_Status(self,angle_rol, angle_pit, angle_yaw, alt, fly_model, armed):

_cnt = 0 _temp = 0 _temp2 = alt data_to_send = [] data_to_send.append(0xAA) data_to_send.append(0xAA) data_to_send.append(0x01) data_to_send.append(0)
_temp = int(angle_rol * 100) data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = int(angle_pit * 100) data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = int(angle_yaw * 100) data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
data_to_send.append(self.BYTE3(_temp2)) data_to_send.append(self.BYTE2(_temp2)) data_to_send.append(self.BYTE1(_temp2)) data_to_send.append(self.BYTE0(_temp2))
data_to_send.append(fly_model)
data_to_send.append(armed) self.cs(data_to_send)
# PID信息 def ANO_DT_Send_PID(self,group, p1_p, p1_i, p1_d, p2_p, p2_i, p2_d, p3_p, p3_i, p3_d): _cnt = 0 _temp = 0 data_to_send = [] data_to_send.append(0xAA) data_to_send.append(0xAA) data_to_send.append(0x10 + group -1) data_to_send.append(0)
_temp = p1_p * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = p1_i * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = p1_d * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
_temp = p2_p * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = p2_i * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = p2_d * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
_temp = p3_p * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = p3_i * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = p3_d * 1000 data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
self.cs(data_to_send)
# 传感器数据 def ANO_DT_Send_Senser(self,a_x, a_y, a_z, g_x, g_y, g_z, m_x, m_y, m_z): _cnt = 0 _temp = 0 data_to_send = [] data_to_send.append(0xAA) data_to_send.append(0xAA) data_to_send.append(0x02) data_to_send.append(0)
_temp = a_x data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = a_y data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = a_z data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
_temp = g_x data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = g_y data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = g_z data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
_temp = m_x data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = m_y data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp)) _temp = m_z data_to_send.append(self.BYTE1(_temp)) data_to_send.append(self.BYTE0(_temp))
self.cs(data_to_send)
# 电机pwm数据 0-1000 def ANO_DT_Send_MotoPWM(self, m_1, m_2, m_3, m_4, m_5, m_6, m_7, m_8): _cnt = 0
data_to_send = [] data_to_send.append(0xAA) data_to_send.append(0xAA) data_to_send.append(0x06) data_to_send.append(0)
data_to_send.append(self.BYTE1(m_1)) data_to_send.append(self.BYTE0(m_1)) data_to_send.append(self.BYTE1(m_2)) data_to_send.append(self.BYTE0(m_2)) data_to_send.append(self.BYTE1(m_3)) data_to_send.append(self.BYTE0(m_3)) data_to_send.append(self.BYTE1(m_4)) data_to_send.append(self.BYTE0(m_4)) data_to_send.append(self.BYTE1(m_5)) data_to_send.append(self.BYTE0(m_5)) data_to_send.append(self.BYTE1(m_6)) data_to_send.append(self.BYTE0(m_6)) data_to_send.append(self.BYTE1(m_7)) data_to_send.append(self.BYTE0(m_7)) data_to_send.append(self.BYTE1(m_8)) data_to_send.append(self.BYTE0(m_8))
        self.cs(data_to_send)


匿名上位机若干协议移植,2.4G控制


下面是个相似的项目:


  • 惯性测量单元 (IMU)

  • SBUS接口(用于连接SBUS RC接收器)



就是这么简单的东西。。。




airPy 代码

airPy代码分为3个模块:

  • airPy 固件:运行在 pyboard 上的 python 代码

  • airPy 地面站:在 PC 上运行的 JavaFx 代码,用于配置/调整 airPy 板

  • airPy Link Protocol:用于板卡与地面站通信的串行协议


airPy Link Protocol目前仅用于配置和调整目的。

它是一种可变大小的串行协议,用于 pyboard 和地面站之间的数据通信。

这个想法是不仅在 USB 上而且在 Wi-fi、蓝牙和 ad hoc RF 通道(例如 433MHz)上使用这个协议。


每个 APLINK 消息都包含一个标头和一个可变大小的有效负载(取决于内容),如以下架构中所指定。












参考文章:


https://github.com/micropython-IMU/micropython-mpu9x50

microPython-MPU9x50驱动

https://www.jianshu.com/u/38cc558a2c62

micropython驱动的移植者

https://blog.csdn.net/qq_38721302/article/details/83095545

STM32 驱动MPU6050(使用了本文的寄存器定义)

https://github.com/Sokrates80/air-py

特别的,有一个microPython的飞控开源项目了,减少了我的工作量。

https://github.com/Sokrates80/airPy-GS

相应搭配的地面站

https://github.com/micropython-IMU/micropython-fusion

MicroPython 的单个平台上获取传感器数据并执行融合的情况

https://github.com/wagnerc4/flight_controller

另外一个完整的项目

https://github.com/ArduPilot/ardupilot

pid参考了这里,APM飞控

http://www.air-py.com/

相关网站

开源的mpy飞控系统



评论
  • 随着智慧科技的快速发展,智能显示器的生态圈应用变得越来越丰富多元,智能显示器不仅仅是传统的显示设备,透过结合人工智能(AI)和语音助理,它还可以成为家庭、办公室和商业环境中的核心互动接口。提供多元且个性化的服务,如智能家居控制、影音串流拨放、实时信息显示等,极大提升了使用体验。此外,智能家居系统的整合能力也不容小觑,透过智能装置之间的无缝连接,形成了强大的多元应用生态圈。企业也利用智能显示器进行会议展示和多方远程合作,大大提高效率和互动性。Smart Display Ecosystem示意图,作
    百佳泰测试实验室 2025-01-16 15:37 172浏览
  • 随着消费者对汽车驾乘体验的要求不断攀升,汽车照明系统作为确保道路安全、提升驾驶体验以及实现车辆与环境交互的重要组成,日益受到业界的高度重视。近日,2024 DVN(上海)国际汽车照明研讨会圆满落幕。作为照明与传感创新的全球领导者,艾迈斯欧司朗受邀参与主题演讲,并现场展示了其多项前沿技术。本届研讨会汇聚来自全球各地400余名汽车、照明、光源及Tier 2供应商的专业人士及专家共聚一堂。在研讨会第一环节中,艾迈斯欧司朗系统解决方案工程副总裁 Joachim Reill以深厚的专业素养,主持该环节多位
    艾迈斯欧司朗 2025-01-16 20:51 109浏览
  • 一个易用且轻量化的UI可以大大提高用户的使用效率和满意度——通过快速启动、直观操作和及时反馈,帮助用户快速上手并高效完成任务;轻量化设计则可以减少资源占用,提升启动和运行速度,增强产品竞争力。LVGL(Light and Versatile Graphics Library)是一个免费开源的图形库,专为嵌入式系统设计。它以轻量级、高效和易于使用而著称,支持多种屏幕分辨率和硬件配置,并提供了丰富的GUI组件,能够帮助开发者轻松构建出美观且功能强大的用户界面。近期,飞凌嵌入式为基于NXP i.MX9
    飞凌嵌入式 2025-01-16 13:15 199浏览
  • 80,000人到访的国际大展上,艾迈斯欧司朗有哪些亮点?感未来,光无限。近日,在慕尼黑electronica 2024现场,ams OSRAM通过多款创新DEMO展示,以及数场前瞻洞察分享,全面展示自身融合传感器、发射器及集成电路技术,精准捕捉并呈现环境信息的卓越能力。同时,ams OSRAM通过展会期间与客户、用户等行业人士,以及媒体朋友的深度交流,向业界传达其以光电技术为笔、以创新为墨,书写智能未来的深度思考。electronica 2024electronica 2024构建了一个高度国际
    艾迈斯欧司朗 2025-01-16 20:45 146浏览
  • 实用性高值得收藏!! (时源芯微)时源专注于EMC整改与服务,配备完整器件 TVS全称Transient Voltage Suppre,亦称TVS管、瞬态抑制二极管等,有单向和双向之分。单向TVS 一般应用于直流供电电路,双向TVS 应用于电压交变的电路。在直流电路的应用中,TVS被并联接入电路中。在电路处于正常运行状态时,TVS会保持截止状态,从而不对电路的正常工作产生任何影响。然而,一旦电路中出现异常的过电压,并且这个电压达到TVS的击穿阈值时,TVS的状态就会
    时源芯微 2025-01-16 14:23 152浏览
  • 百佳泰特为您整理2025年1月各大Logo的最新规格信息,本月有更新信息的logo有HDMI、Wi-Fi、Bluetooth、DisplayHDR、ClearMR、Intel EVO。HDMI®▶ 2025年1月6日,HDMI Forum, Inc. 宣布即将发布HDMI规范2.2版本。新规范将支持更高的分辨率和刷新率,并提供更多高质量选项。更快的96Gbps 带宽可满足数据密集型沉浸式和虚拟应用对传输的要求,如 AR/VR/MR、空间现实和光场显示,以及各种商业应用,如大型数字标牌、医疗成像和
    百佳泰测试实验室 2025-01-16 15:41 157浏览
  • 晶台光耦KL817和KL3053在小家电产品(如微波炉等)辅助电源中的广泛应用。具备小功率、高性能、高度集成以及低待机功耗的特点,同时支持宽输入电压范围。▲光耦在实物应用中的产品图其一次侧集成了交流电压过零检测与信号输出功能,该功能产生的过零信号可用于精确控制继电器、可控硅等器件的过零开关动作,从而有效减小开关应力,显著提升器件的使用寿命。通过高度的集成化和先进的控制技术,该电源大幅减少了所需的外围器件数量,不仅降低了系统成本和体积,还进一步增强了整体的可靠性。▲电路示意图该电路的过零检测信号由
    晶台光耦 2025-01-16 10:12 97浏览
  • 日前,商务部等部门办公厅印发《手机、平板、智能手表(手环)购新补贴实施方案》明确,个人消费者购买手机、平板、智能手表(手环)3类数码产品(单件销售价格不超过6000元),可享受购新补贴。每人每类可补贴1件,每件补贴比例为减去生产、流通环节及移动运营商所有优惠后最终销售价格的15%,每件最高不超过500元。目前,京东已经做好了承接手机、平板等数码产品国补优惠的落地准备工作,未来随着各省市关于手机、平板等品类的国补开启,京东将第一时间率先上线,满足消费者的换新升级需求。为保障国补的真实有效发放,基于
    华尔街科技眼 2025-01-17 10:44 129浏览
  • 近期,智能家居领域Matter标准的制定者,全球最具影响力的科技联盟之一,连接标准联盟(Connectivity Standards Alliance,简称CSA)“利好”频出,不仅为智能家居领域的设备制造商们提供了更为快速便捷的Matter认证流程,而且苹果、三星与谷歌等智能家居平台厂商都表示会接纳CSA的Matter认证体系,并计划将其整合至各自的“Works with”项目中。那么,在本轮“利好”背景下,智能家居的设备制造商们该如何捉住机会,“掘金”万亿市场呢?重认证快通道计划,为家居设备
    华普微HOPERF 2025-01-16 10:22 176浏览
  • 全球领先的光学解决方案供应商艾迈斯欧司朗(SIX:AMS)近日宣布,与汽车技术领先者法雷奥合作,采用创新的开放系统协议(OSP)技术,旨在改变汽车内饰照明方式,革新汽车行业座舱照明理念。结合艾迈斯欧司朗开创性的OSIRE® E3731i智能LED和法雷奥的动态环境照明系统,两家公司将为车辆内饰设计和功能设立一套全新标准。汽车内饰照明的作用日益凸显,座舱设计的主流趋势应满足终端用户的需求:即易于使用、个性化,并能提供符合用户生活方式的清晰信息。因此,动态环境照明带来了众多新机遇。智能LED的应用已
    艾迈斯欧司朗 2025-01-15 19:00 78浏览
  • 电竞鼠标应用环境与客户需求电竞行业近年来发展迅速,「鼠标延迟」已成为决定游戏体验与比赛结果的关键因素。从技术角度来看,传统鼠标的延迟大约为20毫秒,入门级电竞鼠标通常为5毫秒,而高阶电竞鼠标的延迟可降低至仅2毫秒。这些差异看似微小,但在竞技激烈的游戏中,尤其在对反应和速度要求极高的场景中,每一毫秒的优化都可能带来致胜的优势。电竞比赛的普及促使玩家更加渴望降低鼠标延迟以提升竞技表现。他们希望通过精确的测试,了解不同操作系统与设定对延迟的具体影响,并寻求最佳配置方案来获得竞技优势。这样的需求推动市场
    百佳泰测试实验室 2025-01-16 15:45 236浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦