[MAUI 项目实战] 笔记App(二):数据库设计

文章目录

    • Sqlite配置
    • 创建实体
      • 笔记实体类
      • 笔记分组实体
      • 笔记片段实体
      • 笔记片段负载实体
      • 笔记片段仓库实体
      • 笔记模板(场景)实体
      • 笔记片段模板实体
      • 笔记片段模板负载实体
    • 配置EF
    • 创建映射
    • 迁移和种子数据
    • 项目地址

Sqlite配置

应用程序里使用Sqlite作为数据库,使用EntityFramworkCore作为ORM,使用CodeFirst方式用EFCore初始化Sqlite数据库文件:mato.db

在MatoProductivity.Core项目的appsettings.json中添加本地sqlite连接字符串

  "ConnectionStrings": {"Default": "Data Source=file:{0};"},...

这里文件是一个占位符,通过代码hardcode到配置文件

在MatoProductivityCoreModule.cs中,重写PreInitialize并设置Configuration.DefaultNameOrConnectionString:

public override void PreInitialize()
{LocalizationConfigurer.Configure(Configuration.Localization);Configuration.Settings.Providers.Add<CommonSettingProvider>();string documentsPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), MatoProductivityConsts.LocalizationSourceName);var configuration = AppConfigurations.Get(documentsPath, development);var connectionString = configuration.GetConnectionString(MatoProductivityConsts.ConnectionStringName);var dbName = "mato.db";string dbPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), MatoProductivityConsts.LocalizationSourceName, dbName);Configuration.DefaultNameOrConnectionString = String.Format(connectionString, dbPath);base.PreInitialize();
}

创建实体

接下来定义实体类

笔记实体类

笔记用于存储实体,在笔记列表中,每个笔记都有标题和内容,创建时间等内容。

定义于\MatoProductivity.Core\Models\Entities\Note.cs


public class Note : FullAuditedEntity<long>
{public Note(){}public Note(string name, bool isHidden, bool isRemovable){Title = name;IsHidden = isHidden;IsRemovable = isRemovable;}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }public ICollection<NoteSegment> NoteSegments { get; set; }public string Title { get; set; }public string Type { get; set; }public string Status { get; set; }public string Desc { get; set; }public string Icon { get; set; }public string Color { get; set; }public string BackgroundColor { get; set; }public string BackgroundImage { get; set; }public string PreViewContent { get; set; }public bool IsEditable { get; set; }public bool IsHidden { get; set; }public bool IsRemovable { get; set; }public bool CanSimplified { get; set; }}

笔记分组实体

定义于\MatoProductivity.Core\Models\Entities\NoteGroup.cs

public class NoteGroup : FullAuditedEntity<long>
{public NoteGroup(){}public NoteGroup(string name, bool isHidden, bool isRemovable){Title = name;IsHidden = isHidden;IsRemovable = isRemovable;}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }public string Title { get; set; }public bool IsHidden { get; set; }public bool IsRemovable { get; set; }public ICollection<Note> Notes { get; set; }
}

笔记片段实体

定义于\MatoProductivity.Core\Models\Entities\NoteSegment.cs

