Harmony鸿蒙南向驱动开发-DAC

DAC(Digital to Analog Converter)是一种通过电流、电压或电荷的形式将数字信号转换为模拟信号的设备。

DAC模块支持数模转换的开发。它主要用于:

  • 作为过程控制计算机系统的输出通道,与执行器相连,实现对生产过程的自动控制。

  • 在利用反馈技术的模数转换器设计中,作为重要的功能模块呈现。

基本概念

  • 分辨率

    分辨率指的是D/A转换器能够转换的二进制位数,位数越多分辨率越高。

  • 转换精度

    精度是指输入端加有最大数值时,DAC的实际输出值和理论计算值之差,DAC转换器的转换精度与DAC转换器的集成芯片结构和接口电路配置有关。理想情况下,DAC的转换精度越小越好,因此为了获得更高精度的DAC转换结果,首先要保证选择的DAC转换器具备足够高的分辨率。其次,要保证接口电路的器件或电源误差最小或者不存在误差,否则会造成DAC转换的误差,若这些误差超过一定程度,就会导致DAC转换错误。

  • 转换速度

    转换速度一般由建立时间决定。从输入由全0突变为全1时开始,到输出电压稳定在FSR±½LSB范围(或以FSR±x%FSR指明范围)内为止,这段时间称为建立时间,它是DAC的最大响应时间,所以用它衡量转换速度的快慢。

    • 满量程范围FSR(Full Scale Range),是指DAC输出信号幅度的最大范围,不同的DAC有不同的满量程范围,该范围可以用正、负电流或者正、负电压来限制。

    • 最低有效位LSB(Least Significant Byte),指的是一个二进制数字中的第0位(即最低位)。

运作机制

在HDF框架中,同类型设备对象较多时(可能同时存在十几个同类型配置器),若采用独立服务模式,则需要配置更多的设备节点,且相关服务会占据更多的内存资源。相反,采用统一服务模式可以使用一个设备服务作为管理器,统一处理所有同类型对象的外部访问(这会在配置文件中有所体现),实现便捷管理和节约资源的目的。DAC模块即采用统一服务模式(如图1所示)。

DAC模块各分层的作用为:

  • 接口层:提供打开设备、写入数据和关闭设备接口的能力。

  • 核心层:主要提供绑定设备、初始化设备以及释放设备的能力。

  • 适配层:由驱动适配者实现与硬件相关的具体功能,如控制器的初始化等。

在统一模式下,所有的控制器都被核心层统一管理,并由核心层统一发布一个服务供接口层,因此这种模式下驱动无需再为每个控制器发布服务。

说明:
核心层可以调用接口层的函数,也可以通过钩子函数调用适配层函数,从而使得适配层间接的可以调用接口层函数,但是不可逆转接口层调用适配层函数。

图 1 DAC统一服务模式结构图

DAC统一服务模式结构图

约束与限制

DAC模块当前仅支持轻量和小型系统内核(LiteOS-A)。

开发指导

场景介绍

DAC模块主要在设备中数模转换、音频输出和电机控制等设备使用,设置将DAC模块传入的数字信号转换为输出模拟信号时需要用到DAC数模转换驱动。当驱动开发者需要将DAC设备适配到OpenHarmony时,需要进行DAC驱动适配,下文将介绍如何进行DAC驱动适配。

接口说明

为了保证上层在调用DAC接口时能够正确的操作硬件,核心层在//drivers/hdf_core/framework/support/platform/include/dac/dac_core.h中定义了以下钩子函数。驱动适配者需要在适配层实现这些函数的具体功能,并与这些钩子函数挂接,从而完成接口层与核心层的交互。

DacMethod和DacLockMethod定义:

struct DacMethod {// 写入数据的钩子函数int32_t (*write)(struct DacDevice *device, uint32_t channel, uint32_t val);// 启动DAC设备的钩子函数int32_t (*start)(struct DacDevice *device);// 停止DAC设备的钩子函数int32_t (*stop)(struct DacDevice *device);
};struct DacLockMethod {int32_t (*lock)(struct DacDevice *device);void (*unlock)(struct DacDevice *device);
};

