9.多级缓存、JVM进程缓存、Lua语法

news2025/1/9 15:07:28

多级缓存

文章目录

  • 多级缓存
  • 一、多级缓存介绍
    • 1.1 传统缓存的问题
    • 1.2 多级缓存方案
  • 二、JVM进程缓存
    • 2.1 案例准备
      • 2.1.1 导入SQL
      • 2.1.2 导入item-service项目
      • 2.1.3 导入商品查询页面
    • 2.2 初始 Caffeine
      • 2.2.1 基本用法
    • 2.3 实现进程缓存
  • 三、Lua语法
    • 3.1 初识Lua
    • 3.2 变量和循环
      • 3.2.1 数据类型
      • 3.2.2 变量
      • 3.2.3 循环
    • 3.3 函数、条件控制
      • 3.3.1 函数
      • 3.3.2 条件控制
    • 3.4 案例:自定义函数,打印table

一、多级缓存介绍

缓存的作用:减轻对数据库的压力,缩短服务响应的时间,从而提高服务的并发能力

但是如果是很高的并发量的话,仅仅靠Redis并不能满足高并发需求,而多级缓存能够应对亿级流量

1.1 传统缓存的问题

传统的缓存策略一般是请求到大Tomcat后,先查询Redis,如果未命中则查询数据库,存在下面的问题

  • 请求要经过Tomcat处理,Tomcat的性能成为整个系统的瓶颈

用户请求依然进入Tomcat,而后由Tomcat去查询Redis缓存,并且Tomcat本身的并发能力并不如Redis,在这个场景中,Tomcat成为业务的瓶颈

  • Redis缓存失效时,会对数据库产生冲击

image-20230715221830760

1.2 多级缓存方案

多级缓存就是充分利用请求处理的每个环节,分别添加缓存,减轻Tomcat压力,提升服务性能:

  • ① 用户可以通过浏览器或者手机去访问我们的服务器得到数据并且渲染,在这里我们就可以形成一级缓存了,称为浏览器客户端缓存

浏览器可以把服务器返回的静态资源保存在本地的,下一次再去访问静态资源的时候,我们的服务器只需要检查一下数据有没有变化,没有变化便返回304状态码,浏览器接收到304状态码就直接把本地的数据渲染,用户就能直接看到了,这样就可以减少很多的数据传输,提高渲染和响应的速度

  • ② 但是对于非静态数据,我们就不得不访问服务器端了,比如说请求到达了Nginx服务器

以前的Nginx是用来做反向代理的,但是这里要形成第二级缓存,称为Nginx本地缓存

我们可以把数据缓存在Nginx本地,用户请求来了之后先看二级缓存有没有,如果有直接返回

  • ③ 如果Nginx的本地缓存没有,直接查询Redis

此处查询Redis形成第三级缓存。如果Redis命中了,就返回,如果没有命中才会到大Tomcat

以前Redis是在Tomcat之后查,现在是Nginx之后查

  • ④Tomcat成为第四级缓存,叫做Tomcat进程缓存

会在服务器的内部,利用类似map这样的形式(但并不是map),形成一个进程缓存

请求到大Tomcat之后先读取本地缓存,如果进程缓存命中后就直接返回,并不用去访问数据库了

这样Redis失效后,请求并不会直接打到mysql数据库上

模型图

image-20230715223633094


现在大部分的压力集中在了Nginx内部,此时Redis不再是一个反向代理服务器,就变成了一个真正的外部服务,可以在里面写业务逻辑,那可以将Nginx形成一个集群,才能应对一个高并发

也可以用一个单独的Nginx,来做一个反向代理,请求先到达这个单独的Nginx,然后再由其反向代理到我们多个这样的本地缓存(编写业务的Nginx服务器),之后就是Redis、Tomcat、MYSQL

image-20230715223948710

二、JVM进程缓存

也就是下图标红的位置。也就是Tomcat服务内部去添加缓存,业务进来以后会优先查进程缓存,缓存未命中再查询数据库

image-20230715224153597

2.1 案例准备

2.1.1 导入SQL

/*
 Navicat Premium Data Transfer

 Source Server         : 192.168.150.101
 Source Server Type    : MySQL
 Source Server Version : 50725
 Source Host           : 192.168.150.101:3306
 Source Schema         : heima

 Target Server Type    : MySQL
 Target Server Version : 50725
 File Encoding         : 65001

 Date: 16/08/2021 14:45:07
*/

SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;

-- ----------------------------
-- Table structure for tb_item
-- ----------------------------
DROP TABLE IF EXISTS `tb_item`;
CREATE TABLE `tb_item`  (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '商品id',
  `title` varchar(264) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '商品标题',
  `name` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL DEFAULT '' COMMENT '商品名称',
  `price` bigint(20) NOT NULL COMMENT '价格(分)',
  `image` varchar(200) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '商品图片',
  `category` varchar(200) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '类目名称',
  `brand` varchar(100) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '品牌名称',
  `spec` varchar(200) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '规格',
  `status` int(1) NULL DEFAULT 1 COMMENT '商品状态 1-正常,2-下架,3-删除',
  `create_time` datetime NULL DEFAULT NULL COMMENT '创建时间',
  `update_time` datetime NULL DEFAULT NULL COMMENT '更新时间',
  PRIMARY KEY (`id`) USING BTREE,
  INDEX `status`(`status`) USING BTREE,
  INDEX `updated`(`update_time`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 50002 CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '商品表' ROW_FORMAT = COMPACT;

-- ----------------------------
-- Records of tb_item
-- ----------------------------
INSERT INTO `tb_item` VALUES (10001, 'RIMOWA 21寸托运箱拉杆箱 SALSA AIR系列果绿色 820.70.36.4', 'SALSA AIR', 16900, 'https://m.360buyimg.com/mobilecms/s720x720_jfs/t6934/364/1195375010/84676/e9f2c55f/597ece38N0ddcbc77.jpg!q70.jpg.webp', '拉杆箱', 'RIMOWA', '{\"颜色\": \"红色\", \"尺码\": \"26寸\"}', 1, '2019-05-01 00:00:00', '2019-05-01 00:00:00');
INSERT INTO `tb_item` VALUES (10002, '安佳脱脂牛奶 新西兰进口轻欣脱脂250ml*24整箱装*2', '脱脂牛奶', 68600, 'https://m.360buyimg.com/mobilecms/s720x720_jfs/t25552/261/1180671662/383855/33da8faa/5b8cf792Neda8550c.jpg!q70.jpg.webp', '牛奶', '安佳', '{\"数量\": 24}', 1, '2019-05-01 00:00:00', '2019-05-01 00:00:00');
INSERT INTO `tb_item` VALUES (10003, '唐狮新品牛仔裤女学生韩版宽松裤子 A款/中牛仔蓝(无绒款) 26', '韩版牛仔裤', 84600, 'https://m.360buyimg.com/mobilecms/s720x720_jfs/t26989/116/124520860/644643/173643ea/5b860864N6bfd95db.jpg!q70.jpg.webp', '牛仔裤', '唐狮', '{\"颜色\": \"蓝色\", \"尺码\": \"26\"}', 1, '2019-05-01 00:00:00', '2019-05-01 00:00:00');
INSERT INTO `tb_item` VALUES (10004, '森马(senma)休闲鞋女2019春季新款韩版系带板鞋学生百搭平底女鞋 黄色 36', '休闲板鞋', 10400, 'https://m.360buyimg.com/mobilecms/s720x720_jfs/t1/29976/8/2947/65074/5c22dad6Ef54f0505/0b5fe8c5d9bf6c47.jpg!q70.jpg.webp', '休闲鞋', '森马', '{\"颜色\": \"白色\", \"尺码\": \"36\"}', 1, '2019-05-01 00:00:00', '2019-05-01 00:00:00');
INSERT INTO `tb_item` VALUES (10005, '花王(Merries)拉拉裤 M58片 中号尿不湿(6-11kg)(日本原装进口)', '拉拉裤', 38900, 'https://m.360buyimg.com/mobilecms/s720x720_jfs/t24370/119/1282321183/267273/b4be9a80/5b595759N7d92f931.jpg!q70.jpg.webp', '拉拉裤', '花王', '{\"型号\": \"XL\"}', 1, '2019-05-01 00:00:00', '2019-05-01 00:00:00');

-- ----------------------------
-- Table structure for tb_item_stock
-- ----------------------------
DROP TABLE IF EXISTS `tb_item_stock`;
CREATE TABLE `tb_item_stock`  (
  `item_id` bigint(20) NOT NULL COMMENT '商品id,关联tb_item表',
  `stock` int(10) NOT NULL DEFAULT 9999 COMMENT '商品库存',
  `sold` int(10) NOT NULL DEFAULT 0 COMMENT '商品销量',
  PRIMARY KEY (`item_id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = COMPACT;

-- ----------------------------
-- Records of tb_item_stock
-- ----------------------------
INSERT INTO `tb_item_stock` VALUES (10001, 99996, 3219);
INSERT INTO `tb_item_stock` VALUES (10002, 99999, 54981);
INSERT INTO `tb_item_stock` VALUES (10003, 99999, 189);
INSERT INTO `tb_item_stock` VALUES (10004, 99999, 974);
INSERT INTO `tb_item_stock` VALUES (10005, 99999, 18649);

SET FOREIGN_KEY_CHECKS = 1;

其中包含两张表:

  • tb_item:商品表,包含商品的基本信息
  • tb_item_stock:商品库存表,包含商品的库存信息

之所以将库存分离出来,是因为库存是更新比较频繁的信息,写操作较多。而其他信息修改的频率非常低。

其实就是一张表,但是做了一下分离

因为有一些信息需要经常修改,有一些信息很少修改,

修改之后需要修改缓存,操作Redis,那我们可以把经常修改的放到一张表,减少整条数据缓存过期的次数

2.1.2 导入item-service项目

image-20230715224616348

其中的业务包括:

  • 分页查询商品
  • 新增商品
  • 修改商品
  • 修改库存
  • 删除商品
  • 根据id查询商品
  • 根据id查询库存

业务全部使用mybatis-plus来实现,如有需要请自行修改业务逻辑。

启动服务,访问:http://localhost:8081/item/10001即可查询数据

image-20230715232659409

2.1.3 导入商品查询页面

商品查询是购物页面,与商品管理的页面是分离的。

部署方式如图:

image-20230715232819323

我们需要准备一个反向代理的nginx服务器,如上图红框所示,将静态的商品页面放到nginx目录中。

页面需要的数据通过ajax向服务端(nginx业务集群)查询。

运行nginx服务

注意!Nginx一定不要放在有中文的目录下或者是有空格的目录录下

然后访问 http://localhost/item.html?id=10001即可(请求到了80端口,也就是Nginx服务器)

下面页面中的数据都是一些假数据,页面写死的

image-20230719221500985

页面首先向Nginx发送请求,反向代理服务器获得请求,但是拿到后不能处理,它要把请求代理到我们的后台Nginx集群中,由后台Nginx集群完成后续的多级缓存处理

所以我们需要在这个反向代理服务器里,完成这个反向代理的配置

完成这个反向代理的配置

在Nginx中的conf目录下的nginx.conf文件下

如下面所示

image-20230719222037331

Nginx的业务集群:做Nginx的本地缓存、redis缓存、Tomcat缓存/查询等等。

如果下面的两个服务找不到,就会报错(先写一个8081,不写8082)。

image-20230719222344767

比如说,以api开头的请求,会被我们的nginx配置拦截到了,拦截到后反向代理到proxy_pass http://nginx-cluster;

proxy_pass http://nginx-cluster; 这是我们nginx里的负载均衡的配置

image-20230719222139330

总结

我们Windows上启动的Nginx就是起着一个反向代理的作用。

如果我们在Nginx集群中有多个Nginx,我们就可以在nginx-cluster中配置多个

image-20230719222911417

2.2 初始 Caffeine

导入商品案例后,就可以给商品的查询添加JVM缓存了

缓存在日常开发中启动至关重要的作用,由于是存储在内存中,数据的读取速度是非常快的,能大量减少对数据库的访问,减少数据库的压力。我们把缓存分为两类:

  • 分布式缓存

    例如Redis
    优点:存储容量更大、可靠性更好、可以在集群间共享

    缺点:访问缓存有网络开销.

    场景:缓存数据量较大、可靠性要求较高、需要在集群间共享

    假如有n台Tomcat,且都需要访问这个缓存,那我们就可以使用Redis缓存,因为Redis缓存是独立于Tomcat之外的

  • 进程本地缓存

    例如HashMap、GuavaCache

    优点:读取本地内存,没有网络开销,速度更快

    缺点:存储容量有限、可靠性较低、无法共享

    场景:性能要求较高,缓存数据量较小0

    进程本地缓存其实是作为一个分布式缓存的一个补充,当Redis缓存未命中时,再去查看一下进程本地缓存


企业当中是两者结合使用

本地缓存我们采用Caffeine

Caffeine是一个基于Java8开发的,提供了近乎最佳命中率的高性能的本地缓存库。目前Spring内部的缓存使用的就是Caffeine

GitHub地址: https://github.com/ben-manes/caffeine

在wiki里面有使用说明

image-20230719224529254

2.2.1 基本用法

/*
  基本用法测试
 */
@Test
void testBasicOps() {
    // 创建缓存对象
    Cache<String, String> cache = Caffeine.newBuilder().build();

    // 存数据
    cache.put("gf", "迪丽热巴");

    // 取数据,不存在则返回null
    String gf = cache.getIfPresent("gf");
    System.out.println("gf = " + gf);

    // 取数据,不存在则去数据库查询
    // 后面的参数是一个Function,如果未命中就执行这个函数。
    // Function的作用就是根据一个key,去找缓存的值。然后将Function的返回值返回
    String defaultGF = cache.get("defaultGF", key -> {
        // 这里可以去数据库根据 key查询value
        return "柳岩";
    });
    System.out.println("defaultGF = " + defaultGF);
}

image-20230719225322222

那一直这么存的话,肯定会将缓存存满,这怎么办呢

驱逐策略。

如下三种:

  • 基于容量

    设置缓存的数量上限

//创建缓存对象
        Cache<String, String> cache = Caffeine.newBuilder()
                .maximumSize(1)//设置缓存大小上限为1个key
                .build();
/*
     基于大小设置驱逐策略:
     */
    @Test
    void testEvictByNum() throws InterruptedException {
        // 创建缓存对象
        Cache<String, String> cache = Caffeine.newBuilder()
                // 设置缓存大小上限为 1
                .maximumSize(1)
                .build();
        // 存数据
        cache.put("gf1", "柳岩");
        cache.put("gf2", "范冰冰");
        cache.put("gf3", "迪丽热巴");
        // 延迟10ms,给清理线程一点时间
//        Thread.sleep(10L);
        // 获取数据
        System.out.println("gf1: " + cache.getIfPresent("gf1"));
        System.out.println("gf2: " + cache.getIfPresent("gf2"));
        System.out.println("gf3: " + cache.getIfPresent("gf3"));
    }

如果我们的线程不休眠,三个key对应的值都会输出

image-20230719230659550

但是如果线程休眠后,只会输出gf3

image-20230719230636692

  • 基于时间

    设置缓存的有效时间

        Cache<String, String> cache = Caffeine.newBuilder()
                //设置缓存有效期为10秒,从最后一次写入开始计时
                .expireAfterWrite(Duration.ofSeconds(10))
                .build();
/*
 基于时间设置驱逐策略:
 */
@Test
void testEvictByTime() throws InterruptedException {
    // 创建缓存对象
    Cache<String, String> cache = Caffeine.newBuilder()
            .expireAfterWrite(Duration.ofSeconds(1)) // 设置缓存有效期为 10 秒
            .build();
    // 存数据
    cache.put("gf", "柳岩");
    // 获取数据
    System.out.println("gf: " + cache.getIfPresent("gf"));
    // 休眠一会儿
    Thread.sleep(1200L);
    System.out.println("gf: " + cache.getIfPresent("gf"));
}
  • 基于引用

    设置缓存为软引用或弱引用,利用GC来回收缓存数据。性能较差,不建议使用。

在默认情况下,当一个缓存元素过期的时候,Cafeine不会自动立即将其清理和驱逐。而是在一次读或写操作后,或者在空闲时间完成对失效数据的驱逐。

2.3 实现进程缓存

利用Caffeine实现下列需求

  • 给根据id查询商品的业务添加缓存,缓存未命中时查询数据库
//  根据id查询商品
    @GetMapping("/{id}")
    public Item findById(@PathVariable("id") Long id){
       return itemCache.get(id,key->{
           return itemService.query()
                   .ne("status", 3).eq("id", key)
                   .one();
        });
//        return itemService.query()
//                .ne("status", 3).eq("id", id)
//                .one();
    }
  • 给根据id查询商品库存的业务添加缓存,缓存未命中时查询数据库
//  根据id查询库存
    @GetMapping("/stock/{id}")
    public ItemStock findStockById(@PathVariable("id") Long id){
       return stockCache.get(id,key->{
            return  stockService.getById(id);
        });
//        return stockService.getById(id);
    }
  • 缓存初始大小为100

  • 缓存上限为10000

商品库和库存库并不是一个库,不要混淆

config配置类

@Configuration
public class CaffeineConfig {
    //  import com.github.benmanes.caffeine.cache.Cache;
    @Bean
    public Cache<Long, Item> itemCache() {
        return Caffeine.newBuilder()
//              初始化容量为100
                .initialCapacity(100)
//              最大容量10000
                .maximumSize(10000)
                .build();
    }

    @Bean
    public Cache<Long, ItemStock> stockCache() {
        return Caffeine.newBuilder()
//              初始化容量为100
                .initialCapacity(100)
//              最大容量10000
                .maximumSize(10000)
                .build();
    }
}

验证商品是否成功

localhost:8081/item/10001

image-20230719232538714

也成功的查询到了数据库,然后我们刷新,控制台没有任何反应

image-20230719232555564

同理验证库存是否成功

localhost:8081/item/stock/10001

三、Lua语法

在Nginx写业务使用Lua语法

image-20230720213030999

下面只是lua中常见的语法,并不是完整的

3.1 初识Lua

Lua是一种轻量小巧的脚本语言,用标准C语言编写并以源代码形式开放,其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。

官网: https://www.lua.org/

  • 在Linux虚拟机的任意目录下,新建一个hello.lua文件
touch hello.lua
  • 添加下面的内容

在文件中添加下面的内容

print("Hello World!")
  • 运行
lua hello.lua

3.2 变量和循环

3.2.1 数据类型

数据类型描述
nil只有值nil属于该类,表示一个无效值(在条件表达式中相当于false)
boolean包含两个值:false和true
number表示双精度类型的实浮点数
string字符串由一对双引号或单引号来表示
function由C或Lua编写的函数
tableLua 中的表(table)其实是一个"关联数组"(associative arrays),数组的索引可以是数字、字符串或表类型。在 Lua里,table 的创建是通过”构造表达式"来完成,最简单构造表达式是{},用来创建一个空表。

可以把这里的table理解为java中的map(更接近一点而已)

可以利用type函数测试给定变量或值的类型:

print(type("Hello world"))

print(type(10.4*3))

3.2.2 变量

Lua声明变量的时候,并不需要指定数据类型:

-- 声明字符串
local str = 'hello!'
-- 声明数字
local num = 21
-- 声明布尔类型
local flag = true
-- 声明数组(数组下标从1开始) key为索引的 table
local arr = ['java', 'python', 'lua']
-- 声明table,类似java的map
local map ={name='Jack',age=21}

访问table

-- 访问数组,lua数组的角标从1开始
print(arr[1])
-- 访问table
print(map['name'])
print(map.name)

3.2.3 循环

如果我们的变量是一个数组,我们想循环取出数据

  • 遍历数组

ipairs表示我要解析这个数组,然后解析出来的内容就是一个键值对,key与value的形式

-- 声明数组 key为索引的 table
local arr = ['java', 'lua', 'python']
-- 遍历数组
for index,value in ipairs(arr) do
    print(index,value)
end
  • 遍历table
-- 声明map,也就是table
local map = {name='Jack', age=21]
-- 遍历table
for key,value in pairs(map) do
  print(key,value)
end

image-20230720220615914

3.3 函数、条件控制

3.3.1 函数

定义函数的语法

function 函数名( argument1,argument2..., argumentn)
    -- 函数体
    return 返回值
end

例如打印数组

function printArr(arr)
    for index, value in ipairs(arr) do
        print(value)
    end
end

3.3.2 条件控制

类似java的条件控制,例如if、else语法

if(布尔表达式)
then
    --[ 布尔表达式为 true 时执行该语句块 --]
else
    --[ 布尔表达式为 false 时执行该语句块 --]
end

3.4 案例:自定义函数,打印table

需求:自定义一个函数,可以打印table,当参数为nil时,打印错误信息

local arr = {'java','lua','python']
    
local map = {name='jack', age= 21}
    
local function printArr(arr)
   if (not arr) then
        print('数组不能为空!')
        return nil
    end
    for i, val in ipairs(arr) do
         print(val)
     end
 end
    
printArr(arr)

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

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

相关文章

这篇文章是用AI大模型自动生成的

昨天用豆包AI大模型尝试生成了一段关于&#xff1a;企业转型、企业转型的要点、企业转型成功的标志&#xff0c;这样的文字。我又加了点自己的思考。 &#xff08;1&#xff09;企业转型的原因 企业转型的原因有很多&#xff0c;以下是一些常见的原因&#xff1a; 1. 市场变化&…

针对电子企业的WMS仓储管理系统解决方案

随着全球电子行业的快速发展&#xff0c;电子企业对于仓储管理的需求和挑战也日益增长。WMS仓储管理系统作为电子企业的核心管理工具&#xff0c;需要满足高效率、低成本、高灵活性以及精确控制库存等需求。本文将为电子企业提供一种针对WMS仓储管理系统的解决方案。 一、WMS仓…

Eclipse使用SFTP方式远程连接

安装插件 首先需要安装远程系统连接插件&#xff0c;安装方式可参考&#xff1a; Eclipse安装FTP连接工具_哭哭啼的博客-CSDN博客在过滤器字段中,键入"remote".选择Mobile and Device Development&#xff0c;并选择。找到Remote System Explorer->Connection。…

Outlook邮箱如何设置自动回复

很多小伙伴在刚开始使用Outlook邮箱的时候&#xff0c;不是很清楚Outlook邮箱如何设置自动回复&#xff0c;这里小编就给大家详细介绍一下Outlook邮箱设置自动回复的方法&#xff0c;有需要的小伙伴可以来看一看。 Outlook邮箱设置自动回复的方法&#xff1a; 1、打开软件&am…

ARM-M0内核MCU,内置24bit ADC,采样率4KSPS,传感器、电子秤、体脂秤专用,国产

ARM-M0内核MCU 内置24bit ADC &#xff0c;采样率4KSPS flash 64KB&#xff0c;SRAM 32KB 适用于传感器&#xff0c;电子秤&#xff0c;体脂秤等等

QML android 采集手机传感器数据 并通过udp 发送

利用 qt 开发 安卓 app &#xff0c;采集手机传感器数据 并通过udp 发送 #ifndef UDPLINK_H #define UDPLINK_H#include <QObject> #include <QUdpSocket> #include <QHostAddress>class UdpLink : public QObject {Q_OBJECT public:explicit UdpLink(QObjec…

Win10无法访问你可能没有权限使用网络资源怎么解决

当我们使用Win10电脑打开文件时&#xff0c;弹出提示无法访问你可能没有权限使用网络资源&#xff0c;这是怎么回事&#xff0c;遇到这种问题应该怎么解决呢&#xff0c;下面小编就给大家详细介绍一下Win10无法访问你可能没有权限使用网络资源的解决方法&#xff0c;有需要的小…

使用亚马逊云科技人工智能内容审核服务,打造安全的图像生成和扩散模型

生成式人工智能技术发展日新月异&#xff0c;现在已经能够根据文本输入生成文本和图像。Stable Diffusion 是一种文本转图像模型&#xff0c;可让您创建栩栩如生的图像应用。您可以通过 Amazon SageMaker JumpStart&#xff0c;使用 Stable Diffusion 模型轻松地从文本生成图像…

继续聊聊API接口

什么是API接口 API接口(Application Programming Interface Interface)是应用程序与开发人员或其他程序互相通信的方式。它允许开发者访问应用程序的数据和功能。 API接口,软件的“握手”与“交流”之道,软件世界的“好基友”。想让软件聊得来?想开发App却无从下手?API来相救…

一文讲清楚:SaaS系统是什么?优势在哪?盘点国内行业龙头SaaS系统!

SaaS系统究竟是什么&#xff1f;应该如何了解SaaS系统&#xff1f;在SaaS系统飞速发展的2023年&#xff0c;国内涌现出了一大批优秀的SaaS系统公司&#xff0c;都有哪些企业位列其中呢&#xff1f;SaaS系统有着什么样独特的竞争力&#xff0c;能够不断发展&#xff0c;成为目前…

无涯教程-JavaScript - NPER函数

描述 NPER函数基于定期,固定付款和固定利率返回投资的期数。 语法 NPER (rate,pmt,pv,[fv],[type])争论 Argument描述Required/OptionalRateThe interest rate per period.RequiredPmt 在每个期间付款。 在年金的使用期限内,它不能改变。 通常,pmt包含本金和利息,但不包含其…

快递物流博览会开幕,多家快递企业黑科技齐聚亮相

快递物流供应链|分拣系统|AGV机器人|新能源物流车|绿色包装|自动识别|冷链物流 2024年4月12-14日 | 杭州国际博览中心 同期展会&#xff1a;2024中国数字物流技术与应用展 2024国际电商物流包装产业展 2024新能源商用车、物流车展 指导单位&#xff1a;浙江省邮政管理局 中…

c++ 学习之类型,常量以及变量的重点知识

const 和 volatile 组合考点 const int ( * ) 等价于 int const ( * ) const int x 1 ; 说明 x 是常量&#xff0c;无法修改 如何区分指针常量和常量指针 指针常量 为 先有指针后有常量 故为 形式如 &#xff1a; int * const p & x ; 且const 修饰的是 p &#xff0c…

Falcon 180B 目前最强大的开源模型

Technology Innovation Institute最近发布了Falcon 180B大型语言模型(LLM)&#xff0c;它击败了Llama-2 70b&#xff0c;与谷歌Bard的基础模型PaLM-2 Large不相上下。 180B是是Falcon 40B模型一个最新版本。以下是该模型的快速概述: 180B参数模型&#xff0c;两个版本(base和…

基于Python+Django实现一个电商购物网站系统

随着互联网的高速发展&#xff0c;电子商务行业也正迎来了其黄金时代。如何搭建一个功能完备、体验良好的电商网站成了许多开发者的关心话题。今天&#xff0c;我将带大家使用Python语言和Django框架&#xff0c;快速打造一个电商购物系统。如果你有一定的Python基础&#xff0…

java Math类中的random方法和Random类中方法的区别

文章目录 Math类中的random()方法Random类 Math类中的random()方法 Math类中的random()方法没有参数&#xff0c;它会默认返回等于0.0、小于1.0的double类型随机数。对double()方法返回的数字稍加处理&#xff0c;即可实现任意范围随机数的功能 public class MathTest {publi…

小白备战大厂算法笔试(六)——堆

文章目录 堆常用操作堆的实现存储与表示访问堆顶元素元素入堆元素出堆 常见应用建堆操作自上而下构建自下而上构建 TOP-K问题遍历选择排序堆 堆 堆是一种满足特定条件的完全二叉树&#xff0c;主要可分为下图所示的两种类型。 大顶堆&#xff1a;任意节点的值 ≥ 其子节点的值…

什么牌子的led台灯质量好?Led台灯品牌质量排行榜

台灯如何选择&#xff0c;随着人们生活水平的提高及科技的不断进步&#xff0c;台灯的品质也得到了极大的提高&#xff0c;在生活中很多时候都需要使用台灯&#xff0c;但是市面上的台灯那么多&#xff0c;台灯如何选择。推荐五款质量高的护眼台灯。 一、书客护眼台灯L1 书客…

Spring学习笔记——4

Spring学习笔记——4 一、基于AOP的声明式事务控制1.1、Spring事务编程概述1.2、搭建测试环境1.3、基于XML声明式事务控制1.4、基于注解声明式事务控制 二、Spring整合web环境2.1、JavaWeb三大组件作用及其特点2.2、Spring整合web环境的思路及实现2.3、Spring的Web开发组件spri…

基础术语和模式的学习记录

1 范围 本文件界定了政府和社会资本合作(PPP) 的基础术语&#xff0c;给出了政府和社会资本合作(PPP) 的 模 式 分类和代码。 本文件适用于政府和社会资本合作(PPP) 的所有活动。 2 规范性引用文件 本文件没有规范性引用文件。 3 基础术语 3.1 PPP 主体 3.1.1 政府和社…