public class NoteSegment : FullAuditedEntity<long>, INoteSegment
{public NoteSegment(){}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }[ForeignKey(nameof(NoteId))]public Note Note { get; set; }public ICollection<NoteSegmentPayload> NoteSegmentPayloads { get; set; }public long NoteId { get; set; }public string Title { get; set; }public string Type { get; set; }public string Status { get; set; }public string Desc { get; set; }public string Icon { get; set; }public string Color { get; set; }public int Rank { get; set; }public bool IsHidden { get; set; }public bool IsRemovable { get; set; }public INoteSegmentPayload GetNoteSegmentPayload(string key){if (NoteSegmentPayloads != null){return NoteSegmentPayloads.FirstOrDefault(c => c.Key == key);}return default;}public void SetNoteSegmentPayload(INoteSegmentPayload noteSegmentPayload){if (NoteSegmentPayloads != null){var currentPayload = NoteSegmentPayloads.FirstOrDefault(c => c.Key == noteSegmentPayload.Key);if (currentPayload != null){NoteSegmentPayloads.Remove(currentPayload);}if (!this.IsTransient()){(noteSegmentPayload as NoteSegmentPayload).NoteSegmentId = this.Id;}NoteSegmentPayloads.Add((noteSegmentPayload as NoteSegmentPayload));}}public INoteSegmentPayload GetOrSetNoteSegmentPayload(string key, INoteSegmentPayload noteSegmentPayload){if (NoteSegmentPayloads != null){var currentPayload = NoteSegmentPayloads.FirstOrDefault(c => c.Key == key);if (currentPayload != null){return currentPayload;}if (noteSegmentPayload != null){if (!this.IsTransient()){(noteSegmentPayload as NoteSegmentPayload).NoteSegmentId = this.Id;}NoteSegmentPayloads.Add((noteSegmentPayload as NoteSegmentPayload));}return noteSegmentPayload;}return noteSegmentPayload;}}

笔记片段负载实体

笔记片段负载与笔记片段实体为一对多的关系,用于存储笔记片段的详细内容。

定义于\MatoProductivity.Core\Models\Entities\NoteSegmentPayload.cs

public class NoteSegmentPayload : FullAuditedEntity<long>, INoteSegmentPayload
{public NoteSegmentPayload(){}public NoteSegmentPayload(string key, object value, string valuetype = null){if (value is string){this.SetStringValue((value as string).ToString());}else if (value is byte[]){this.Value = value as byte[];}else if (value is DateTime){this.SetStringValue(((DateTime)value).ToString("yyyy-MM-dd HH:mm:ss"));}else{this.SetStringValue(value.ToString());}this.Key = key;this.ValueType = valuetype;}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }[ForeignKey(nameof(NoteSegmentId))]public NoteSegment NoteSegment { get; set; }public long NoteSegmentId { get; set; }public string Key { get; set; }public byte[] Value { get; set; }public string ValueType { get; set; }[NotMapped]public string StringValue => GetStringValue();public T GetConcreteValue<T>() where T : struct{var value = Encoding.UTF8.GetString(Value);T result = value.To<T>();return result;}public string GetStringValue(){var value = Encoding.UTF8.GetString(Value);return value;}public void SetStringValue(string value){this.Value = Encoding.UTF8.GetBytes(value);}
}

笔记片段仓库实体

用于在编辑笔记页面的添加片段菜单中,加载所有可用的片段

定义于\MatoProductivity.Core\Models\Entities\NoteSegmentStore.cs

public class NoteSegmentStore : Entity<long>
{[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }public string Title { get; set; }public string Type { get; set; }public string Category { get; set; }public string Status { get; set; }public string Desc { get; set; }public string Icon { get; set; }public string Color { get; set; }public bool IsHidden { get; set; }public bool IsRemovable { get; set; }}

笔记模板(场景)实体

定义于\MatoProductivity.Core\Models\Entities\NoteTemplate.cs

public class NoteTemplate : FullAuditedEntity<long>
{public NoteTemplate(){}public NoteTemplate(string name, bool isHidden, bool isRemovable){Title = name;IsHidden = isHidden;IsRemovable = isRemovable;}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }public ICollection<NoteSegmentTemplate> NoteSegmentTemplates { get; set; }public string Title { get; set; }public string Type { get; set; }public string Status { get; set; }public string Desc { get; set; }public string Icon { get; set; }public string Color { get; set; }public string BackgroundColor { get; set; }public string BackgroundImage { get; set; }public string PreViewContent { get; set; }public bool IsEditable { get; set; }public bool IsHidden { get; set; }public bool IsRemovable { get; set; }public bool CanSimplified { get; set; }}

