CoverM contig mean 和 trim_mean 方法原理

  • 由于 CoverM 存储库更新, 记录的代码可能与最新代码稍有不同

Contig

分析 mean 方法

  • contig 方法的入口位于 src/bin/coverm::main 的 “contig” 分支中, 忽略帮助等参数, 假设已有 bam 文件, 不设置 reads 的额外 filter_params, 则函数主体简写如下:
    fn main() {let mut app = build_cli();let matches = app.clone().get_matches();set_log_level(&matches, false);let mut print_stream;match matches.subcommand_name() {Some("contig") => {let m = matches.subcommand_matches("contig").unwrap();let print_zeros = true;// Add metabat filtering params since we are running contiglet filter_params = FilterParameters::generate_from_clap(m);let threads = *m.get_one::<u16>("threads").unwrap();print_stream = OutputWriter::generate(m.get_one::<String>("output-file").map(|x| &**x));let mut estimators_and_taker =EstimatorsAndTaker::generate_from_clap(m, print_stream.clone());estimators_and_taker =estimators_and_taker.print_headers("Contig", print_stream.clone());if m.contains_id("bam-files") {let bam_files: Vec<&str> = m.get_many::<String>("bam-files").unwrap().map(|x| &**x).collect();if filter_params.doing_filtering() {unreachable!()} else if m.get_flag("sharded") {unreachable!()} else {let bam_readers =coverm::bam_generator::generate_named_bam_readers_from_bam_files(bam_files);run_contig(&mut estimators_and_taker,bam_readers,print_zeros,filter_params.flag_filters,threads,&mut print_stream,);}} else {unreachable!()}}_ => {app.print_help().unwrap();println!();}}
    }
    
    • filter_params = FilterParameters::generate_from_clap(m); 获得了对 bam 文件过滤的参数, 这些参数的获取在 “genome”, “filter”, 和 “contig” 三个分支中是相同的, 在这里先忽略
    • let bam_readers = coverm::bam_generator::generate_named_bam_readers_from_bam_files(bam_files); 将 bam 文件名转换为结构化的对象
    • 随后通过 run_contig 进行读取
run_contig
  • run_contig 如下:
    fn run_contig<R: coverm::bam_generator::NamedBamReader,T: coverm::bam_generator::NamedBamReaderGenerator<R>,
    >(estimators_and_taker: &mut EstimatorsAndTaker,bam_readers: Vec<T>,print_zeros: bool,flag_filters: FlagFilter,threads: u16,print_stream: &mut OutputWriter,
    ) {let reads_mapped = coverm::contig::contig_coverage(bam_readers,&mut estimators_and_taker.taker,&mut estimators_and_taker.estimators,print_zeros,&flag_filters,threads,);debug!("Finalising printing ..");estimators_and_taker.printer.finalise_printing(&estimators_and_taker.taker,print_stream,Some(&reads_mapped),&estimators_and_taker.columns_to_normalise,estimators_and_taker.rpkm_column,estimators_and_taker.tpm_column,);
    }
    
    • 其首先使用 coverm::contig::contig_coverage 计算 coverage, 随后再通过 estimators_and_taker.printer.finalise_printing 输出
contig_coverage
  • contig_coverage 的主体框架如下:
    pub fn contig_coverage<R: NamedBamReader, G: NamedBamReaderGenerator<R>, T: CoverageTaker>(bam_readers: Vec<G>,coverage_taker: &mut T,coverage_estimators: &mut Vec<CoverageEstimator>,print_zero_coverage_contigs: bool,flag_filters: &FlagFilter,threads: u16,
    ) -> Vec<ReadsMapped> {let mut reads_mapped_vector = vec![];for bam_generator in bam_readers {let mut bam_generated = bam_generator.start();中间步骤let reads_mapped = ReadsMapped {num_mapped_reads: num_mapped_reads_total,num_reads: bam_generated.num_detected_primary_alignments(),};reads_mapped_vector.push(reads_mapped);if bam_generated.num_detected_primary_alignments() == 0 {warn!("No primary alignments were observed for sample {} \- perhaps something went wrong in the mapping?",stoit_name);}bam_generated.finish();}reads_mapped_vector
    }
    
    • 可见, coverm 逐个处理 bam 文件, 记录每个 bam 的 num_mapped_reads 和 reads 总数 (bam_generated.num_detected_primary_alignments(), 仅不考虑 is_secondaryis_supplementary 情况, 但是允许 unmapped), 随后返回这个列表
  • 在这个函数的中间步骤中首先初始化参数, 随后进入一个巨大的 loop 循环
    // for record in records
    loop {match bam_generated.read(&mut record) {None => {break;}Some(Ok(())) => {}Some(e) => {panic!("Error reading BAM record: {:?}", e)}}
    
    • 在这个 loop 循环中, 首先读取 reads 比对 (不考虑 paired)
    • 仅处理质量足够好的 reads:
      if !flag_filters.passes(&record) {trace!("Skipping read based on flag filtering");continue;
      }
      let tid = record.tid();// if mapped
      if !record.is_unmapped() {
      
      • 假设 bam 文件已经按 contig 完成排序 (也就是说, 所有比对到同一个 reference 上的 reads 都在一起), 那么需要报告所有上一条 reads 的结果:
        // if reference has changed, print the last record
        if tid != last_tid {if tid < last_tid {error!("BAM file appears to be unsorted. Input BAM files must be sorted by reference (i.e. by samtools sort)");panic!("BAM file appears to be unsorted. Input BAM files must be sorted by reference (i.e. by samtools sort)");}process_previous_contigs(last_tid,tid,coverage_estimators,&ups_and_downs,num_mapped_reads_in_current_contig,total_edit_distance_in_current_contig,total_indels_in_current_contig,&mut num_mapped_reads_total,);
        
        • 由于这种结构, 可见 bam 中最后一个 contig 的比对序列结束后循环也结束了, 因此在循环结束后还需进行一次 process_previous_contigs
        • 随后重置参数, 以处理 (当前 reads mapping 的) 下一条 contig 的结果
          ups_and_downs =vec![0; header.target_len(tid as u32).expect("Corrupt BAM file?") as usize];
          debug!("Working on new reference {}",std::str::from_utf8(target_names[tid as usize]).unwrap()
          );
          last_tid = tid;
          num_mapped_reads_in_current_contig = 0;
          total_edit_distance_in_current_contig = 0;
          total_indels_in_current_contig = 0;
          
      • 按照 reads mapping 的 cigar 序列, 向整数向量 ups_and_downs 中添加 mapped bases 的变化率:
        }
        if !record.is_supplementary() && !record.is_secondary() {num_mapped_reads_in_current_contig += 1;
        }// for each chunk of the cigar string
        trace!("read name {:?}",std::str::from_utf8(record.qname()).unwrap()
        );
        let mut cursor: usize = record.pos() as usize;
        for cig in record.cigar().iter() {trace!("Found cigar {:} from {}", cig, cursor);match cig {Cigar::Match(_) | Cigar::Diff(_) | Cigar::Equal(_) => {// if M, X, or = increment start and decrement end indextrace!("Adding M, X, or = at {} and {}",cursor,cursor + cig.len() as usize);ups_and_downs[cursor] += 1;let final_pos = cursor + cig.len() as usize;if final_pos < ups_and_downs.len() {// True unless the read hits the contig end.ups_and_downs[final_pos] -= 1;}cursor += cig.len() as usize;}Cigar::Del(_) => {cursor += cig.len() as usize;total_indels_in_current_contig += cig.len() as u64;}Cigar::RefSkip(_) => {// if D or N, move the cursorcursor += cig.len() as usize;}Cigar::Ins(_) => {total_indels_in_current_contig += cig.len() as u64;}Cigar::SoftClip(_) | Cigar::HardClip(_) | Cigar::Pad(_) => {}}
        }
        
        • 找到所有可比对的区域 (M, X, or =)
        • 比对区域起始处 ups_and_downs[cursor] += 1
        • 终止处 ups_and_downs[cursor + cig.len() as usize] -= 1
process_previous_contigs
  • process_previous_contigs 是一个在 contig_coverage 中定义的函数, 接受如下参数:
    let mut process_previous_contigs =|last_tid,tid,coverage_estimators: &mut Vec<CoverageEstimator>,ups_and_downs: &[i32],num_mapped_reads_in_current_contig,total_edit_distance_in_current_contig,total_indels_in_current_contig,num_mapped_reads_total: &mut u64| {
    
    • 随后 (忽略 last_tid == -2 即初始情况), 进行覆盖度的估计:
      for estimator in coverage_estimators.iter_mut() {estimator.add_contig(ups_and_downs,num_mapped_reads_in_current_contig,total_edit_distance_in_current_contig - total_indels_in_current_contig,)
      }
      let coverages: Vec<f32> = coverage_estimators.iter_mut().map(|estimator| estimator.calculate_coverage(&[0])).collect();
      
      1. 使用 estimator.add_contig 总结当前 contig 的 mapping 情况
      2. 使用 estimator.calculate_coverage 计算丰度, 此时设 unobserved_contig_lengths: &[0] 即忽略较短的 contigs
    • 输出到表格对应 last_tid 的行中
      coverage_taker.start_entry(last_tid as usize,std::str::from_utf8(target_names[last_tid as usize]).unwrap(),
      );
      for (coverage, estimator) incoverages.iter().zip(coverage_estimators.iter_mut())
      {estimator.print_coverage(coverage, coverage_taker);
      }
      coverage_taker.finish_entry();
      
    • 在 contig 中, 无需考虑多个 contig 的情况, 因此重置 estimator
      for estimator in coverage_estimators.iter_mut() {estimator.setup();
      }
      
MeanGenomeCoverageEstimator
add_contig-MeanGenomeCoverageEstimator
  • 考虑 mean 方法:
    fn add_contig(&mut self,ups_and_downs: &[i32],num_mapped_reads_in_contig: u64,total_mismatches_in_contig: u64,
    ) {match self {CoverageEstimator::MeanGenomeCoverageEstimator {ref mut total_count,ref mut total_bases,ref mut num_covered_bases,ref mut num_mapped_reads,ref mut total_mismatches,contig_end_exclusion,..} => {*num_mapped_reads += num_mapped_reads_in_contig;*total_mismatches += total_mismatches_in_contig;let len = ups_and_downs.len();match *contig_end_exclusion * 2 < len as u64 {true => *total_bases += len as u64 - 2 * *contig_end_exclusion,false => {debug!("Contig too short - less than twice the contig-end-exclusion");return; //contig is all ends, too short}}let mut cumulative_sum: i32 = 0;let start_from = *contig_end_exclusion as usize;let end_at = len - *contig_end_exclusion as usize - 1;for (i, current) in ups_and_downs.iter().enumerate() {cumulative_sum += current;if i >= start_from && i <= end_at {if cumulative_sum > 0 {*num_covered_bases += 1}*total_count += cumulative_sum as u64;}}
    
    • ups_and_downs 的信息, 更新 estimator:
    • 最终对应于每条 contigMeanGenomeCoverageEstimator 的信息如下:
      • total_count: u64: 全部能 mapped 到这条 contig 上的 reads 的匹配区域的总碱基数 (modified by contig_end_exclusion)
      • total_bases: u64: 作为模板的 contig 的总长度 (modified by contig_end_exclusion)
      • num_covered_bases: u64: 作为模板的 contig 上, 有 reads 覆盖的总长度 (modified by contig_end_exclusion)
      • num_mapped_reads: u64: 匹配且是首选匹配到这条 contig 上的 reads 的数量
      • total_mismatches: u64
      • min_fraction_covered_bases: f32
      • contig_end_exclusion: u64: 默认 75, 可能是考虑到 contig 末端匹配的可靠性相对较低
      • exclude_mismatches: bool
calculate_coverage-MeanGenomeCoverageEstimator
  • 考虑 mean 方法:
    fn calculate_coverage(&mut self, unobserved_contig_lengths: &[u64]) -> f32 {match self {CoverageEstimator::MeanGenomeCoverageEstimator {total_count,total_bases,num_covered_bases,num_mapped_reads: _,total_mismatches,contig_end_exclusion,min_fraction_covered_bases,exclude_mismatches,} => {let final_total_bases = *total_bases// + 0+ CoverageEstimator::calculate_unobserved_bases(unobserved_contig_lengths,*contig_end_exclusion,);debug!("Calculating coverage with unobserved {:?}, \total bases {}, num_covered_bases {}, total_count {}, \total_mismatches {}, final_total_bases {}",unobserved_contig_lengths,total_bases,num_covered_bases,total_count,total_mismatches,final_total_bases,);if final_total_bases == 0|| (*num_covered_bases as f32 / final_total_bases as f32)< *min_fraction_covered_bases{0.0} else {let calculated_coverage = match exclude_mismatches {true => (*total_count - *total_mismatches) as f32,false => *total_count as f32,} / final_total_bases as f32;debug!("Found mean coverage {}", calculated_coverage);calculated_coverage}
    
    • 在 contig 方法中, unobserved_contig_lengths 设为 0
    • num_covered_bases / final_total_bases 较低, 则直接返回 0
      • 意味着所有 map 到这个 contig 上的 reads 全部不算数了~
    • 在当前版本中, 暂不考虑 exclude_mismatches
    • mean 方法的 calculated_coverage 计算方式为:
      • total_count / total_bases
      • 其中每条 contig 两侧长达 contig_end_exclusion 的片段不在计算平均值的范围内.

分析 trimmed_mean 方法

  • 假设已有 bam 文件, 不设置 reads 的额外 filter_params, contig 函数主体简写见上
    • 计算 coverage method 的方法选择是通过 mut estimators_and_taker = EstimatorsAndTaker::generate_from_clap(m, print_stream.clone()); 设置的:
      let mut estimators_and_taker =EstimatorsAndTaker::generate_from_clap(m, print_stream.clone());
      
      • EstimatorsAndTaker::generate_from_clap 具有一个 CoverageEstimator 列表 (estimators_and_taker.estimators)
        • 前述的 “mean” 对应的即为 MeanGenomeCoverageEstimator
        • “trimmed_mean” 对应的即为 TrimmedMeanGenomeCoverageEstimator
        • “covered_fraction” 对应的即为 CoverageFractionGenomeCoverageEstimator
  • 一直到进行覆盖度的估计, 所有中间的所有过程都没有区别, 直到
    1. 使用 estimator.add_contig 总结当前 contig 的 mapping 情况
    2. 使用 estimator.calculate_coverage 计算丰度, 此时设 unobserved_contig_lengths: &[0] 即忽略较短的 contigs
TrimmedMeanGenomeCoverageEstimator
add_contig-TrimmedMeanGenomeCoverageEstimator
  • trimmed_mean 方法:
    CoverageEstimator::TrimmedMeanGenomeCoverageEstimator {ref mut counts,ref mut observed_contig_length,ref mut num_covered_bases,ref mut num_mapped_reads,contig_end_exclusion,..
    } => {*num_mapped_reads = num_mapped_reads_in_contig;let len1 = ups_and_downs.len();match *contig_end_exclusion * 2 < len1 as u64 {true => {debug!("Adding len1 {}", len1);*observed_contig_length += len1 as u64 - 2 * *contig_end_exclusion}false => {debug!("Contig too short - less than twice the contig-end-exclusion");return; //contig is all ends, too short}}debug!("Total observed length now {}", *observed_contig_length);let mut cumulative_sum: i32 = 0;let start_from = *contig_end_exclusion as usize;let end_at = len1 - *contig_end_exclusion as usize - 1;debug!("ups and downs {:?}", ups_and_downs);for (i, current) in ups_and_downs.iter().enumerate() {if *current != 0 {debug!("At i {}, cumulative sum {} and current {}",i, cumulative_sum, current);}cumulative_sum += current;if i >= start_from && i <= end_at {if cumulative_sum > 0 {*num_covered_bases += 1}if counts.len() <= cumulative_sum as usize {(*counts).resize(cumulative_sum as usize + 1, 0);}(*counts)[cumulative_sum as usize] += 1}}
    
    • 此时不考虑 total_mismatches_in_contig (错配)
    • 最终对应于每条 contigTrimmedMeanGenomeCoverageEstimator 的信息如下:
      • counts: Vec: counts[n_base] = n_site 记录了 reference contig 上, 有多少 (n_site) 个碱基能比对上给定数量 (n_base) 条的 reads
      • observed_contig_length: u64: 同 MeanGenomeCoverageEstimator.total_bases
      • num_covered_bases: u64: (同上)
      • num_mapped_reads: u64: 当前 contig 的 num_mapped_reads
        • *num_mapped_reads = num_mapped_reads_in_contig
      • min: f32:
      • max: f32:
      • min_fraction_covered_bases: f32: (同上)
      • contig_end_exclusion: u64: (同上)
calculate_coverage-TrimmedMeanGenomeCoverageEstimator
  • trimmed_mean 方法 类似于 mean 方法, 但首先考虑了 coverage 为 0 的情况:
    CoverageEstimator::TrimmedMeanGenomeCoverageEstimator {counts,observed_contig_length,num_covered_bases,num_mapped_reads: _,contig_end_exclusion,min_fraction_covered_bases,min,max,
    } => {let unobserved_contig_length = CoverageEstimator::calculate_unobserved_bases(unobserved_contig_lengths,*contig_end_exclusion,);let total_bases = *observed_contig_length + unobserved_contig_length;debug!("Calculating coverage with num_covered_bases {}, observed_length {}, unobserved_length {:?} and counts {:?}",num_covered_bases, observed_contig_length, unobserved_contig_lengths, counts);match total_bases {0 => 0.0,_ => {if (*num_covered_bases as f32 / total_bases as f32)< *min_fraction_covered_bases{0.0} else {let min_index: usize = (*min * total_bases as f32).floor() as usize;let max_index: usize = (*max * total_bases as f32).ceil() as usize;if *num_covered_bases == 0 {return 0.0;}counts[0] += unobserved_contig_length;
    
    • 同上:
      • unobserved_contig_lengthscounts[0] 在 contig 情况下为 0
      • 忽略所有 num_covered_bases / final_total_bases 较低的 contigs 的 reads 和丰度
    • 根据 TrimmedMeanGenomeCoverageEstimator.minTrimmedMeanGenomeCoverageEstimator.max, 计算了对应 contig 总长的 min_indexmax_index (尽量取大范围)
    • 其他 (即 coverage 不等于 0) 的情况:
                  let mut num_accounted_for: usize = 0;let mut total: usize = 0;let mut started = false;for (i, num_covered) in counts.iter().enumerate() {num_accounted_for += *num_covered as usize;debug!("start: i {}, num_accounted_for {}, total {}, min {}, max {}",i, num_accounted_for, total, min_index, max_index);if num_accounted_for >= min_index {debug!("inside");if started {if num_accounted_for > max_index {debug!("num_accounted_for {}, *num_covered {}",num_accounted_for, *num_covered);let num_excess =num_accounted_for - *num_covered as usize;let num_wanted = match max_index >= num_excess {true => max_index - num_excess + 1,false => 0,};debug!("num wanted1: {}", num_wanted);total += num_wanted * i;break;} else {total += *num_covered as usize * i;}} else if num_accounted_for > max_index {// all coverages are the same in the trimmed settotal = (max_index - min_index + 1) * i;started = true} else if num_accounted_for < min_index {debug!("too few on first")} else {let num_wanted = num_accounted_for - min_index + 1;debug!("num wanted2: {}", num_wanted);total = num_wanted * i;started = true;}}debug!("end i {}, num_accounted_for {}, total {}",i, num_accounted_for, total);}total as f32 / (max_index - min_index) as f32}}
      }
      
      • 按 reads 覆盖数 (从小到大) 排序后, 逐步处理其中的序列
        • 一开始 started = false, 并逐步累加 num_accounted_for (当前处理的总 contig base 数量)
        • 当超过 min_index 后, 开始累加总匹配 reads 的 base 数到 total
        • 直到超过 min_index, 直接中断返回
    • trimmed_mean 方法的 calculated_coverage 计算方式为:
      • total_count / total_bases
      • 其中每条 contig 两侧长达 contig_end_exclusion 的片段不在计算平均值的范围内
      • 此外, 覆盖度最高的 max% 和最低的 min% 部分的 base 不在计算平均值的范围内

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

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

相关文章

4. seaborn-线性关系可视化

Seaborn本身并不是为了统计分析而生的&#xff0c;seaborn中的回归图主要用于添加视觉指南&#xff0c;以帮助在探索性数据分析EDA中强调存在于数据集的模式。 import numpy as np import matplotlib.pyplot as plt import seaborn as sns import warningssns.set(stylewhiteg…

JDK 版本切换工具 JEnv

1. 下载 JEnv 包 下载地址&#xff1a; JEnv-for-Windows 下载 JEnv.zip 然后解压缩&#xff0c;放到一个目录下&#xff0c;我这里放到了目录&#xff1a;D:\Program Files\JEnv 2. 将 JEnv 添加到环境变量 首先先在自己的电脑上去下载 JAVA 的各个版本&#xff0c;我这里…

二、简单控件

二、简单控件 #mermaid-svg-TR8KwIeb54zOjfmt {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-TR8KwIeb54zOjfmt .error-icon{fill:#552222;}#mermaid-svg-TR8KwIeb54zOjfmt .error-text{fill:#552222;stroke:#55222…

正反转控制电路图

1、倒顺开关正、反转控制电路图 倒顺开关直接接在主电路中&#xff0c;不适合用作大容量的电动机控制&#xff0c;一般用在额定电流10A、功率3kW以下的小容量电动机控制电路中。 2、接触器联锁正、反转控制电路图 接触器联锁正、反转控制电路的主电路中连接了两个接触器KM1和…

从零开始c++精讲:第四篇——模板初阶

文章目录 一、泛型编程二、函数模板2.1函数模板概念2.2函数模板格式2.3函数模板原理2.4函数模板实例化2.5函数模板匹配原则 三、类模板3.1类模板的定义格式3.2类模板的实例化 一、泛型编程 如何实现一个通用的交换函数呢&#xff1f; void Swap(int& left, int& righ…

linux内核源码编译2.6失败

centos7环境 iso选择 https://mirrors.tuna.tsinghua.edu.cn/centos/7/isos/x86_64/CentOS-7-x86_64-DVD-2009.iso 自带qemu&#xff0c;未实测是否可用 选择编译版本2.6 下载地址 遇到的编译错误解决 yum list | grep curses yum install ncurses-devel.x86_64 -y yum i…

最优传输学习及问题总结

文章目录 参考内容lam0.1lam3lam10lam50lam100lam300画图线性规划matlabpython代码 参考内容 https://blog.csdn.net/qq_41129489/article/details/128830589 https://zhuanlan.zhihu.com/p/542379144 我主要想强调的是这个例子的解法存在的一些细节问题 lam0.1 lam 0.1P,…

[java基础揉碎]进制

目录 进制 进制的图示 进制的转换: 第一组 二进制转换成十进制示例 八进制转换成十进制示例 十六进制转换成十进制示例 ​第二组 十进制转换成二进制 十进制转换成八进制 十进制转换成十六进制 第三组 二进制转换成八进制 二进制转换成十六进制 第四组 八进制…

一文读懂JavaScript DOM节点操作(JavaScript DOM节点操作详解)

一、什么是节点 二、节点类型 1、元素节点 2、属性节点 3、文本节点 4、节点类型、名字、值表格 三、通过文档对象方法获取节点 1、通过id属性获取节点 2、通过标签名字获取节点 3、通过类名获取节点 4、通过name属性获取节点 四、通过层级关系获取节点 1、子节点 …

网络安全防护部署所需要注意的几点

顶层设计概念 考虑项目各层次和各要素&#xff0c;追根溯源&#xff0c;统揽全局&#xff0c;在最高层次上寻求问题的解决之道 顶层设计”不是自下而上的“摸着石头过河”&#xff0c;而是自上而下的“系统谋划” 网络安全分为 物理、网络、主机、应用、管理制度 边界最强 接…

【C++】List模拟实现过程中值得注意的点

&#x1f440;樊梓慕&#xff1a;个人主页 &#x1f3a5;个人专栏&#xff1a;《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C》《Linux》《算法》 &#x1f31d;每一个不曾起舞的日子&#xff0c;都是对生命的辜负 目录 前言 1.List迭代器 2.适…

格密码基础:详解LWE问题(2)

目录 一. LWE问题的标准式 二. LWE单向函数与SIS单向函数 2.1 SIS问题的标准型 2.2 SIS与LWE标准型之间的关系 三. LWE问题有多难&#xff1f; 3.1 结论 3.2 归约过程 四. LWE归约性质 五. LWE问题的两个版本 一. LWE问题的标准式 系列文章&#xff1a; 格密码基础&…

Java SE入门及基础(25)

目录 方法带参&#xff08;续第24篇&#xff09; 6.方法参数传递规则 方法传参来自官方的说明 基本数据类型传值案例 基本数据类型传值时传递的是值的拷贝 引用数据类型传值案例 引用数据类型传值时传递的是对象在堆内存上的空间地址 Java SE文章参考:Java SE入门及基础知…

文件操作和IO(1)

认识文件 先来认识狭义上的文件(存储在硬盘(磁盘)上).针对硬盘这种持久化的I/O设备,当我们想要进行数据保存时,往往不是保存成一个整体,而是独立成一个个的单位进行保存,这个独立的单位就被抽象成文件的概念,就类似办公桌上的一份份真实的文件一般. 注意:硬盘 ! 磁盘 磁盘属于…

基于C++11的数据库连接池【C++/数据库/多线程/MySQL】

一、概述 概述&#xff1a;数据库连接池可提前把多个数据库连接建立起来&#xff0c;然后把它放到一个池子里边&#xff0c;就是放到一个容器里边进行维护。这样的话就能够避免数据库连接的频繁的创建和销毁&#xff0c;从而提高程序的效率。线程池其实也是同样的思路&#xf…

Java-NIO篇章(4)——Selector选择器详解

Selector介绍 选择器&#xff08;Selector&#xff09;是什么呢&#xff1f;选择器和通道的关系又是什么&#xff1f;这里详细说明&#xff0c;假设不用选择器&#xff0c;那么一个客户端请求数据传输那就需要建立一个连接&#xff0c;为了避免线程阻塞&#xff0c;那么每个客…

【Linux】进程间通信——system V 共享内存、消息队列、信号量

需要云服务器等云产品来学习Linux的同学可以移步/–>腾讯云<–/官网&#xff0c;轻量型云服务器低至112元/年&#xff0c;优惠多多。&#xff08;联系我有折扣哦&#xff09; 文章目录 写在前面1. 共享内存1.1 共享内存的概念1.2 共享内存的原理1.3 共享内存的使用1.3.1 …

磁盘分区机制

lsblk查看分区 Linux分区 挂载的经典案例 1. 虚拟机增加磁盘 点击这里&#xff0c;看我的这篇文章操作 添加之后&#xff0c;需要重启系统&#xff0c;不重启在系统里看不到新硬盘哦 出来了&#xff0c;但还没有分区 2. 分区 还没有格式化 3. 格式化磁盘 4. 挂载 5. 卸载…

国标GB28181安防视频监控EasyCVR级联后上级平台视频加载慢的原因排查

国标GB28181协议安防视频监控系统EasyCVR视频综合管理平台&#xff0c;采用了开放式的网络结构&#xff0c;可以提供实时远程视频监控、视频录像、录像回放与存储、告警、语音对讲、云台控制、平台级联、磁盘阵列存储、视频集中存储、云存储等丰富的视频能力&#xff0c;同时还…

不要让别人的一言一行变成攻击自己的利器

绝对没错&#xff01;生活中&#xff0c;我们常常会遇到各种各样的人和事&#xff0c;有时候别人的一句话或一种行为可能会让我们感到受伤或不悦。然而&#xff0c;重要的是学会不让这些负面情绪变成攻击自己的利器&#xff0c;而是选择更加积极的处理方式。 以下是一些建议&a…