在适配层中,DacMethod必须被实现,DacLockMethod可根据实际情况考虑是否实现。核心层提供了默认的DacLockMethod,其中使用Spinlock作为保护临界区的锁:

static int32_t DacDeviceLockDefault(struct DacDevice *device)
{if (device == NULL) {HDF_LOGE("%s: device is null", __func__);return HDF_ERR_INVALID_OBJECT;}return OsalSpinLock(&device->spin);
}static void DacDeviceUnlockDefault(struct DacDevice *device)
{if (device == NULL) {HDF_LOGE("%s: device is null", __func__);return;}(void)OsalSpinUnlock(&device->spin);
}static const struct DacLockMethod g_dacLockOpsDefault = {.lock = DacDeviceLockDefault,.unlock = DacDeviceUnlockDefault,
};

若实际情况不允许使用Spinlock,驱动适配者可以考虑使用其他类型的锁来实现一个自定义的DacLockMethod。一旦实现了自定义的DacLockMethod,默认的DacLockMethod将被覆盖。

表 1 DacMethod结构体成员的钩子函数功能说明

函数成员入参出参返回值功能
writedevice:结构体指针,核心层DAC控制器
channel:uint32_t类型,传入的通道号
val:uint32_t类型,要传入的数据
HDF_STATUS相关状态写入DA的目标值
startdevice:结构体指针,核心层DAC控制器HDF_STATUS相关状态开启DAC设备
stopdevice:结构体指针,核心层DAC控制器HDF_STATUS相关状态关闭DAC设备

表 2 DacLockMethod结构体成员函数功能说明

函数成员入参出参返回值功能
lockdevice:结构体指针,核心层DAC设备对象。HDF_STATUS相关状态获取临界区锁
unlockdevice:结构体指针,核心层DAC设备对象。HDF_STATUS相关状态释放临界区锁

开发步骤

DAC模块适配包含以下四个步骤:

  1. 实例化驱动入口

  2. 配置属性文件

  3. 实例化核心层函数

  4. 驱动调试

开发实例