笔记片段模板实体

定义于\MatoProductivity.Core\Models\Entities\NoteSegmentTemplate.cs

public class NoteSegmentTemplate : FullAuditedEntity<long>, INoteSegment
{public NoteSegmentTemplate(){}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }[ForeignKey(nameof(NoteTemplateId))]public NoteTemplate NoteTemplate { get; set; }public ICollection<NoteSegmentTemplatePayload> NoteSegmentTemplatePayloads { get; set; }public long NoteTemplateId { get; set; }public string Title { get; set; }public string Type { get; set; }public string Status { get; set; }public string Desc { get; set; }public string Icon { get; set; }public string Color { get; set; }public int Rank { get; set; }public bool IsHidden { get; set; }public bool IsRemovable { get; set; }public INoteSegmentPayload GetNoteSegmentPayload(string key){if (NoteSegmentTemplatePayloads != null){return NoteSegmentTemplatePayloads.FirstOrDefault(c => c.Key == key);}return default;}public void SetNoteSegmentPayload(INoteSegmentPayload noteSegmentPayload){if (NoteSegmentTemplatePayloads != null){var currentPayload = NoteSegmentTemplatePayloads.FirstOrDefault(c => c.Key == noteSegmentPayload.Key);if (currentPayload != null){NoteSegmentTemplatePayloads.Remove(currentPayload);}if (!this.IsTransient()){(noteSegmentPayload as NoteSegmentTemplatePayload).NoteSegmentTemplateId = this.Id;}NoteSegmentTemplatePayloads.Add((noteSegmentPayload as NoteSegmentTemplatePayload));}}public INoteSegmentPayload GetOrSetNoteSegmentPayload(string key, INoteSegmentPayload noteSegmentPayload){if (NoteSegmentTemplatePayloads != null){var currentPayload = NoteSegmentTemplatePayloads.FirstOrDefault(c => c.Key == key);if (currentPayload != null){return currentPayload;}if (noteSegmentPayload != null){if (!this.IsTransient()){(noteSegmentPayload as NoteSegmentTemplatePayload).NoteSegmentTemplateId = this.Id;}NoteSegmentTemplatePayloads.Add((noteSegmentPayload as NoteSegmentTemplatePayload));}return noteSegmentPayload;}return noteSegmentPayload;}}

笔记片段模板负载实体

定义于\MatoProductivity.Core\Models\Entities\NoteSegmentTemplatePayload.cs

public class NoteSegmentTemplatePayload : FullAuditedEntity<long>, INoteSegmentPayload
{public NoteSegmentTemplatePayload(){}public NoteSegmentTemplatePayload(string key, object value, string valuetype = null){if (value is string){this.SetStringValue((value as string).ToString());}else if (value is byte[]){this.Value = value as byte[];}else if (value is DateTime){this.SetStringValue(((DateTime)value).ToString("yyyy-MM-dd HH:mm:ss"));}else{this.SetStringValue(value.ToString());}this.Key = key;this.ValueType = valuetype;}[Key, DatabaseGenerated(DatabaseGeneratedOption.Identity)]public override long Id { get; set; }[ForeignKey(nameof(NoteSegmentTemplateId))]public NoteSegmentTemplate NoteSegmentTemplate { get; set; }public long NoteSegmentTemplateId { get; set; }public string Key { get; set; }public byte[] Value { get; set; }public string ValueType { get; set; }[NotMapped]public string StringValue => GetStringValue();public T GetConcreteValue<T>() where T : struct{var value = Encoding.UTF8.GetString(Value);T result = value.To<T>();return result;}public string GetStringValue(){var value = Encoding.UTF8.GetString(Value);return value;}public void SetStringValue(string value){this.Value = Encoding.UTF8.GetBytes(value);}
}

配置EF

