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

ZKP Understanding Nova (1): MinRoot Example

Understanding Nova

Kothapalli, Abhiram, Srinath Setty, and Ioanna Tzialla. “Nova: Recursive zero-knowledge arguments from folding schemes.” Annual International Cryptology Conference. Cham: Springer Nature Switzerland, 2022.

Nova: Paper Code

1. Understanding the MinRoot Example

type E1 = Bn256EngineKZG;
type E2 = GrumpkinEngine;
type EE1 = nova_snark::provider::mlkzg::EvaluationEngine<E1>;
type EE2 = nova_snark::provider::ipa_pc::EvaluationEngine<E2>;
type S1 = nova_snark::spartan::snark::RelaxedR1CSSNARK<E1, EE1>; // non-preprocessing SNARK
type S2 = nova_snark::spartan::snark::RelaxedR1CSSNARK<E2, EE2>; // non-preprocessing SNARK
  • 定义了一些type
struct MinRootIteration<G: Group> {x_i: G::Scalar,y_i: G::Scalar,x_i_plus_1: G::Scalar,y_i_plus_1: G::Scalar,
}

这段代码定义了一个名为 MinRootIteration 的结构体,它是用于表示最小根迭代的数据结构。这个结构体需要一个泛型参数 G,这个参数必须实现了 Group trait。
在这个结构体中,我们定义了四个字段,它们都是 G::Scalar 类型。这个类型是由 Group trait 定义的关联类型,通常用于表示一个群的标量值。
具体来说,这四个字段分别是:

  • x_i:表示当前迭代的 x 值。
  • y_i:表示当前迭代的 y 值。
  • x_i_plus_1:表示下一次迭代的 x 值。
  • y_i_plus_1:表示下一次迭代的 y 值。
    这个结构体可能被用于存储和管理在最小根迭代过程中的数据。
impl<G: Group> MinRootIteration<G> {// produces a sample non-deterministic advice, executing one invocation of MinRoot per stepfn new(num_iters: usize, x_0: &G::Scalar, y_0: &G::Scalar) -> (Vec<G::Scalar>, Vec<Self>) {// exp = (p - 3 / 5), where p is the order of the group// x^{exp} mod p provides the fifth root of xlet exp = {let p = G::group_params().2.to_biguint().unwrap();let two = BigUint::parse_bytes(b"2", 10).unwrap();let three = BigUint::parse_bytes(b"3", 10).unwrap();let five = BigUint::parse_bytes(b"5", 10).unwrap();let five_inv = five.modpow(&(&p - &two), &p);(&five_inv * (&p - &three)) % &p};

为什么这段代码得到 exp = (p - 3) / 5 mod p?