下方将Hi3516DV300的驱动//device/soc/hisilicon/common/platform/dac/dac_hi35xx.c为例,展示驱动适配者需要提供哪些内容来完整实现设备功能。

  1. 实例化驱动入口

    驱动开发首先需要实例化驱动入口,驱动入口必须为HdfDriverEntry(在hdf_device_desc.h中定义)类型的全局变量,且moduleName要和//vendor/hisilicon/hispark_taurus/hdf_config/device_info/device_info.hcs中保持一致。HDF框架会汇总所有加载的驱动的HdfDriverEntry对象入口,形成一个类似数组的段地址空间,方便上层调用。

    一般在加载驱动时HDF会先调用Init函数加载该驱动。当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。

    static struct HdfDriverEntry g_dacDriverEntry = {.moduleVersion = 1,.Init = VirtualDacInit,.Release = VirtualDacRelease,.moduleName = "virtual_dac_driver",       //【必要且与device_info.hcs文件内的模块名匹配】
    };
    HDF_INIT(g_dacDriverEntry);                   // 调用HDF_INIT将驱动入口注册到HDF框架中// 核心层dac_core.c管理器服务的驱动入口
    struct HdfDriverEntry g_dacManagerEntry = {.moduleVersion = 1,.Bind = DacManagerBind,                   // DAC不需要实现Bind,本例是一个空实现,驱动适配者可根据自身需要添加相关操作.Init = DacManagerInit,                   // 见Init参考.Release = DacManagerRelease,             // 见Release参考.moduleName = "HDF_PLATFORM_DAC_MANAGER", // 这与device_info.hcs文件中device0对应
    };
    HDF_INIT(g_dacManagerEntry);                  // 调用HDF_INIT将驱动入口注册到HDF框架中
  2. 配置属性文件

    • 添加//vendor/hisilicon/hispark_taurus/hdf_config/device_info/device_info.hcs器件属性文件。

      器件属性值对于驱动适配者的驱动实现以及核心层DacDevice相关成员的默认值或限制范围有密切关系,比如设备通道的个数以及传输速率的最大值,会影响DacDevice相关成员的默认值。

      由于采用了统一服务模式,device_info.hcs文件中第一个设备节点必须为DAC管理器,其各项参数如表3所示:

      表 3 device_info.hcs节点参数说明

      成员名
      policy驱动服务发布的策略,DAC管理器具体配置为2,表示驱动对内核态和用户态都发布服务
      priority驱动启动优先级(0-200),值越大优先级越低,优先级相同则不保证device的加载顺序。DAC管理器具体配置为52
      permission驱动创建设备节点权限,DAC管理器具体配置为0664
      moduleName驱动名称,DAC管理器固定为HDF_PLATFORM_DAC_MANAGER
      serviceName驱动对外发布服务的名称,DAC管理器服务名设置为HDF_PLATFORM_DAC_MANAGER
      deviceMatchAttr驱动私有数据匹配的关键字,DAC管理器没有使用,可忽略

      从第二个节点开始配置具体DAC控制器信息,此节点并不表示某一路DAC控制器,而是代表一个资源性质设备,用于描述一类DAC控制器的信息。本例只有一个DAC设备,如有多个设备,则需要在device_info.hcs文件增加deviceNode信息,以及在dac_config.hcs文件中增加对应的器件属性。

    • device_info.hcs配置参考

      root {device_dac :: device {// device0是DAC管理器device0 :: deviceNode {policy = 0;priority = 52;permission = 0644;serviceName = "HDF_PLATFORM_DAC_MANAGER";moduleName = "HDF_PLATFORM_DAC_MANAGER";}}// dac_virtual是DAC控制器dac_virtual :: deviceNode {policy = 0;priority = 56;permission = 0644;moduleName = "virtual_dac_driver";        // 【必要】用于指定驱动名称,需要与期望的驱动Entry中的moduleName一致。serviceName = "VIRTUAL_DAC_DRIVER";       // 【必要】驱动对外发布服务的名称,必须唯一。deviceMatchAttr = "virtual_dac";          // 【必要】用于配置控制器私有数据,要与dac_config.hcs中对应控制器保持一致。}
      }
    • 添加dac_test_config.hcs器件属性文件。

      在具体产品对应目录下新增文件用于驱动配置参数(例如hispark_taurus开发板:vendor/hisilicon/hispark_taurus/hdf_config/hdf_test/dac_test_config.hcs),其中配置参数如下:

      root {platform {dac_config {match_attr = "virtual_dac"; // 【必要】需要和device_info.hcs中的deviceMatchAttr值一致template dac_device {deviceNum = 0;          // 设备号     validChannel = 0x1;     // 有效通道1rate = 20000;           // 速率}device_0 :: dac_device {deviceNum = 0;          // 设备号validChannel = 0x2;     // 有效通道2}}}
      }

      需要注意的是,新增dac_config.hcs配置文件后,必须在hdf.hcs文件中将其包含,否则配置文件无法生效。

      例如:本例中dac_config.hcs所在路径为device/soc/hisilicon/hi3516dv300/sdk_liteos/hdf_config/dac/dac_config.hcs,则必须在产品对应的hdf.hcs中添加如下语句:

      #include "../../../../device/soc/hisilicon/hi3516dv300/sdk_liteos/hdf_config/dac/dac_config.hcs" // 配置文件相对路径
  3. 实例化核心层函数

    • 初始化DacDevice成员。

      在VirtualDacParseAndInit函数中对DacDevice成员进行初始化操作。

      // 虚拟驱动自定义结构体
      struct VirtualDacDevice {// DAC设备结构体struct DacDevice device;// DAC设备号uint32_t deviceNum;// 有效通道uint32_t validChannel;// DAC速率uint32_t rate;
      };
      // 解析并且初始化核心层DacDevice对象
      static int32_t VirtualDacParseAndInit(struct HdfDeviceObject *device, const struct DeviceResourceNode *node)
      {// 定义返回值int32_t ret;// DAC设备虚拟指针struct VirtualDacDevice *virtual = NULL;(void)device;// 给virtual指针开辟空间virtual = (struct VirtualDacDevice *)OsalMemCalloc(sizeof(*virtual));if (virtual == NULL) {// 为空则返回错误参数HDF_LOGE("%s: Malloc virtual fail!", __func__);return HDF_ERR_MALLOC_FAIL;}// 读取属性文件配置参数ret = VirtualDacReadDrs(virtual, node);if (ret != HDF_SUCCESS) {// 读取失败HDF_LOGE("%s: Read drs fail! ret:%d", __func__, ret);// 释放virtual空间OsalMemFree(virtual);// 指针置为NULLvirtual = NULL;return ret;}// 初始化虚拟指针VirtualDacDeviceInit(virtual);// 对DacDevice中priv对象初始化virtual->device.priv = (void *)node;// 对DacDevice中devNum对象初始化virtual->device.devNum = virtual->deviceNum;// 对DacDevice中ops对象初始化virtual->device.ops = &g_method;// 添加DAC设备ret = DacDeviceAdd(&virtual->device);if (ret != HDF_SUCCESS) {// 添加设备失败HDF_LOGE("%s: add Dac controller failed! ret = %d", __func__, ret);// 释放virtual空间OsalMemFree(virtual);// 虚拟指针置空virtual = NULL;return ret;}return HDF_SUCCESS;
      }
    • 自定义结构体参考。

      通过自定义结构体定义DAC数模转换必要的参数,在定义结构体时需要根据设备的功能参数来实现自定义结构体,从驱动的角度看,自定义结构体是参数和数据的载体,dac_config.hcs文件中传递的参数和数据会被HDF驱动模块的DacTestReadConfig函数读入,通过DeviceResourceIface来初始化结构体成员,其中一些重要数值也会传递给核心层DacDevice对象,例如设备号、总线号等。

      struct VirtualDacDevice {struct DacDevice device; //【必要】是核心层控制对象,具体描述见下面uint32_t deviceNum;      //【必要】设备号uint32_t validChannel;   //【必要】有效通道uint32_t rate;           //【必要】采样率
      };// DacDevice是核心层控制器结构体,其中的成员在Init函数中会被赋值。
      struct DacDevice {const struct DacMethod *ops;OsalSpinlock spin;       // 自旋锁uint32_t devNum;         // 设备号uint32_t chanNum;        // 设备通道号const struct DacLockMethod *lockOps;void *priv;
      };
    • 实例化DacDevice成员DacMethod。

      VirtualDacWrite、VirtualDacStop、VirtualDacStart函数会在dac_virtual.c文件中进行模块功能的实例化。

      static const struct DacMethod g_method = {.write = VirtualDacWrite, // DAC设备写入值.stop = VirtualDacStop,   // 停止DAC设备.start = VirtualDacStart, // 开始启动DAC设备
      };

      说明:
      DacDevice成员DacMethod的定义和成员说明见接口说明。

    • Init函数开发参考

      入参:

      HdfDeviceObject这个是整个驱动对外提供的接口参数,具备HCS配置文件的信息。

      返回值:

      HDF_STATUS相关状态(表4为部分展示,如需使用其他状态,可参考//drivers/hdf_core/interfaces/inner_api/utils/hdf_base.h中HDF_STATUS定义)。

      表 4 HDF_STATUS相关状态说明

      状态(值)问题描述
      HDF_ERR_INVALID_OBJECT控制器对象非法。
      HDF_ERR_INVALID_PARAM参数非法。
      HDF_ERR_MALLOC_FAIL内存分配失败。
      HDF_ERR_IOI/O 错误。
      HDF_SUCCESS传输成功。
      HDF_FAILURE传输失败。

      函数说明:

      初始化自定义结构体对象,初始化DacDevice成员,并调用核心层DacDeviceAdd函数。

      static int32_t VirtualDacParseAndInit(struct HdfDeviceObject *device, const struct DeviceResourceNode *node)
      {// 定义返回值参数int32_t ret;// DAC设备的结构体指针struct VirtualDacDevice *virtual = NULL;(void)device;// 分配指定大小的内存virtual = (struct VirtualDacDevice *)OsalMemCalloc(sizeof(*virtual));if (virtual == NULL) {// 分配内存失败HDF_LOGE("%s: Malloc virtual fail!", __func__);return HDF_ERR_MALLOC_FAIL;}// 读取hcs中的node节点参数,函数定义见下方ret = VirtualDacReadDrs(virtual, node);if (ret != HDF_SUCCESS) {// 读取节点失败HDF_LOGE("%s: Read drs fail! ret:%d", __func__, ret);goto __ERR__;}// 初始化DAC设备指针VirtualDacDeviceInit(virtual);// 节点数据传入私有数据virtual->device.priv = (void *)node;// 传入设备号virtual->device.devNum = virtual->deviceNum;// 传入方法virtual->device.ops = &g_method;// 添加DAC设备ret = DacDeviceAdd(&virtual->device);if (ret != HDF_SUCCESS) {// 添加DAC设备失败HDF_LOGE("%s: add Dac controller failed! ret = %d", __func__, ret);goto __ERR__;}// 成功添加DAC设备return HDF_SUCCESS;
      __ERR__:// 如果指针为空if (virtual != NULL) {// 释放内存OsalMemFree(virtual);// 指针置空virtual = NULL;}return ret;
      }static int32_t VirtualDacInit(struct HdfDeviceObject *device)
      {// 定义返回值参数int32_t ret;// 设备结构体子节点const struct DeviceResourceNode *childNode = NULL;// 入参指针进行判断if (device == NULL || device->property == NULL) {// 入参指针为空HDF_LOGE("%s: device or property is NULL", __func__);return HDF_ERR_INVALID_OBJECT;}// 入参指针不为空ret = HDF_SUCCESS;DEV_RES_NODE_FOR_EACH_CHILD_NODE(device->property, childNode) {// 解析子节点ret = VirtualDacParseAndInit(device, childNode);if (ret != HDF_SUCCESS) {// 解析失败break;}}// 解析成功return ret;
      }static int32_t VirtualDacReadDrs(struct VirtualDacDevice *virtual, const struct DeviceResourceNode *node)
      {struct DeviceResourceIface *drsOps = NULL;// 获取drsOps方法drsOps = DeviceResourceGetIfaceInstance(HDF_CONFIG_SOURCE);if (drsOps == NULL || drsOps->GetUint32 == NULL || drsOps->GetUint16 == NULL) {HDF_LOGE("%s: Invalid drs ops fail!", __func__);return HDF_FAILURE;}// 将配置参数依次读出,并填充至结构体中if (drsOps->GetUint32(node, "deviceNum", &virtual->deviceNum, 0) != HDF_SUCCESS) {HDF_LOGE("%s: Read deviceNum fail!", __func__);return HDF_ERR_IO;}if (drsOps->GetUint32(node, "validChannel", &virtual->validChannel, 0) != HDF_SUCCESS) {HDF_LOGE("%s: Read validChannel fail!", __func__);return HDF_ERR_IO;}if (drsOps->GetUint32(node, "rate", &virtual->rate, 0) != HDF_SUCCESS) {HDF_LOGE("%s: Read rate fail!", __func__);return HDF_ERR_IO;}return HDF_SUCCESS;
      }
    • Release函数开发参考

      入参:

      HdfDeviceObject是整个驱动对外提供的接口参数,具备HCS配置文件的信息。

      返回值:

      无。

      函数说明:

      释放内存和删除控制器,该函数需要在驱动入口结构体中赋值给Release接口,当HDF框架调用Init函数初始化驱动失败时,可以调用Release释放驱动资源。

      说明:
      所有强制转换获取相应对象的操作前提是在Init函数中具备对应赋值的操作。

      static void VirtualDacRemoveByNode(const struct DeviceResourceNode *node)
      {// 定义返回值参数int32_t ret;// 定义DAC设备号int16_t devNum;// DAC设备结构体指针struct DacDevice *device = NULL;// DAC虚拟结构体指针struct VirtualDacDevice *virtual = NULL;// 设备资源接口结构体指针struct DeviceResourceIface *drsOps = NULL;// 通过实例入口获取设备资源drsOps = DeviceResourceGetIfaceInstance(HDF_CONFIG_SOURCE);// 入参指判空if (drsOps == NULL || drsOps->GetUint32 == NULL) {// 指针为空HDF_LOGE("%s: invalid drs ops fail!", __func__);return;}// 获取devNum节点的数据ret = drsOps->GetUint16(node, "devNum", (uint16_t *)&devNum, 0);if (ret != HDF_SUCCESS) {// 获取失败HDF_LOGE("%s: read devNum fail!", __func__);return;}// 获取DAC设备号device = DacDeviceGet(devNum);// 判断DAC设备号以及数据是否为空if (device != NULL && device->priv == node) {// 为空释放DAC设备号DacDevicePut(device);// 移除DAC设备号DacDeviceRemove(device);virtual = (struct VirtualDacDevice *)device;// 释放虚拟指针OsalMemFree(virtual);}return;
      }static void VirtualDacRelease(struct HdfDeviceObject *device)
      {// 定义设备资源子节点结构体指const struct DeviceResourceNode *childNode = NULL;// 入参指针判空if (device == NULL || device->property == NULL) {// 入参指针为空HDF_LOGE("%s: device or property is NULL", __func__);return;}DEV_RES_NODE_FOR_EACH_CHILD_NODE(device->property, childNode) {// 通过节点移除DACVirtualDacRemoveByNode(childNode);}
      }
  4. 驱动调试

    【可选】针对新增驱动程序,建议验证驱动基本功能,例如挂载后的测试用例是否成功等。

最后

有很多小伙伴不知道学习哪些鸿蒙开发技术?不知道需要重点掌握哪些鸿蒙应用开发知识点?而且学习时频繁踩坑,最终浪费大量时间。所以有一份实用的鸿蒙(HarmonyOS NEXT)资料用来跟着学习是非常有必要的。 

这份鸿蒙(HarmonyOS NEXT)资料包含了鸿蒙开发必掌握的核心知识要点,内容包含了ArkTS、ArkUI开发组件、Stage模型、多端部署、分布式应用开发、音频、视频、WebGL、OpenHarmony多媒体技术、Napi组件、OpenHarmony内核、Harmony南向开发、鸿蒙项目实战等等)鸿蒙(HarmonyOS NEXT)技术知识点。

希望这一份鸿蒙学习资料能够给大家带来帮助,有需要的小伙伴自行领取,限时开源,先到先得~无套路领取!!

获取这份完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

鸿蒙(HarmonyOS NEXT)最新学习路线

  •  HarmonOS基础技能

  • HarmonOS就业必备技能 
  •  HarmonOS多媒体技术

  • 鸿蒙NaPi组件进阶

  • HarmonOS高级技能

  • 初识HarmonOS内核 
  • 实战就业级设备开发

有了路线图,怎么能没有学习资料呢,小编也准备了一份联合鸿蒙官方发布笔记整理收纳的一套系统性的鸿蒙(OpenHarmony )学习手册(共计1236页)鸿蒙(OpenHarmony )开发入门教学视频,内容包含:ArkTS、ArkUI、Web开发、应用模型、资源分类…等知识点。

获取以上完整版高清学习路线,请点击→纯血版全套鸿蒙HarmonyOS学习资料

《鸿蒙 (OpenHarmony)开发入门教学视频》

《鸿蒙生态应用开发V2.0白皮书》

图片

《鸿蒙 (OpenHarmony)开发基础到实战手册》

OpenHarmony北向、南向开发环境搭建

图片

 《鸿蒙开发基础》

  • ArkTS语言
  • 安装DevEco Studio
  • 运用你的第一个ArkTS应用
  • ArkUI声明式UI开发
  • .……

图片

 《鸿蒙开发进阶》

  • Stage模型入门
  • 网络管理
  • 数据管理
  • 电话服务
  • 分布式应用开发
  • 通知与窗口管理
  • 多媒体技术
  • 安全技能
  • 任务管理
  • WebGL
  • 国际化开发
  • 应用测试
  • DFX面向未来设计
  • 鸿蒙系统移植和裁剪定制
  • ……

图片

《鸿蒙进阶实战》

  • ArkTS实践
  • UIAbility应用
  • 网络案例
  • ……

图片

 获取以上完整鸿蒙HarmonyOS学习资料,请点击→纯血版全套鸿蒙HarmonyOS学习资料

总结

总的来说,华为鸿蒙不再兼容安卓,对中年程序员来说是一个挑战,也是一个机会。只有积极应对变化,不断学习和提升自己,他们才能在这个变革的时代中立于不败之地。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/803032.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL数据库 数据库基本操作(四):表的增删查改(下)

1. 联合查询 注:联合查询是面试中的重点,只要考到sql,大多数情况下都考的是联合查询,而且联合查询也是我们学习中的难点. 1.1 笛卡尔积 在实际开发中,数据往往来自不同的表,所以要多表联合查询.多表查询是对多张表的数据笛卡尔积. 它们是两张表的各行数据通过全排列得到的. …

基于SSM+Jsp+Mysql的高校毕业设计管理系统

开发语言:Java框架:ssm技术:JSPJDK版本:JDK1.8服务器:tomcat7数据库:mysql 5.7(一定要5.7版本)数据库工具:Navicat11开发软件:eclipse/myeclipse/ideaMaven包…

秋招复习笔记——八股文部分:操作系统

笔试得刷算法题,那面试就离不开八股文,所以特地对着小林coding的图解八股文系列记一下笔记。 这一篇笔记是图解系统的内容。 硬件结构 CPU执行程序 计算机基本结构为 5 个部分,分别是运算器、控制器、存储器、输入设备、输出设备&#xf…

轻量的 WebHook 工具:歪脖虎克

本篇文章聊聊轻量的网络钩子(WebHook)工具:歪脖虎克。 写在前面 这是一篇迟到很久的文章,在 21 年和 22 年的时候,我分享过两篇关于轻量的计划任务工具 Cronicle 的文章:《轻量的定时任务工具 Cronicle&a…

运筹学基础(六)列生成算法(Column generation)

文章目录 前言从Cutting stock problem说起常规建模Column generation reformulation 列生成法核心思想相关概念Master Problem (MP)Linear Master Problem (LMP)Restricted Linear Master Problem (RLMP)subproblem(核能预警,非常重要) 算法…

kvm基础命令

前言 一、基础命令 1.虚拟机查看 2.虚拟机开启与关闭 3.虚拟机删除 4.查看虚拟机的配置 5.配置文件重定向 6.命令行登录虚拟机 二、调整虚拟机磁盘大小 三、虚拟机创建快照 四、virsh console报错 总结 前言 今天我们分享一下如何使用kvm基础命令。 一、基础命令 1.虚拟机查看…

杨笛一新作:社恐有救了,AI大模型一对一陪聊,帮i人变成e人

ChatGPT狂飙160天,世界已经不是之前的样子。 新建了免费的人工智能中文站https://ai.weoknow.com 新建了收费的人工智能中文站ai人工智能工具 更多资源欢迎关注 在社交活动中,大语言模型既可以是你的合作伙伴(partner)&#xff0…

MySQL-用户与权限管理:用户管理、权限管理、角色管理

用户与权限管理 用户与权限管理1.用户管理1.1 登录MySQL服务器1.2 创建用户1.3 修改用户1.4 删除用户1.5 设置当前用户密码1.6 修改其它用户密码 2. 权限管理2.1 权限列表2.2 授予权限的原则2.3 授予权限2.4 查看权限2.5 收回权限 访问控制连接核实阶段请求核实阶段 3. 角色管理…

Redis 八种常用数据类型常用命令和应用场景

5 种基础数据类型:String(字符串)、List(列表)、Set(集合)、Hash(散列)、Zset(有序集合)。 3 种特殊数据类型:HyperLogLog&#xff0…

【git】checkout origin/xxx 出现 detached HEAD问题

git 检出远程分支出现Head分离的是什么原因导致的呢?? 因为Head指向了origin的一个commit, 但是这个origin分支你的本地又没有,也就是说你本地没有追踪这个分支,那就要track一下 git checkout -h 看一下有没有追踪的命令 果不其…

服务器开发 Socket 相关函数

Socket 函数 #include <sys/types.h> #include <sys/socket.h> int socket(int domain, int type, int protocol)domain: AF_INET 这是大多数用来产生 socket 的协议&#xff0c;使用TCP或UDP来传输&#xff0c;用IPv4的地址 AF_INET6 与上面类似&#xff0c;不过…

Python高级

不定长参数 位置不定长参数&#xff0c;获取参数args会整合为一个元组 def info(*args):print(arg is, args)print(type(arg) is, type(args))info(1, 2, 3, 4, a, b)# 输出 # arg is (1, 2, 3, 4, a, b) # type(arg) is <class tuple> 关键字不定长参数&#xff0c;&…

蓝桥杯2023年第十四届省赛真题-棋盘

solution1(暴力) 暴力蓝桥杯可以过&#xff0c;虽然理论上会超时~ #include<iostream> using namespace std; const int maxn 2010; int a[maxn][maxn] {0};//0白棋&#xff0c;1黑棋 int main(){int n, m, x1, x2, y1, y2;scanf("%d%d", &n, &m)…

可视化大屏 附源码(Vue3 + TS + DataV + ECharts)

目录 前言 ✨项目代码 1、带有node_modules的项目包 &#x1f680; 2、不带有node_modules的项目包 &#x1f680; ⚒️项目屏幕大小调整 &#x1f48e; 使用开源项目 1、DataV &#x1f530; 2、Echarts &#x1f530; 3、PPchart &#x1f530; 4、表格平滑滚动 &a…

C# 如何修改项目名称

目录 背景具体步骤1、Visual Studio中修改项目名和程序集名称以及命名空间2、修改项目文件夹名3、修改解决方案里项目的路径4、再次打开解决方案&#xff0c;问题解决步骤总结 名词解释解决方案&#xff08;Solution&#xff09;项目&#xff08;Project&#xff09;程序集&…

sa-token非Web上下文无法获取Request

0x02 非Web上下文无法获取Request 问题定位 在我们使用sa-token安全框架的时候&#xff0c;有时候会提示&#xff1a;SaTokenException:非Web上下文无法获取Request 错误截图&#xff1a; 在官方网站中&#xff0c;查看常见问题排查&#xff1a; 非Web上下文无法获取Reques…

运行游戏找不到steam_api64.dll怎么办?steam_api64.dll丢失解决方法

steam_api64.dll是64位Windows操作系统上的一个动态链接库&#xff08;DLL&#xff09;文件&#xff0c;其大小通常在1.5-3.5 MB之间。这个文件对于Steam平台至关重要&#xff0c;因为它实现了游戏验证、更新等功能&#xff0c;并确保了用户拥有游戏的合法使用权。它通过提供一…

基于单片机水塔水位检测控制系统设计

**单片机设计介绍&#xff0c; 基于单片机水塔水位检测控制系统设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机水塔水位检测控制系统设计的主要目标是实现水塔水位的自动监测与控制&#xff0c;确保水塔内的水位始…

网页版五子棋对战实现和自动化测试

文章目录 前言一、项目描述项目演示链接 二、实现的功能与操作1.登录注册2.游戏大厅线程安全问题多开处理 3.五子棋对战 三、项目测试1.测试用例2.测试技术点3.部分测试用例展示&#xff08;1&#xff09;注册页面&#xff08;2&#xff09;登录页面&#xff08;3&#xff09;游…

生成式AI:最有商业前景的人工智能技术,不再是改变分发关系,而是升级生产力

根据最新的报告可以看出ChatGPT到底有多厉害&#xff0c;多个方面实现从判别决策到创造生成 生成式AI VS Web 3.0 &#xff1a;不追求生产关系的重塑&#xff0c;但将大幅度提升和创造生产力 创造是生成式AI的核心&#xff0c;本质是对生产力的大幅度提升和创造。生成式AI通过…