当前位置: 首页 > news >正文

14 地址映射

14 地址映射

  • 1、地址划分
  • 2、相关函数
    • 2.1 ioremap/iounmap
    • 2.2 mmap地址映射
  • 3、总结

1、地址划分

  • 明确:在linux系统中,不管是应用程序还是驱动程序,都不允许直接访问外设的物理地址,要想访问必须将物理地址映射到用户虚拟地址或者内核虚拟地址,一旦映射完毕,应用或者内核程序访问映射的虚拟地址,就是在访问实际的物理地址
  • 在linux系统中,4G虚拟地址空间的划分
    用户虚拟地址范围:0x00000000~0xBFFFFFFF
    内核虚拟地址范围:0xC0000000~0xFFFFFFFF
  • 如何将物理地址映射到内核虚拟地址?- 利用ioremap函数
  • 如何将物理地址映射到用户虚拟地址?- 利用mmap
  • 一个物理地址可以有多个虚拟地址,一个虚拟地址不能有多个物理地址

2、相关函数

2.1 ioremap/iounmap

void *ioremap(unsigned long phy_address, unsigned long len)
- 功能:将物理地址映射到内核虚拟地址
- 参数:- phy_address:传递要映射的起始的物理地址- len:传递要映射的物理地址空间的大小
- 返回值:返回映射的起始内核虚拟地址
void iounmap(void *vir_address)
- 功能:解除物理地址和内核虚拟地址的映射关系
- 参数:vir_address:传递映射好的起始内核虚拟地址
  • 案例:使用地址映射操作gpio
  寄存器     		  物理地址		 内核虚拟地址 
GPIOCALTFN0			0xC001C020		  gpiocaltfn0
GPIOCOUTENB			0xC001C004		  gpiocoutenb
GPIOCOUTENB			0xC001C000		  gpiocout 
//地址映射,两种方案:
方案1unsigned long *gpiocout, *gpiocoutenb, *gpiocaltfn0;gpiocout = ioremap(0xC001C000, 4);gpiocoutenb = ioremap(0xC001C004, 4);gpiocaltfn0 = ioremap(0xC001C020, 4);方案2:由于寄存器的物理地址空间都是连续的,所以连续映射:void *gpiobase;unsigned long *gpiocout, *gpiocoutenb, *gpiocaltfn0;gpiobase = ioremap(0xC001C000, 0x24);//地址换算gpiocout = (unsigned long *)(gpiobase + 0x00);gpiocoutenb = (unsigned long *)(gpiobase + 0x04);gpiocaltfn0 = (unsigned long *)(gpiobase + 0x20);//配置模式
*gpiocaltfn0 &= ~(3 << 24);
*gpiocaltfn0 |= (1 << 24);//使能
*gpiocoutenb |= (1 << 12);//输出
*gpiocout |= (1 << 12);
*gpiocout &= ~(1 << 12);

案例:实现一个软件能够编辑处理器任意一个寄存器
驱动代码

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/miscdevice.h>
#include <linux/io.h> //ioremap/iounmap
#include <linux/uaccess.h>
//描述寄存器操作信息的结构体
struct reg_info {unsigned long phy_address; //寄存器物理地址unsigned long data;		   //寄存器数据
};#define REG_WRITE 	0x100001 	//写寄存器命令
#define REG_READ	0x100002	//读寄存器命令
static long reg_ioctl(struct file *file,unsigned int cmd,unsigned long buf) {unsigned long *gpiobase;struct reg_info kreg;copy_from_user(&kreg,(struct reg_info *)buf,sizeof(kreg));//结果:kreg.phy_address=物理地址,kreg.data=?//将寄存器物理地址映射到内核虚拟地址gpiobase = ioremap(kreg.phy_address, 4);if(gpiobase == NULL) {printk("ioremap failed.\n");return -1;}switch(cmd) {case REG_WRITE:*gpiobase = kreg.data;break;case REG_READ:kreg.data = *gpiobase;copy_to_user((struct reg_info *)buf,&kreg, sizeof(kreg));break;}//解除地址映射iounmap(gpiobase);return 0;
}
static struct file_operations reg_fops = {.unlocked_ioctl = reg_ioctl
};
static struct miscdevice reg_misc = {.name = "reg",.minor = MISC_DYNAMIC_MINOR,.fops = &reg_fops
};
static int reg_init(void) {misc_register(&reg_misc);return 0;
};
static void reg_exit(void){misc_deregister(&reg_misc);
};module_init(reg_init);
module_exit(reg_exit);
MODULE_LICENSE("GPL");