数据库上下文对象MatoProductivityDbContext定义如下

    public class MatoProductivityDbContext : AbpDbContext{//Add DbSet properties for your entities...public DbSet<Note> Note { get; set; }public DbSet<NoteGroup> NoteGroup { get; set; }public DbSet<NoteSegment> NoteSegment { get; set; }public DbSet<NoteSegmentStore> NoteSegmentStore { get; set; }public DbSet<NoteSegmentPayload> NoteSegmentPayload { get; set; }public DbSet<NoteTemplate> NoteTemplate { get; set; }public DbSet<NoteSegmentTemplate> NoteSegmentTemplate { get; set; }public DbSet<NoteSegmentTemplatePayload> NoteSegmentTemplatePayload { get; set; }public DbSet<Theme> Theme { get; set; }public DbSet<Setting> Setting { get; set; }public MatoProductivityDbContext(DbContextOptions<MatoProductivityDbContext> options) : base(options){}}

MatoProductivity.EntityFrameworkCore是应用程序数据库的维护和管理项目,依赖于Abp.EntityFrameworkCore。
在MatoProductivity.EntityFrameworkCore项目中csproj文件中,引用下列包

<PackageReference Include="Abp.EntityFrameworkCore" Version="7.4.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="7.0.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="7.0.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite.Design" Version="1.1.6" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="7.0.0">

在该项目MatoProductivityEntityFrameworkCoreModule.cs 中,将注册上下文对象,并在程序初始化运行迁移,此时将在设备上生成mato.db文件

public override void PostInitialize()
{Helper.WithDbContextHelper.WithDbContext<MatoProductivityDbContext>(IocManager, RunMigrate);if (!SkipDbSeed){SeedHelper.SeedHostDb(IocManager);}
}public static void RunMigrate(MatoProductivityDbContext dbContext)
{dbContext.Database.Migrate();
}

创建映射

从场景到笔记,或者说从模板到实例,我们需要映射,例如从笔记片段菜单中选择一个片段添加,那么需要从笔记片段仓库实体(NoteSegmentStore)映射到笔记片段实体(NoteSegment)或者,在编辑场景中,映射到笔记片段模板实体(NoteSegmentTemplate)。

[AutoMapTo(typeof(NoteSegment), typeof(NoteSegmentTemplate))]public class NoteSegmentStore : Entity<long>
{...
}

使用时:

var note = ObjectMapper.Map<NoteSegment>(noteSegmentStore);

ABP框架默认使用AutoMapper进行映射,所以需要配置映射关系。

Configuration.Modules.AbpAutoMapper().Configurators.Add(config =>
{IgnoreAbpProperties(config.CreateMap<NoteTemplate, Note>().ForMember(c => c.NoteSegments,options => options.MapFrom(input => input.NoteSegmentTemplates)).ForMember(c => c.Id,options => options.Ignore()));IgnoreAbpProperties(config.CreateMap<Note, NoteTemplate>().ForMember(c => c.NoteSegmentTemplates,options => options.MapFrom(input => input.NoteSegments)).ForMember(c => c.Id,options => options.Ignore()));IgnoreAbpProperties(config.CreateMap<NoteSegmentTemplate, NoteSegment>().ForMember(c => c.Note,options => options.MapFrom(input => input.NoteTemplate)).ForMember(c => c.NoteSegmentPayloads,options => options.MapFrom(input => input.NoteSegmentTemplatePayloads)).ForMember(c => c.NoteId,options => options.Ignore()).ForMember(c => c.Id,options => options.Ignore()));IgnoreAbpProperties(config.CreateMap<NoteSegmentStore, NoteSegment>().ForMember(c => c.Id,options => options.Ignore()));IgnoreAbpProperties(config.CreateMap<NoteSegment, NoteSegmentTemplate>().ForMember(c => c.NoteTemplate,options => options.MapFrom(input => input.Note)).ForMember(c => c.NoteTemplateId,options => options.Ignore()).ForMember(c => c.NoteSegmentTemplatePayloads,options => options.MapFrom(input => input.NoteSegmentPayloads)).ForMember(c => c.Id,options => options.Ignore()));IgnoreAbpProperties(config.CreateMap<NoteSegmentTemplatePayload, NoteSegmentPayload>().ForMember(c => c.NoteSegment,options => options.MapFrom(input => input.NoteSegmentTemplate)).ForMember(c => c.NoteSegmentId,options => options.Ignore()).ForMember(c => c.Id,options => options.Ignore()));IgnoreAbpProperties(config.CreateMap<NoteSegmentPayload, NoteSegmentTemplatePayload>().ForMember(c => c.NoteSegmentTemplate,options => options.MapFrom(input => input.NoteSegment)).ForMember(c => c.NoteSegmentTemplateId,options => options.Ignore()));});

迁移和种子数据

MatoProductivity.EntityFrameworkCore.Seed.SeedHelper可在程序启动时,访问数据库,并初始化种子数据。

public override void PostInitialize()
{Helper.WithDbContextHelper.WithDbContext<MatoProductivityDbContext>(IocManager, RunMigrate);if (!SkipDbSeed){SeedHelper.SeedHostDb(IocManager);}
}

它通过SkipDbSeed来决定是否跳过执行种子数据初始化。我们需要在安装完成App后第一次运行才执行种子数据初始化。

MAUI中提供了VersionTracking.Default.IsFirstLaunchEver方式获取是否是第一次在此设备上启动应用,请查看官方文档

public override async void Initialize()
{IocManager.RegisterAssemblyByConvention(typeof(MatoProductivityModule).GetAssembly());if (VersionTracking.Default.IsFirstLaunchEver){MatoProductivityEntityFrameworkCoreModule.SkipDbSeed = false;}else{MatoProductivityEntityFrameworkCoreModule.SkipDbSeed = true;}
}

在InitialDbBuilder中我们定义了大多数的业务初始数据,具体的实现方式请查阅源码。

internal void Create()
{CreateSetting("Theme", "Light");CreateSetting("DetailPageMode", "PreviewPage");CreateNoteSegmentStore("时间戳", "时间/提醒", "DateTimeSegment", "记录一个瞬时时间", FaIcons.IconClockO, "#D8292B");CreateNoteSegmentStore("计时器", "时间/提醒", "TimerSegment", "创建一个计时器,当它结束时会通知您", FaIcons.IconBell, "#D8292B");CreateNoteSegmentStore("笔记", "文本", "TextSegment", "随时用文本记录您的想法", FaIcons.IconStickyNoteO, "#E1A08B");CreateNoteSegmentStore("Todo", "文本", "TodoSegment", "记录一个Todo项目", FaIcons.IconCheckSquareO, "#E1A08B");CreateNoteSegmentStore("数值", "文本", "KeyValueSegment", "记录数值,以便统计数据", FaIcons.IconLineChart, "#E1A08B");CreateNoteSegmentStore("手绘", "文件", "ScriptSegment", "创建一个手绘", FaIcons.IconPaintBrush, "#AD9CC2");CreateNoteSegmentStore("照片/视频", "文件", "MediaSegment", "拍照或摄像", FaIcons.IconCamera, "#AD9CC2");CreateNoteSegmentStore("文档", "文件", "DocumentSegment", "从您设备中选取一个文档", FaIcons.IconFile, "#AD9CC2");CreateNoteSegmentStore("录音", "文件", "VoiceSegment", "记录一段声音", FaIcons.IconMicrophone, "#AD9CC2");CreateNoteSegmentStore("地点", "其它", "LocationSegment", "获取当前地点,或者从地图上选取一个地点", FaIcons.IconMapMarker, "#6D987C");CreateNoteSegmentStore("天气", "其它", "WeatherSegment", "获取当前天气信息", FaIcons.IconCloud, "#6D987C");CreateNoteSegmentStore("联系人", "其它", "ContactSegment", "从您设备的通讯录中选择一个联系人", FaIcons.IconUser, "#6D987C");
}

项目地址

GitHub:MatoProductivity

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

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

相关文章

Visual Studio 2022美化

说明&#xff1a; VS版本&#xff1a;Visual Studio Community 2022 背景美化 【扩展】【管理扩展】搜索“ClaudiaIDE”&#xff0c;【下载】&#xff0c;安装完扩展要重启VS 在wallhaven下载壁纸图片作为文本编辑器区域背景图片 【工具】【选项】搜索ClaudiaIDE&#xff…

计算机视觉8 图像增广

图像增广&#xff08;image augmentation&#xff09;是通过对训练图像进行一系列随机改变&#xff0c;从而产生相似但又不同的训练样本的技术。 图像增广有以下两个主要作用&#xff1a; 扩大训练数据集的规模&#xff1b;随机改变训练样本可以降低模型对某些属性的依赖&#…

python---迭代器生成器

迭代器是一种支持next()操作的对象,它包含了一组元素,当执行next()操作时,返回其中一个元素。 当所有元素都被返回后&#xff0c;再执行next()报异常---StopIteration #coding:utf-8 """ 迭代器&#xff1a;对象包含__iter__方法就是一个迭代器 特点:惰性使用,…

linux 或 mac 系统的操作指令

sudo 是 "superuser do" 的缩写&#xff0c;它是一个Linux和Unix系统中常用的命令&#xff0c;用于以超级用户&#xff08;root&#xff09;权限执行命令。 lsof -i :8080 lsof 是 "list open files" 的缩写。在 Unix 和类 Unix 操作系统&#xff08;如…

如何在 Vue 和 JavaScript 中截取视频任意帧图片

大家好&#xff01;今天我们来聊聊如何在 Vue 和 JavaScript 中截取视频的任意一帧图片。这个功能在很多场景下都非常有用&#xff0c;比如视频编辑、视频预览等。本文将带你一步步实现这个功能&#xff0c;并且会提供详细的代码示例。 准备工作 首先&#xff0c;我们需要一个…

如何判断c盘d盘e盘哪个是固态硬盘

怎么区分电脑里面的C盘、D 盘、E盘是机械硬盘还是固态硬盘&#xff1f;在电脑里&#xff0c;硬盘是存储数据的核心部件&#xff0c;负责存储操作系统、软件以及用户数据。硬盘的性能也会影响电脑的流畅度。平时我们最常使用的硬盘分为机械硬盘和固态硬盘。在日常使用中&#xf…

Python酷库之旅-第三方库Pandas(032)

目录 一、用法精讲 91、pandas.Series.set_flags方法 91-1、语法 91-2、参数 91-3、功能 91-4、返回值 91-5、说明 91-6、用法 91-6-1、数据准备 91-6-2、代码示例 91-6-3、结果输出 92、pandas.Series.astype方法 92-1、语法 92-2、参数 92-3、功能 92-4、返回…

创建tls并替换至Elasticsearch8,实现通过IP实现https访问

O、背景 今天在配置Elasticsearch8相关内容&#xff0c;原本很简单就可以应用&#xff0c;安装ES时&#xff0c;可以选择是否使用https&#xff0c;如果就是测试使用的话&#xff0c;或内网使用&#xff0c;直接使用http协议即可&#xff0c;比较简单。但手头的项目比较特殊&a…

​1:1公有云能力整体输出,腾讯云“七剑”下云端

【全球云观察 &#xff5c; 科技热点关注】 曾几何时&#xff0c;云计算技术的兴起&#xff0c;为千行万业的数字化创新带来了诸多新机遇&#xff0c;同时也催生了新产业新业态新模式&#xff0c;激发出高质量发展的科技新动能。很显然&#xff0c;如今的云创新已成为高质量发…

vue3封装el-table及实现表头自定义筛选

带完善内容 提示&#xff1a;二合一&#xff0c;封装el-table以及给表头配置类自定义筛选和排序 文章目录 一、pandas是什么&#xff1f; 一、pandas是什么&#xff1f; el-table.vue <template><div class"page-view"><el-table v-loading"ta…

液氮罐搬运过程中的安全注意事项有哪些

在液氮罐搬运过程中&#xff0c;安全性是至关重要的考虑因素。液氮是一种极低温的液体&#xff0c;其温度可达零下196摄氏度&#xff0c;在接触到人体或物体时会迅速引发严重的冷冻伤害。因此&#xff0c;正确的搬运和使用液氮罐是保障操作安全的关键。 液氮是一种无色、无味的…

RK3568笔记四十:设备树

若该文为原创文章&#xff0c;转载请注明原文出处。 一、介绍 设备树 (Device Tree) 的作用就是描述一个硬件平台的硬件资源&#xff0c;一般描述那些不能动态探测到的设备&#xff0c;可以被动态探测到的设备是不需要描述。 设备树可以被 bootloader(uboot) 传递到内核&#x…

分布式服务框架zookeeper+消息队列kafaka

一、zookeeper概述 zookeeper是一个分布式服务框架&#xff0c;它主要是用来解决分布式应用中经常遇到的一些数据管理问题&#xff0c;如&#xff1a;命名服务&#xff0c;状态同步&#xff0c;配置中心&#xff0c;集群管理等。 在分布式环境下&#xff0c;经常需要对应用/服…

项目的一些操作

一、发送qq邮箱验证码以及倒计时 要发送验证码需要用到邮箱的授权码&#xff1a; qq邮箱获取方式&#xff0c;打开qq邮箱点设置找到如下界面&#xff1a; 然后获取授权码&#xff1b; 导入依赖 <dependency><groupId>com.sun.mail</groupId><artifactId&…

cmake configure_package_config_file指令详解

在 CMake 中&#xff0c;configure_package_config_file 命令用于生成包配置文件&#xff08;Package Configuration File&#xff09;&#xff0c;这些文件用于指定如何使用和链接某个库或工具。通常情况下&#xff0c;这些文件用于支持 CMake 的 find_package 命令来查找和加…

LeetCode 算法:单词搜索 c++

原题链接&#x1f517;&#xff1a;单词搜索 难度&#xff1a;中等⭐️⭐️ 题目 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 单词必须按照字母顺序&#xff0c;通…

Perl 语言的特点

Perl 语言入门学习可以涵盖多个方面&#xff0c;包括其特点、基本语法、高级特性以及学习资源和社区支持等。以下是一个详细的入门学习指南&#xff1a; 一、Perl 语言的特点 文本处理能力强&#xff1a;Perl 提供了丰富的字符串处理函数和正则表达式的支持&#xff0c;非常适…

[C++]运算符重载

一、运算符重载 在C中&#xff0c;运算符重载&#xff08;Operator Overloading&#xff09;是一种允许程序员为已有的运算符&#xff08;如、-、*、/等&#xff09;赋予特定于类的含义的技术。通过运算符重载&#xff0c;可以使类的使用更加直观和自然&#xff0c;提高代码的…

详解MLOps,从Jupyter开发到生产部署

大家好&#xff0c;Jupyter notebook 是机器学习的便捷工具&#xff0c;但在应用部署方面存在局限。为了提升其可扩展性和稳定性&#xff0c;需结合DevOps和MLOps技术。通过自动化的持续集成和持续交付流程&#xff0c;可将AI应用高效部署至HuggingFace平台。 本文将介绍MLOps…

kotlin compose 实现应用内多语言切换(不重新打开App)

1. 示例图 2.具体实现 如何实现上述示例,且不需要重新打开App ①自定义 MainApplication 实现 Application ,定义两个变量: class MainApplication : Application() { object GlobalDpData { var language: String = "" var defaultLanguage: Strin…