  1. let p = G::group_params().2.to_biguint().unwrap();
  • 从一个类型为 G 的结构体中获取群组参数,并提取第三个元素作为整数 p。
  • to_biguint().unwrap() 将其转换为大整数类型 (BigUint)。
  1. let two = BigUint::parse_bytes(b"2", 10).unwrap();
  • 创建一个大整数 two,其值为 2。
  1. let three = BigUint::parse_bytes(b"3", 10).unwrap();
  • 创建一个大整数 three,其值为 3。
  1. let five = BigUint::parse_bytes(b"5", 10).unwrap();
  • 创建一个大整数 five,其值为 5。
  1. let five_inv = five.modpow(&(&p - &two), &p);
  • 计算 5 的逆元(模 p 的情况下)。
  • 逆元的计算使用了模幂运算,即 5 的 (p - 2) 次方模 p。
  1. (&five_inv * (&p - &three)) % &p
  • 计算 (&p - &three) 乘以 five_inv 的结果,并对 p 取模。
  • 最终的计算结果被赋值给变量 exp,它表示 (p - 3) / 5 的值,将用于后续的模运算,以获得给定数的五次方根。
    关于第5步:
    涉及到模运算中的欧拉定理(Euler’s theorem)和费马小定理(Fermat’s little theorem)。
    费马小定理表明,如果 p 是一个质数,且 a 是不可被 p 整除的整数,那么 a^{p-1} mod p 等于 1。这是费马小定理的简化版本,而对于逆元的情况,我们可以将 a^{p-1} mod p 表示为 a^{p-2} mod p。
    在这里,5^{p-2} mod p 表示的是 5 在模 p 的情况下的逆元。这是因为 p 是一个大于 5 的质数,所以根据费马小定理,5^{p-1} mod p 等于 1,从而 5^{p-2} mod p 就是 5 在模 p 的情况下的逆元。
    逆元的概念是一个数在给定模下的乘法逆元,即与该数相乘的结果模该数的模等于 1。在这个场景中,计算 5^{p-2} mod p 就是为了得到 5 在模 p 的情况下的逆元,以便进行后续的乘法操作。
let mut res = Vec::new();let mut x_i = *x_0;let mut y_i = *y_0;for _i in 0..num_iters {let x_i_plus_1 = (x_i + y_i).pow_vartime(&exp.to_u64_digits()); // computes the fifth root of x_i + y_i// sanity checkif cfg!(debug_assertions) {let sq = x_i_plus_1 * x_i_plus_1;let quad = sq * sq;let fifth = quad * x_i_plus_1;assert_eq!(fifth, x_i + y_i);}let y_i_plus_1 = x_i;res.push(Self {x_i,y_i,x_i_plus_1,y_i_plus_1,});x_i = x_i_plus_1;y_i = y_i_plus_1;}let z0 = vec![*x_0, *y_0];(z0, res)}
  1. let mut res = Vec::new();
  • 创建一个可变的空向量 res,用于存储计算的结果。
  1. let mut x_i = *x_0;
  • 初始化一个可变变量 x_i,其初始值为 x_0 的值。
  1. let mut y_i = *y_0;
  • 初始化一个可变变量 y_i,其初始值为 y_0 的值。
  1. for _i in 0..num_iters {
  • 开始一个循环,迭代 num_iters 次。
  1. let x_i_plus_1 = (x_i + y_i).pow_vartime(&exp.to_u64_digits());
  • 计算 x_i + y_i 的五次方根,其中 exp 是之前计算得到的指数。
  1. if cfg!(debug_assertions) { ... }
  • 在调试模式下进行断言检查。这里执行了一个“sanity check”(合理性检查),确保计算的结果满足一定的条件。具体地,它验证了 x_i_plus_1 是否满足 (x_i_plus_1)^5 == x_i + y_i。
  1. let y_i_plus_1 = x_i;
  • 计算下一个 y_i 的值,赋值给 y_i_plus_1。
  1. res.push(Self { ... });
  • 将当前循环迭代的结果以结构体的形式存储在向量 res 中。
  1. x_i = x_i_plus_1;
  • 更新 x_i 的值为下一次迭代计算得到的 x_i_plus_1。
  1. y_i = y_i_plus_1;
  • 更新 y_i 的值为下一次迭代计算得到的 y_i_plus_1。
  1. let z0 = vec![*x_0, *y_0];(z0, res);
  • 循环结束后,创建一个包含初始输入值 x_0 和 y_0 的向量 z0。返回包含 z0 和 res 的元组。
    整体而言,这段代码执行了一系列的计算和更新操作,产生了一组结果,这些结果以结构体的形式存储在向量 res 中。在每次迭代中,通过计算 (x_i + y_i)^{1/5},将 x_i 和 y_i 更新为下一轮迭代的值。在调试模式下,还进行了一个断言检查,确保计算的结果符合预期的数学性质。
#[derive(Clone, Debug)]
struct MinRootCircuit<G: Group> {seq: Vec<MinRootIteration<G>>,
}impl<G: Group> StepCircuit<G::Scalar> for MinRootCircuit<G> {fn arity(&self) -> usize {2}fn synthesize<CS: ConstraintSystem<G::Scalar>>(&self,cs: &mut CS,z: &[AllocatedNum<G::Scalar>],) -> Result<Vec<AllocatedNum<G::Scalar>>, SynthesisError> {let mut z_out: Result<Vec<AllocatedNum<G::Scalar>>, SynthesisError> =Err(SynthesisError::AssignmentMissing);// use the provided inputslet x_0 = z[0].clone();let y_0 = z[1].clone();// variables to hold running x_i and y_ilet mut x_i = x_0;let mut y_i = y_0;for i in 0..self.seq.len() {// non deterministic advicelet x_i_plus_1 =AllocatedNum::alloc(cs.namespace(|| format!("x_i_plus_1_iter_{i}")), || {Ok(self.seq[i].x_i_plus_1)})?;// check the following conditions hold:// (i) x_i_plus_1 = (x_i + y_i)^{1/5}, which can be more easily checked with x_i_plus_1^5 = x_i + y_i// (ii) y_i_plus_1 = x_i// (1) constraints for condition (i) are below// (2) constraints for condition (ii) is avoided because we just used x_i wherever y_i_plus_1 is usedlet x_i_plus_1_sq = x_i_plus_1.square(cs.namespace(|| format!("x_i_plus_1_sq_iter_{i}")))?;let x_i_plus_1_quad =x_i_plus_1_sq.square(cs.namespace(|| format!("x_i_plus_1_quad_{i}")))?;cs.enforce(|| format!("x_i_plus_1_quad * x_i_plus_1 = x_i + y_i_iter_{i}"),|lc| lc + x_i_plus_1_quad.get_variable(),|lc| lc + x_i_plus_1.get_variable(),|lc| lc + x_i.get_variable() + y_i.get_variable(),);if i == self.seq.len() - 1 {z_out = Ok(vec![x_i_plus_1.clone(), x_i.clone()]);}// update x_i and y_i for the next iterationy_i = x_i;x_i = x_i_plus_1;}z_out}
}
  1. MinRootCircuit 结构体:
  • 定义了一个包含 MinRootIteration 结构体的向量 seq 的结构体。
  • 实现了 Clone 和 Debug trait,使得该结构体可以被克隆和打印调试信息。
  1. StepCircuit trait 实现:
  • 该 trait 提供了电路执行的接口,用于计算电路的输出。
  • arity 方法返回电路的输入数量,这里是2。
  • synthesize 方法用于在给定的约束系统 CS 上执行电路的计算。
  1. synthesize 方法具体实现:
  • 创建一个用于保存输出的 Result 对象 z_out,初始值为 Err(SynthesisError::AssignmentMissing)。
  • 通过模式匹配和 let 语句从输入数组 z 中获取两个输入 x_0 和 y_0。
  • 使用 for 循环遍历迭代序列 self.seq 中的元素。
  • 在循环中:
    • 使用 AllocatedNum::alloc 创建一个新的分配数字 x_i_plus_1,其值来自于 self.seq[i].x_i_plus_1。
    • 根据给定的条件,执行一些约束操作,确保 (x_i + y_i)^{1/5} = x_i_plus_1。
    • 如果当前迭代是最后一次迭代,将 z_out 更新为包含 x_i_plus_1 和 x_i 的 Ok 值。
    • 更新 x_i 和 y_i 为下一次迭代做准备。
  • 返回最终的 z_out。
fn main() {println!("Nova-based VDF with MinRoot delay function");println!("=========================================================");let num_steps = 10;for num_iters_per_step in [1024, 2048, 4096, 8192, 16384, 32768, 65536] {// number of iterations of MinRoot per Nova's recursive step...
}
  • 测试代码的主循环,分别设置每Nova递归步长MinRoot的迭代次数num_iters_per_step为[1024, 2048, 4096, 8192, 16384, 32768, 65536]
let circuit_primary = MinRootCircuit {seq: vec![MinRootIteration {x_i: <E1 as Engine>::Scalar::zero(),y_i: <E1 as Engine>::Scalar::zero(),x_i_plus_1: <E1 as Engine>::Scalar::zero(),y_i_plus_1: <E1 as Engine>::Scalar::zero(),};num_iters_per_step],};let circuit_secondary = TrivialCircuit::default();println!("Proving {num_iters_per_step} iterations of MinRoot per step");
  • 参考资料:https://www.youtube.com/watch?v=gopJn_QAdqU
  • circuit_primary 是业务电路,具体为MinRootCircuit。
  • circuit_secondary是trivial电路,被初始化为TrivialCircuit的default值。
  • 初始化x_i ,y_i ,x_i_plus_1 ,y_i_plus_1 都初始化为0
// produce public parameters
let start = Instant::now();
println!("Producing public parameters...");
let pp = PublicParams::<E1,E2,MinRootCircuit<<E1 as Engine>::GE>,TrivialCircuit<<E2 as Engine>::Scalar>,
>::setup(&circuit_primary,&circuit_secondary,&*S1::ck_floor(),&*S2::ck_floor(), 
);
println!("PublicParams::setup, took {:?} ", start.elapsed());
  • 在创建 PublicParams 实例的过程中,我们调用了 PublicParams::setup 方法。这个方法接受四个参数:circuit_primary、circuit_secondary、S1::ck_floor() 和 S2::ck_floor()。这四个参数分别表示主从电路,以及两个不同引擎的公共参数。
  • ck: CommitmentKey
    • Some final compressing SNARKs, like variants of Spartan, use computation commitments that require larger sizes for these parameters. These SNARKs provide a hint for these values by implementing RelaxedR1CSSNARKTrait::ck_floor(), which can be passed to this function.
// produce non-deterministic advice
let (z0_primary, minroot_iterations) = MinRootIteration::<<E1 as Engine>::GE>::new(num_iters_per_step * num_steps,&<E1 as Engine>::Scalar::zero(),&<E1 as Engine>::Scalar::one(),
);
let minroot_circuits = (0..num_steps).map(|i| MinRootCircuit {seq: (0..num_iters_per_step).map(|j| MinRootIteration {x_i: minroot_iterations[i * num_iters_per_step + j].x_i,y_i: minroot_iterations[i * num_iters_per_step + j].y_i,x_i_plus_1: minroot_iterations[i * num_iters_per_step + j].x_i_plus_1,y_i_plus_1: minroot_iterations[i * num_iters_per_step + j].y_i_plus_1,}).collect::<Vec<_>>(),}).collect::<Vec<_>>();let z0_secondary = vec![<E2 as Engine>::Scalar::zero()];
  • 首先,我们调用了 MinRootIteration::new 方法来创建一个新的 MinRootIteration 实例,这个实例包含了 z0_primary 和 minroot_iterations 两个部分。MinRootIteration::new 方法接受三个参数:num_iters_per_step * num_steps,::Scalar::zero() 和 ::Scalar::one()。表示我们要进行 num_iters_per_step * num_steps 次的最小根迭代,初始的 x 值为 0,初始的 y 值为 1。
  • 然后,我们创建了一个名为 minroot_circuits 的向量,这个向量包含了 num_steps 个 MinRootCircuit 实例。每个 MinRootCircuit 实例都包含了一个 seq 字段,这个字段是一个向量,包含了 num_iters_per_step 个 MinRootIteration 实例。每个 MinRootIteration 实例都是从 minroot_iterations 向量中取出的,取出的方式是通过计算 i * num_iters_per_step + j 来获取索引。
  • 最后,我们创建了一个名为 z0_secondary 的向量,这个向量只包含了一个元素,这个元素是 ::Scalar::zero()。这可能表示在第二个引擎中,我们的初始 z 值为 0。

type C1 = MinRootCircuit<<E1 as Engine>::GE>;
type C2 = TrivialCircuit<<E2 as Engine>::Scalar>;
// produce a recursive SNARK
println!("Generating a RecursiveSNARK...");
let mut recursive_snark: RecursiveSNARK<E1, E2, C1, C2> =RecursiveSNARK::<E1, E2, C1, C2>::new(&pp,&minroot_circuits[0],&circuit_secondary,&z0_primary,&z0_secondary,).unwrap();for (i, circuit_primary) in minroot_circuits.iter().enumerate() {let start = Instant::now();let res = recursive_snark.prove_step(&pp, circuit_primary, &circuit_secondary);assert!(res.is_ok());println!("RecursiveSNARK::prove_step {}: {:?}, took {:?} ",i,res.is_ok(),start.elapsed());
}
  • 首先定义了两种类型别名 C1 和 C2,分别对应 MinRootCircuit 和 TrivialCircuit 结构体。这两个结构体都需要一个泛型参数,分别是 E1 和 E2 引擎的关联类型。
  • 然后,我们创建了一个名为 recursive_snark 的 RecursiveSNARK 实例。RecursiveSNARK 是一个泛型结构体,需要四个类型参数:E1、E2、C1 和 C2。在创建 RecursiveSNARK 实例的过程中,我们调用了 RecursiveSNARK::new 方法。这个方法接受五个参数:pp、minroot_circuits[0]、circuit_secondary、z0_primary 和 z0_secondary。这五个参数分别表示公共参数、主电路、次电路,以及两个初始的 z 值。
  • 接下来,我们遍历 minroot_circuits 向量中的每一个元素。对于每一个元素,我们都调用了 recursive_snark.prove_step 方法。这个方法接受三个参数:pp、circuit_primary 和 circuit_secondary,并返回一个 Result 类型的值,表示证明步骤的结果。如果证明步骤成功,Result 的值将是 Ok;如果失败,Result 的值将是 Err。
  • 最后,我们使用 assert! 宏来检查 Result 的值。如果 Result 的值是 Err,assert! 宏将会触发一个 panic,程序将会终止运行。我们还使用 println! 宏来打印出每一步证明的结果和所花费的时间。
// verify the recursive SNARK
println!("Verifying a RecursiveSNARK...");
let start = Instant::now();
let res = recursive_snark.verify(&pp, num_steps, &z0_primary, &z0_secondary);
println!("RecursiveSNARK::verify: {:?}, took {:?}",res.is_ok(),start.elapsed()
);
assert!(res.is_ok());
- 验证 RecursiveSNARK 
// produce a compressed SNARKprintln!("Generating a CompressedSNARK using Spartan with multilinear KZG...");let (pk, vk) = CompressedSNARK::<_, _, _, _, S1, S2>::setup(&pp).unwrap();let start = Instant::now();let res = CompressedSNARK::<_, _, _, _, S1, S2>::prove(&pp, &pk, &recursive_snark);println!("CompressedSNARK::prove: {:?}, took {:?}",res.is_ok(),start.elapsed());assert!(res.is_ok());let compressed_snark = res.unwrap();let mut encoder = ZlibEncoder::new(Vec::new(), Compression::default());bincode::serialize_into(&mut encoder, &compressed_snark).unwrap();let compressed_snark_encoded = encoder.finish().unwrap();println!("CompressedSNARK::len {:?} bytes",compressed_snark_encoded.len());// verify the compressed SNARKprintln!("Verifying a CompressedSNARK...");let start = Instant::now();let res = compressed_snark.verify(&vk, num_steps, &z0_primary, &z0_secondary);println!("CompressedSNARK::verify: {:?}, took {:?}",res.is_ok(),start.elapsed());assert!(res.is_ok());
  • 证明/验证CompressedSNARK。
    遗留问题:
    [ ] PrimaryCircuit 和 SecondaryCircuit
    [ ] RecursiveSNARK 和 CompressedSNARK

相关文章:

ZKP Understanding Nova (1): MinRoot Example

Understanding Nova Kothapalli, Abhiram, Srinath Setty, and Ioanna Tzialla. “Nova: Recursive zero-knowledge arguments from folding schemes.” Annual International Cryptology Conference. Cham: Springer Nature Switzerland, 2022. Nova: Paper Code 1. Unders…...

0基础学java-day14

一、集合 前面我们保存多个数据使用的是数组&#xff0c;那么数组有不足的地方&#xff0c;我们分析一下 1.数组 2 集合 数据类型也可以不一样 3.集合的框架体系 Java 的集合类很多&#xff0c;主要分为两大类&#xff0c;如图 &#xff1a;[背下来] package com.hspedu.c…...

创建conan包-工具链

创建conan包-工具链 1 Toolchains 本文是基于对conan官方文档Toolchains翻译而来&#xff0c; 更详细的信息可以去查阅conan官方文档。 1 Toolchains Toolchains are the new way to integrate with build systems in Conan. Recipes can define a generate() method that wi…...

IntelliJ IDE 插件开发 | (二)UI 界面与数据持久化

系列文章 IntelliJ IDE 插件开发 |&#xff08;一&#xff09;快速入门 前言 在上一篇文章中介绍了在IDEA下开发、运行和安装插件的基本步骤&#xff0c;因此创建项目等基础步骤不再赘述&#xff0c;本文则开始介绍如何进行 UI 界面的开发以及相关数据的持久化存储&#xff…...

使用vue UI安装路由插件

1.使用vue创建项目 vue create vue-appvue ui 2.使用vue ui界面创建管理项目 终端页面输入&#xff1a;vue ui 创建项目 安装完成。可以直接在ui界面运行&#xff0c;也可以在编辑器中使用命令运行 安装路由&#xff0c;安装状态 选择插件 - 添加vue-router、添加vuex 安装…...

RPG项目01_脚本代码

基于“RPG项目01_场景及人物动画管理器”&#xff0c;我们创建一个XML文档 在资源文件夹下创建一个文件夹&#xff0c; 命名为Xml 将Xnl文档拖拽至文件夹中&#xff0c; 再在文件夹的Manager下新建脚本LoadManager 写代码&#xff1a; using System.Collections; using System…...

目标检测YOLO实战应用案例100讲-交通目标数据集构建及高性能检测算法研究与应用

目录 前言 国内外研究现状 目标检测研究现状 目标检测数据集研究现状...

浅谈Vue.js的计算属性computed

什么是computed属性 computed 属性用于声明计算属性&#xff0c;这些属性的值是基于其他响应式属性计算而来的&#xff0c;当依赖的响应式属性发生变化时&#xff0c;计算属性会自动重新计算。 与Vue.js 2相比&#xff0c;Vue.js 3的 computed 属性语法稍有变化&#xff0c;不…...

Linux常用指令详解

目录 前言&#xff1a; Linux的目录结构 Linux常用指令简介 whoami指令 ls指令 pwd指令 cd指令 tree指令 touch指令 mkdir指令 rmdir指令与rm指令 man指令 cp&#xff08;copy&#xff09;指令 mv&#xff08;move&#xff09;指令 cat指令 重定向及重定向的类型…...

Nginx(性能优化)

到这里文章的篇幅较长了&#xff0c;最后再来聊一下关于Nginx的性能优化&#xff0c;主要就简单说说收益最高的几个优化项&#xff0c;在这块就不再展开叙述了&#xff0c;毕竟影响性能都有多方面原因导致的&#xff0c;比如网络、服务器硬件、操作系统、后端服务、程序自身、数…...

机器学习笔记 - 如何在Python中对网格和点云进行体素化?

一、简述 本文主要是为了了解如何生成体素表示,体素之于3D就像像素之于2D。体素本质上是 3D 像素,但它们不是正方形,而是完美的立方体。 理论上,体素是复制现实的完美建模技术。 这里我们要了解四个广泛流行的 Python 库(Open3D、Trimesh、PyVista、pyntcloud )生成点云…...

冒个泡!OceanBase亮相 2023 新加坡金融科技节

近日&#xff0c;OceanBase 亮相 Singapore Fintech Festival 2023&#xff08;2023 新加坡金融科技节&#xff09;&#xff01;本届新加坡金融科技节于 2023 年 11 月 15 日至 17 日在新加坡博览展览中心举行&#xff0c;展会期间&#xff0c;OceanBase 得到了众多金融科技机构…...

正则表达式(5):常用符号

正则表达式&#xff08;5&#xff09;&#xff1a;常用符号 小结 本博文转载自 在本博客中&#xff0c;”正则表达式”为一系列文章&#xff0c;如果你想要从头学习怎样在Linux中使用正则&#xff0c;可以参考此系列文章&#xff0c;直达链接如下&#xff1a; 在Linux中使用正…...

Web安全漏洞分析-XSS(下)

随着互联网的迅猛发展&#xff0c;Web应用的普及程度也愈发广泛。然而&#xff0c;随之而来的是各种安全威胁的不断涌现&#xff0c;其中最为常见而危险的之一就是跨站脚本攻击&#xff08;Cross-Site Scripting&#xff0c;简称XSS&#xff09;。XSS攻击一直以来都是Web安全领…...

金南瓜SECS/GEM C# SDK 快速使用指南

本文对如何使用金南瓜SECS/GEM C# SDK 快速创建一个满足SECS/GEM通信要求的应用程序&#xff0c;只需简单3步完成。 第一步&#xff1a;创建C# .NET程序 示例使用Visual Studio 2010&#xff0c;使用者可以选择更高级版本 Visual Studio 第二步&#xff1a;添加DLL库引用&am…...

在一个没有超级用户的mongodb 生产库上如何添加超级用户

说来这个问题&#xff0c;都觉得不可思议&#xff0c;一个数据库怎么没有超级用户呢&#xff0c;我们知道&#xff0c;MYSQL&#xff0c;PG&#xff0c;ORACLE等&#xff0c;创建好后&#xff0c;都有一个默认的超级用户&#xff0c;MONGODB也有超级用户&#xff0c;但需要自己…...

排序算法之二:冒泡排序

冒泡排序的思路 冒泡排序是交换排序 基本思想&#xff1a;所谓交换&#xff0c;就是根据序列中两个记录键值的比较结果来对换这两个记录在序列中的位置&#xff0c;交换排序的特点是&#xff1a;将键值较大的记录向序列的尾部移动&#xff0c;键值较小的记录向序列的前部移动…...

一键搭建你的hnust请假条

hnust请假条 湖南科技大学请假条生成器 https://hnust.rick.icu/new &#xff08;直接使用&#xff09; Hnust Leave Note 去github https://github.com/rickhqh/hnust_leave_note 效果展示 界面展示效果图 v2.0 更新 vant和vue重构了整个源码同步学校新版请假条样式修复了…...

C练习题13

单项选择题(本大题共20小题,每小题2分,共40分。在每小题给出的四个备选项中,选出一个正确的答案,并将所选项前的字母填写在答题纸的相应位置上。) 1.结构化程序由三种基本结构组成、三种基本结构组成的算法是() A.可以完成任何复杂的任务 B. 只能完成部分复杂的任务 C. 只能完…...

交易历史记录20231206 记录

昨日回顾&#xff1a; select top 10000 * from dbo.CODEINFO A left join dbo.全部&#xff21;股20231206010101 B ON A.CODE B.代码 left join dbo.全部&#xff21;股20231206CONF D on A.CODED.代码left join dbo.全部&#xff21;股20231206 G on A.CODEG.代码 left…...

1-5总体分布的推断

...

深信服技术认证“SCSA-S”划重点:XSS漏洞

为帮助大家更加系统化地学习网络安全知识&#xff0c;以及更高效地通过深信服安全服务认证工程师考核&#xff0c;深信服特别推出“SCSA-S认证备考秘笈”共十期内容&#xff0c;“考试重点”内容框架&#xff0c;帮助大家快速get重点知识~ 划重点来啦 *点击图片放大展示 深信服…...

MIT6S081-Lab2总结

大家好&#xff0c;我叫徐锦桐&#xff0c;个人博客地址为www.xujintong.com&#xff0c;github地址为https://github.com/xjintong。平时记录一下学习计算机过程中获取的知识&#xff0c;还有日常折腾的经验&#xff0c;欢迎大家访问。 Lab2就是了解一下xv6的系统调用流程&…...

CMMI5大成熟度等级和4大过程域

CMMI&#xff08;Capability Maturity Model Integration&#xff0c;能力成熟度模型集成&#xff09;模型系列是帮助组织改进其过程的最佳实践的集合。这些模型由来自产业界、政府以及软件工程研究所&#xff08;Software Engineering Institute&#xff0c; SEI&#xff09;的…...

c++新经典模板与泛型编程:const修饰符的移除与增加

const修饰符的移除 让你来写移除const修饰符&#xff0c;你会怎么样来写&#xff1f; &#x1f602;&#x1f602;trait类模板&#xff0c;如下 #include <iostream>// 泛化版本 template<typename T> struct RemoveConst {using type T; };// 特化版本 template…...

AUTOSAR汽车电子嵌入式编程精讲300篇-基于加密算法的车载CAN总线安全通信

目录 前言 研究现状 系统架构研究 异常检测研究 认证与加密研究 相关技术 2.1车联网 2.2车载网络及总线 2.2.1 CAN总线基础 2.2.2 CAN总线网络安全漏洞 2.2.3 CAN总线信息安全需求 2.3密码算法 2.3.1 AES算法 2.3.2 XTEA算法 CAN网络建模与仿真 3.1 CAN网络建模…...

4-Docker命令之docker start

1.docker start介绍 docker start命令是用来启动一个或多个已经被停止的docker容器。 2.docker start用法 docker start [参数] container [container......] [root@centos79 ~]# docker start --helpUsage: docker start [OPTIONS] CONTAINER [CONTAINER...]Start one or…...

AWS Remote Control ( Wi-Fi ) on i.MX RT1060 EVK - 2 “架构 AWS”

接续上一章节&#xff0c;我们把开发环境架设好之后&#xff0c;此章节叙述如何建立 AWS IoT 环境&#xff0c;请务必已经有 AWS Account&#xff0c;申请 AWS Account 之流程将不在此说明。 III-1. 登入AWS IoT&#xff0c; 在“管理”>“所有装置”>“实物”下点击“建…...

日志框架梳理(Log4j,Reload4j,JUL,JCL,SLF4J,Logback,Log4j2)

原文链接 日志框架发展历程 在了解日志框架时总会列出一系列框架&#xff1a;Log4j&#xff0c;Reload4j&#xff0c;JUL&#xff0c;JCL&#xff0c;SLF4J&#xff0c;Logback&#xff0c;Log4j2&#xff0c;这么多框架让人感到混乱&#xff0c;该怎么选取、该怎么用。接下来…...

内核无锁队列kfifo

文章目录 1、抛砖引玉2、内核无锁队列kfifo2.1 kfifo结构2.2 kfifo分配内存2.3 kfifo初始化2.4 kfifo释放2.5 kfifo入队列2.6 kfifo出队列2.7 kfifo的判空和判满2.8 关于内存屏障 1、抛砖引玉 昨天遇到这样一个问题&#xff0c;有多个生产者&#xff0c;多个消费者&#xff0c…...

18、XSS——cookie安全

文章目录 1、cookie重要字段2、子域cookie机制3、路径cookie机制4、HttpOnly Cookie机制5、Secure Cookie机制6、本地cookie与内存cookie7、本地存储方式 一般来说&#xff0c;同域内浏览器中发出的任何一个请求都会带上cookie&#xff0c;无论请求什么资源&#xff0c;请求时&…...

从零开发短视频电商 Jmeter压测示例模板详解(无认证场景)

文章目录 添加线程组添加定时器添加HTTP请求默认值添加HTTP头管理添加HTTP请求添加结果断言响应断言 Response AssertionJSON断言 JSON Assertion持续时间断言 Duration Assertion 添加察看结果树添加聚合报告添加表格察看结果参考 以压测百度搜索为例 https://www.baidu.com/s…...

C++可以函数重载而C不可以的原因

函数重载是指在同一个作用域内&#xff0c;可以定义多个函数&#xff0c;它们具有相同的名称但是参数列表不同。函数重载的主要原理是函数的签名不同&#xff0c;而在 C 中&#xff0c;函数签名包括函数的名称和参数列表。而在 C 中&#xff0c;函数的标识仅依赖于函数的名称&a…...

Spark常见算子汇总

创建RDD 在Spark中创建RDD的方式分为三种: 从外部存储创建RDD从集合中创建RDD从其他RDD创建 textfile 调用SparkContext.textFile()方法&#xff0c;从外部存储中读取数据来创建 RDD parallelize 调用SparkContext 的 parallelize()方法&#xff0c;将一个存在的集合&…...

【华为数据之道学习笔记】3-1 基于数据特性的分类管理框架

华为根据数据特性及治理方法的不同对数据进行了分类定义&#xff1a;内部数据和外部数据、结构化数据和非结构化数据、元数据。其中&#xff0c;结构化数据又进一步划分为基础数据、主数据、事务数据、报告数据、观测数据和规则数据。 对上述数据分类的定义及特征描述。 分类维…...

电脑版便签软件怎么设置在桌面上显示?

对于不少上班族来说&#xff0c;如果想要在使用电脑办公的时候&#xff0c;随手记录一些常用的工作资料、工作注意事项等内容&#xff0c;直接在电脑上使用便签软件记录是比较方便的。电脑桌面便签工具不仅方便我们随时记录各类工作事项&#xff0c;而且支持我们快速便捷使用这…...

【华为数据之道学习笔记】2-建立企业级数据综合治理体系

数据作为一种新的生产要素&#xff0c;在企业构筑竞争优势的过程中起着重要作用&#xff0c;企业应将数据作为一种战略资产进行管理。数据从业务中产生&#xff0c;在IT系统中承载&#xff0c;要对数据进行有效治理&#xff0c;需要业务充分参与&#xff0c;IT系统确保遵从&…...

【IC前端虚拟项目】git和svn项目托管平台的简单使用说明

【IC前端虚拟项目】数据搬运指令处理模块前端实现虚拟项目说明-CSDN博客 代码托管在gitee平台上&#xff0c;进去后会看到文档目录“MVU芯片前端设计验证虚拟项目”和工程目录“mvu_prj”&#xff0c;可以通过git来下载工程&#xff1a; git clone gitgitee.com:gjm9999/ic_vi…...

C++ IO库

IO类 IO对象不能拷贝和赋值 iostream 表示形式的变化&#xff1a; 将100转换成二进制序列 然后格式化输出 x,y共用一块内存 输出的时候用不同的方式解析同一块内存 操作 格式化&#xff1a;内部表示转换为相应字节序列 缓存&#xff1a;要输出的内容放到缓存 编码转换&…...

Springboot 项目关于版本升级到 3.x ,JDK升级到17的相关问题

由于spring 停止对2.x 版本的维护&#xff0c;以及 jdk 频繁发布等客观因素&#xff0c;现需要对已有springboot 工程做一次全面升级&#xff1b;已因对市面上第三方等依赖库的兼容问题&#xff1b; 现有工程使用哥技术栈是版本&#xff1a; freemarker &#xff1a;2.3.32 spr…...

QGraphicsView实现简易地图7『异步加载-多瓦片-无底图』

前文链接&#xff1a;QGraphicsView实现简易地图6『异步加载-单瓦片-无底图』 前一篇文章提到的异步单瓦片加载&#xff0c;是指线程每准备好一个瓦片数据后&#xff0c;立刻抛出信号让主线程加载。而本篇异步多瓦片加载是指线程准备好所有瓦片数据后&#xff0c;一起抛出信号让…...

Spring Boot学习(三十三):集成kafka

前言 下面是zookeeper和kafka的官网下载地址&#xff0c;大家可以学习下载 zookeeper下载地址&#xff1a;http://zookeeper.apache.org/releases.html kafka下载地址&#xff1a;http://kafka.apache.org/downloads.html 1、添加依赖 在 pom.xml 文件中添加kafka依赖&am…...

MOSFET

MOSFET 电子元器件百科 文章目录 MOSFET前言一、MOSFET是什么二、MOSFET类别三、MOSFET应用实例四、MOSFET作用原理总结前言 MOSFET是一种常见的半导体器件,通过栅极电场控制通道区的导通特性,以控制电流流动。它在现代电子电路中发挥着重要的作用,并广泛应用于各种应用领域…...

DriveWorks——参数化设计非标定制利器

DriveWorks基本介绍 DriveWorks是一套被 SOLIDWORKS 认可为金牌合作伙伴产品的设计自动化软件。DriveWorks 可自动创建特定于订单的销售文档和 SOLIDWORKS 制造数据。减少重复性任务&#xff0c;消除错误&#xff0c;增加销售额&#xff0c;并在创纪录的时间内交付定制产品。 为…...

DevEco Studio集成ArkUI-X

语雀知识库地址&#xff1a;语雀HarmonyOS知识库 飞书知识库地址&#xff1a;飞书HarmonyOS知识库 在上篇文章(HarmonyOS应用开发工具DevEco Studio安装与使用)中我说到官方推出了4.0 Beta版本的IDE&#xff0c;这篇文章就来介绍这个版本的安装与使用 该版本集成了HarmonyOS多…...

网络视频服务器的作用是什么?

随着互联网的快速发展和网络带宽的提升&#xff0c;网络视频已经成为人们日常生活中不可或缺的一部分。网络视频服务器作为支持和传输网络视频的关键基础设施&#xff0c;发挥着重要的作用。本文将以网络视频服务器的作用为方向&#xff0c;探讨其在现代社会中的重要性。 首先…...

解决vue3使用iconpark控制台预警提示问题

前言 最近在项目中使用 iconpark-icon 来管理图标&#xff0c;一切都很顺利&#xff0c;引入链接后&#xff0c;图标正常显示&#xff0c;没有报错。但是控制台却发出了预警信息。 [Vue warn]: Failed to resolve component: iconpark-icon If this is a native custom eleme…...

VMware 虚拟机 NAT 模式网络配置

配置的核心点在于 网关要一致&#xff0c;才能访问外网 比如下面的网关都是&#xff1a;192.168.145.2 问题总结&#xff1a; 当时重启电脑后如果连不上外网了&#xff0c;检查下 windows 服务中 NAT服务是否已经启动...

5-redis高级-哨兵

1 哨兵 1.1 python 操作哨兵 1 哨兵 # 主从---》一主多从-主库用来写-从库用来读-主库挂了--》整个系统就不能写数据了#主从复制存在的问题&#xff1a;1 主从复制&#xff0c;主节点发生故障&#xff0c;需要做故障转移&#xff0c;可以手动转移&#xff1a;让其中一个slave变…...

鸿蒙HarmonyOS4.0开发应用学习笔记

黑马程序员鸿蒙4.0视频学习笔记&#xff0c;供自己回顾使用。1.安装开发工具DevEco Studio 鸿蒙harmony开发文档指南 DevEco Studio下载地址 选择或者安装环境 选择和下载SDK 安装总览 编辑器界面 2.TypeScript语法 2.1变量声明 //string 、number、boolean、any、u…...