应用程序

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
struct reg_info {unsigned long phy_address; //寄存器物理地址unsigned long data;		   //寄存器数据
};
#define REG_WRITE	0x100001	//写寄存器命令
#define REG_READ	0x100002	//读寄存器命令
int main(int argc, char *argv[]) {int fd;struct reg_info reg; //暂存寄存器的操作信息if((argc != 3) && (argc != 4)) {printf("Usage: %s w address data\n", argv[0]);printf("%s r address\n", argv[0]);return -1;}fd = open("/dev/reg", O_RDWR);if(fd < 0) {printf("open reg test device failed.\n");return -1;}if(!strcmp(argv[1], "w")) {reg.phy_address = strtoul(argv[2], NULL, 0);reg.data = strtoul(argv[3], NULL, 0);ioctl(fd, REG_WRITE, &reg);} else if(!strcmp(argv[1], "r")) {reg.phy_address = strtoul(argv[2], NULL, 0);ioctl(fd, REG_READ, &reg);printf("%#x:%#x\n",reg.phy_address, reg.data);}close(fd);return 0;
}

测试

./btn_test w 0xC001C000 0xC84 # 开灯
./btn_test w 0xC001C000 0x1C84 # 关灯
  • 字符串转整形标准C库函数:strtoul
unsigned long strtoul(char *pstr, char **ppstr, int n);
- 功能:字符串转整形,例如:"100"->100, "0100"->0100, "0x100"->0x100 
- 参数: - pstr: 传递转换的字符串首地址- ppstr:记录字符串中非整形数的子字符串首地址- n:指定转换进制
- 返回值:返回转换的整形数例如: 
char *pstr = "100";
int a = strtoul(pstr, NULL, 0); //结果:a=100
char *pstr = "0100";
int a = strtoul(pstr, NULL, 0); //结果:a=0100
char *pstr = "0x100";
int a = strtoul(pstr, NULL, 0); //结果:a=0x100
char *pstr = "100";
int a = strtoul(pstr, NULL, 16); //结果:a=0x64
char *pstr = "100abcd";
char *pstr1 = NULL; 
int a = strtoul(pstr, &pstr1, 0); //结果:a=100,pstr1 = "abcd"

2.2 mmap地址映射

mmap就是完成物理地址映射到用户虚拟地址用的

用户3G虚拟地址空间划分:
0x00000000-------------------------------------------------------0xBFFFFFFF代码段  数据段  BSS段  堆区   MMAP虚拟内存区      栈区---->  <------       <----

系统调用函数原型:

void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);
- 功能:将物理地址空间映射到用户虚拟内存空间上			  
- 参数:- addr:NULL,让linux内核帮你在用户虚拟内存区域找一块空间内存用来映射物理地址- length:让linux内核帮你找空闲用户虚拟内存的大小;切记:大小必须是页面大小(4KB)的整数倍- prot:描述内核帮你找的空闲用户虚拟内存的访问权限一般指定为:PROT_READ|PROT_WRITE- flags:其余属性,一般指定为:MAP_SHARED- fd:硬件外设- offset:偏移量,一般给0
- 返回值:linux内核将空闲的用户虚拟内存的首地址进行返回,这个起始用户虚拟地址同样也是4KB整数倍

参考代码:

void *addr;
int fd = open("a.txt", O_RDWR);
addr = mmap(NULL, 0x1000, PROT_READ|PROT_WRITE,MAP_SHARED, fd, 0);
说明:将文件a.txt映射到以addr起始的用户虚拟内存上,将来访问映射的用户虚拟内存就是访问文件
//向映射的用户虚拟内存拷贝字符串数据
本质是向文件a.txt写入数据
memcpy(addr, "hello,world", 12);
  • 了解mmap系统调用函数所做的工作:
    1. 应用程序调用mmap,首先跑到C库的mmap函数定义
    2. C库的mmap函数作两件事:
      1.保存mmap系统调用号到r7寄存器
      2.调用swi/svc指令触发软中断异常
    3. 一旦触发软中断异常,CPU核立马处理软中断异常
      CPU核硬件自动做四件事:…
      软件进一步处理软中断异常(在内核空间完成)
    4. 最后进程跑到内核空间继续运行,跑到软中断异常的入口地址运行,做三件事:
      1.保护现场
      2.调用软中断异常处理函数,而此函数又做两件事:
      1.从r7寄存器中取出mmap系统调用号
      2.以mmap系统调用号为下标在内核的系统调用表中找到mmap对应的内核函数sys_mmap,而内核的sys_mmap做三件事:
      1. 内核的sys_mmap首先在当前进程的3G虚拟地址空间中找一块空闲的用户虚拟内存,将来用于和物理地址做映射
      2. 一旦找到空闲的用户虚拟内存,并且用户mmap本身也给用户虚拟内存指定了一堆的属性(大小,权限等)所以内核用struct vm_area_struct数据结构定义初始化一个对象来描述空闲的用户虚拟内存的属性
      3. 最后内核的sys_mmap调用底层驱动的mmap接口,并且内核sys_mmap将第2步创建的对象的地址也传递给底层驱动的mmap接口
      3.底层驱动mmap执行完毕,然后恢复现场状态恢复和跳转返回,至此mmap调用结束

struct vm_area_struct {
unsigned long vm_start; //空间用户虚拟内存的起始地址 等于mmap的返回值addr
unsigned long vm_end; //结束地址=vm_start+大小
pgprot_t vm_page_prot; //等于mmap传递的PROT_READ|PROT_READ|PROT_WRITE
unsigned long vm_flags; //等于mmap传递的MAP_SHARED
unsigned long vm_pgoff; //等于mmap传递的0

};

3.对应的底层驱动的mmap接口
```c
struct file_operations {int (*mmap) (struct file *file, struct vm_area_struct *vma);
};
- 功能:永远只能唯一做一件事:将已知的物理地址和已知的用户虚拟地址做映射,由于用户虚拟地址在用户空间,所以将来访问操作都是在应用程序完成,而不是在内核驱动完成,访问映射的用户虚拟地址就是在访问物理地址
- file:跟fd亲戚关系
- vma:指向内核sys_mmap创建的一个对象,此对象来描述空闲的用户虚拟内存的各种属性,将来底层驱动mmap接口利用此指针可以获取到用户虚拟内存的属性:vma->vm_start //获取起始用户虚拟地址vma->vm_end vma->vm_flagsvma->vm_page_prot

问:底层驱动的mmap接口到底如何最终完成映射呢?
因为已知物理地址可以看手册获取到,已知的用户虚拟地址通过vma指针能够获取到如何将两者关联在一起呢?
答:只需调用以下函数完成关联映射:

int remap_pfn_range(struct vm_area_struct *vma,unsigned long addr,unsigned long pfn, unsigned long size,pgprot_t prot);
- 功能:完成最终的地址映射
- 参数:- vma:传递内核sys_mmap创建的对象地址,也就是传递驱动mmap接口的第二个参数- addr:传递空闲的用户虚拟内存的首地址,也就是传递vma->vm_start- pfn:传递起始的物理地址>>12 切记:此物理地址大小必须是4KB(0x1000)整数倍例如:0xC001C000>>12:合法0xC001C004>>12:不合法- size:传递映射的用户虚拟内存的大小,也就是传递:vma->vm_end - vma->vm_start - prot:传递用户虚拟内存的访问权限,也就是传递:vma->vm_page_prot

案例:利用mmap实现开关灯操作
驱动代码

#include <linux/init.h>
#include <linux/module.h>
#include <linux/miscdevice.h>
#include <linux/fs.h>
#include <linux/mm.h>static int btn_mmap(struct file *file,struct vm_area_struct *vma){remap_pfn_range(vma,// 指向用户虚拟内存属性vma->vm_start,// 起始用户虚拟地址0xC001C000 >> 12,// 起始物理地址 (4KB为单位)vma->vm_end - vma->vm_start,// 大小vma->vm_page_prot // 读写访问权限);return 0;
}static struct file_operations btn_fops = {.mmap = btn_mmap // 地址映射接口
};static struct miscdevice btn_device={.name = "mybtn",.minor = MISC_DYNAMIC_MINOR,.fops = &btn_fops
};
static int btn_init(void){misc_register(&btn_device);return 0;
}
static void btn_exit(void){misc_deregister(&btn_device);
}module_init(btn_init);
module_exit(btn_exit);
MODULE_LICENSE("GPL");

应用程序

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
int main(int argc,char *argv[]){int fd;void* gpiobase;unsigned long *gpiocout;unsigned long *gpiocoutend;unsigned long *gpiocaltfn0;fd = open("/dev/mybtn",O_RDWR);if(fd<0){printf("open mybtn failed\n");return -1;}//将内核地址映射到用户虚拟地址gpiobase = mmap(NULL,0x1000,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);// 地址换算gpiocout = (unsigned long *)(gpiobase + 0x00);gpiocoutend = (unsigned long *)(gpiobase + 0x04);gpiocaltfn0 = (unsigned long *)(gpiobase + 0x20);// 配置输出*gpiocaltfn0 &=~(3<<24);*gpiocaltfn0 |=(1<<24);// 使能*gpiocoutend |=(1<<12);// 输出为1*gpiocout |=(1<<12);while(1){*gpiocout |=(1<<12);sleep(5);*gpiocout &=~(1<<12);sleep(5);}
}

注意,如果出现了操作了寄存器,但是硬件不同步的现象,需要手动关闭cache功能

static int btn_mmap(struct file *file,struct vm_area_struct *vma){// 关闭cache功能vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);remap_pfn_range(vma,// 指向用户虚拟内存属性vma->vm_start,// 起始用户虚拟地址0xC001C000 >> 12,// 起始物理地址 (4KB为单位)vma->vm_end - vma->vm_start,// 大小vma->vm_page_prot // 读写访问权限);return 0;
}

3、总结

read,write,ioctl和mmap对比
什么时候用:read,write,ioctl,什么时候用: mmap

  • read,write,ioctl数据操作流程
    对设备读操作:read,ioctl
    数据流:硬件寄存器----->内核缓冲区------->用户缓冲区
    gpio_get_value copy_to_user
    对设备写操作:write,ioctl
    数据流:用户缓冲区----->内核缓冲区------->硬件寄存器
    copy_from_user gpio_set_value
    结论:read,write,ioctl数据操作势必要经过两次数据拷贝:
    用户-内核->硬件
    硬件->内核->用户

  • mmap数据操作流程:
    对设备读操作: 应用程序直接以指针形式读取寄存器
    data = *gpiocout;
    对设备写操作: 应用程序直接以指针的形式写入寄存器
    *gpiocout &= ~(1 << 12);
    结论:mmap数据操作只需一个数据拷贝:
    用户->硬件
    硬件->用户

  • 所以:

    1. 如果用户对硬件操作访问的数据量比较小,read,write,ioctl的两次数据拷贝对系统性能肯定有影响,但是这种影响几乎可以忽略不计,如果操作的数据量比较大,两次数据拷贝性能的影响是致命的,例如:摄像头,LCD显示屏,声卡等
    2. 如果访问操作的数据量比较大,用read,write,ioctl势必影响系统的性能,务必采用mmap,将两次数据拷贝变成一次提供系统的性能效率
    3. 由于mmap在使用的时候,分配的用户虚拟内存必须是4KB的整数倍,如果操作的数据量比较小,此时还用mmap即使提高了系统的性能(几乎体会不到),反而是浪费了宝贵的内存资源

相关文章:

14 地址映射

14 地址映射 1、地址划分2、相关函数2.1 ioremap/iounmap2.2 mmap地址映射 3、总结 1、地址划分 明确&#xff1a;在linux系统中,不管是应用程序还是驱动程序&#xff0c;都不允许直接访问外设的物理地址,要想访问必须将物理地址映射到用户虚拟地址或者内核虚拟地址&#xff0…...

Java Resilience4j-RateLimiter学习

一. 介绍 Resilience4j-RateLimiter 是 Resilience4j 中的一个限流模块&#xff0c;我们对 Resilience4j 的 CircuitBreaker、Retry 已经有了一定的了解&#xff0c;现在来学习 RateLimiter 限流器&#xff1b; 引入依赖&#xff1b; <dependency><groupId>io.g…...

Nginx--地址重写Rewrite

一、什么是Rewrite Rewrite对称URL Rewrite&#xff0c;即URL重写&#xff0c;就是把传入Web的请求重定向到其他URL的过程 URL Rewrite最常见的应用是URL伪静态化&#xff0c;是将动态页面显示为静态页面方式的一种技术。比如http://www.123.com/news/index.php?id123 使用U…...

webflux源码解析(1)-主流程

目录 1.关键实例的创建1.1 实例创建1.2 初始化 2.处理请求的关键流程2.1 从ReactorHttpHandlerAdapter开始2.1 DispatcherHandler的初始化2.2查找mapping handler2.3 处理请求(执行handler)2.4 返回结果处理 3.webflux的配置装配参考&#xff1a; WebFlux是Spring 5.0框架推出的…...

ipad作为扩展屏的最简单方式

将iPad用作扩展屏幕有几种简单而有效的方法。以下是几种常见的方式&#xff1a; 1. Sidecar&#xff08;苹果官方功能&#xff09; 适用设备&#xff1a;iPad和Mac&#xff08;macOS Catalina及以上版本&#xff09;。功能&#xff1a;Sidecar 是苹果官方的功能&#xff0c;可…...

【卡码网Python基础课 17.判断集合成员】

目录 题目描述与分析一、集合二、集合的常用方法三、代码编写 题目描述与分析 题目描述&#xff1a; 请你编写一个程序&#xff0c;判断给定的整数 n 是否存在于给定的集合中。 输入描述&#xff1a; 有多组测试数据&#xff0c;第一行有一个整数 k&#xff0c;代表有 k 组测…...

生物研究新范式!AI语言模型在生物研究中的应用

–https://doi.org/10.1038/s41592-024-02354-y 留意更多内容&#xff0c;欢迎关注微信公众号&#xff1a;组学之心 Language models for biological research: a primer 研究团队及研究单位 James Zou–Department of Biomedical Data Science, Stanford University, Stan…...

python语言day08 属性装饰器和property函数 异常关键字 约束

属性装饰器&#xff1a; 三个装饰器实现对私有化属性_creat_time的get&#xff0c;set&#xff0c;del方法&#xff1b; 三个装饰器下的方法名都一样&#xff0c;通过message.creat_time的不同操作实现调用get&#xff0c;set&#xff0c;del方法。 __inti__&#xff1a; 创建并…...

day01JS-数据类型-01

1. 浏览器内核 通常所谓的浏览器内核也就是浏览器所采用的渲染引擎&#xff0c;渲染引擎决定了浏览器如何显示网页的内容以及页面的格式信息。不同的浏览器内核对网页编写语法的解释也有不同&#xff0c;因此同一网页在不同的内核的浏览器里的渲染&#xff08;显示&#xff09;…...

MATLAB 手动实现一种高度覆盖值提取建筑物点云的方法(74)

专栏往期文章,包含本章 MATLAB 手动实现一种高度覆盖值提取建筑物点云的方法(74) 一、算法介绍二、算法实现1.代码2.效果总结一、算法介绍 手动实现一种基于高度覆盖值的建筑物点云提取方法,适用于高大的城市建筑物,比只利用高度提取建筑物的方法更加稳定和具有价值,主要…...

git的下载与安装(Windows)

Git是一个开源的分布式版本控制系统&#xff08;Distributed Version Control System&#xff0c;简称DVCS&#xff09;&#xff0c;它以其高效、灵活和强大的功能&#xff0c;在现代软件开发中扮演着至关重要的角色。 git官网&#xff1a;Git (git-scm.com) 1.进入git官网 2…...

腾讯云AI代码助手 —— 编程新体验,智能编码新纪元

阅读导航 引言一、开发环境介绍1. 支持的编程语言2. 支持的集成开发环境&#xff08;IDE&#xff09; 二、腾讯云AI代码助手使用实例1. 开发环境配置2. 代码补全功能使用&#x1f4bb;自动生成单句代码&#x1f4bb;自动生成整个代码块 3. 技术对话3. 规范/修复错误代码4. 智能…...

使用 ESP32 和 TFT 屏幕显示实时天气信息 —— 基于 OpenWeatherMap API

实时监测环境数据是一个非常常见的应用场景&#xff0c;例如气象站、智能家居等。这篇博客将带你使用 ESP32 微控制器和一个 TFT 屏幕&#xff0c;实时显示当前城市的天气信息。通过 OpenWeatherMap API&#xff0c;我们能够获取诸如温度、天气情况以及经纬度等详细的天气数据&…...

高阶数据结构——B树

1. 常见的搜索结构 以上结构适合用于数据量相对不是很大&#xff0c;能够一次性存放在内存中&#xff0c;进行数据查找的场景。如果数据量很大&#xff0c;比如有100G数据&#xff0c;无法一次放进内存中&#xff0c;那就只能放在磁盘上了&#xff0c;如果放在磁盘上&#xff0…...

Vue2中watch与Vue3中watch对比和踩坑

上一节说到了 computed计算属性对比 &#xff0c;虽然计算属性在大多数情况下更合适&#xff0c;但有时也需要一个自定义的侦听器。这就是为什么 Vue 通过 watch 选项提供了一个更通用的方法&#xff0c;来响应数据的变化。当需要在数据变化时执行异步或开销较大的操作时&#…...

在Java程序中执行Linux命令

在Java中执行Linux命令通常涉及到使用Java的运行时类 (java.lang.Runtime) 或者 ProcessBuilder 类来启动一个外部进程 1. 使用 Runtime.exec() Runtime.exec() 方法可以用来执行一个外部程序。它返回一个 Process 对象&#xff0c;可以通过这个对象与外部程序交互&#xff0…...

微信小程序在不同移动设备上的差异导致原因

在写小程序的时候用了rpx自适应单位&#xff0c;但是还是出现了在不同机型上布局不统一的问题&#xff0c;在此记录一下在首页做一个输入框&#xff0c;在测试的时候&#xff0c;这个输入框在不同的机型上到处跑&#xff0c;后来排查了很久都不知道为什么会这样 解决办法是后 …...

快速体验fastllm安装部署并支持AMD ROCm推理加速

序言 fastllm是纯c实现&#xff0c;无第三方依赖的高性能大模型推理库。 本文以国产海光DCU为例&#xff0c;在AMD ROCm平台下编译部署fastllm以实现LLMs模型推理加速。 测试平台&#xff1a;曙光超算互联网平台SCNet GPU/DCU&#xff1a;异构加速卡AI 显存64GB PCIE&#…...

报错:java: javacTask: 源发行版 8 需要目标发行版 1.8

程序报错&#xff1a; Executing pre-compile tasks... Loading Ant configuration... Running Ant tasks... Running before tasks Checking sources Copying resources... [gulimail-coupon] Copying resources... [gulimail-common] Parsing java… [gulimail-common] java…...

【数据结构篇】~单链表(附源码)

【数据结构篇】~链表 链表前言链表的实现1.头文件2.源文件 链表前言 链表是一种物理存储结构上非连续、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的指针链接次序实现的。 1、链式机构在逻辑上是连续的&#xff0c;在物理结构上不一定连续​ 2、结点一般是从…...

网络六边形受到攻击

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 抽象 现代智能交通系统 &#xff08;ITS&#xff09; 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 &#xff08;…...

前端倒计时误差!

提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

visual studio 2022更改主题为深色

visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中&#xff0c;选择 环境 -> 常规 &#xff0c;将其中的颜色主题改成深色 点击确定&#xff0c;更改完成...

cf2117E

原题链接&#xff1a;https://codeforces.com/contest/2117/problem/E 题目背景&#xff1a; 给定两个数组a,b&#xff0c;可以执行多次以下操作&#xff1a;选择 i (1 < i < n - 1)&#xff0c;并设置 或&#xff0c;也可以在执行上述操作前执行一次删除任意 和 。求…...

鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院查看报告小程序

一、开发环境准备 ​​工具安装​​&#xff1a; 下载安装DevEco Studio 4.0&#xff08;支持HarmonyOS 5&#xff09;配置HarmonyOS SDK 5.0确保Node.js版本≥14 ​​项目初始化​​&#xff1a; ohpm init harmony/hospital-report-app 二、核心功能模块实现 1. 报告列表…...

大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计

随着大语言模型&#xff08;LLM&#xff09;参数规模的增长&#xff0c;推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长&#xff0c;而KV缓存的内存消耗可能高达数十GB&#xff08;例如Llama2-7B处理100K token时需50GB内存&a…...

2023赣州旅游投资集团

单选题 1.“不登高山&#xff0c;不知天之高也&#xff1b;不临深溪&#xff0c;不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

使用Spring AI和MCP协议构建图片搜索服务

目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式&#xff08;本地调用&#xff09; SSE模式&#xff08;远程调用&#xff09; 4. 注册工具提…...

Python常用模块:time、os、shutil与flask初探

一、Flask初探 & PyCharm终端配置 目的: 快速搭建小型Web服务器以提供数据。 工具: 第三方Web框架 Flask (需 pip install flask 安装)。 安装 Flask: 建议: 使用 PyCharm 内置的 Terminal (模拟命令行) 进行安装,避免频繁切换。 PyCharm Terminal 配置建议: 打开 Py…...

跨平台商品数据接口的标准化与规范化发展路径:淘宝京东拼多多的最新实践

在电商行业蓬勃发展的当下&#xff0c;多平台运营已成为众多商家的必然选择。然而&#xff0c;不同电商平台在商品数据接口方面存在差异&#xff0c;导致商家在跨平台运营时面临诸多挑战&#xff0c;如数据对接困难、运营效率低下、用户体验不一致等。跨平台商品数据接口的标准…...