【算法】队列——解密QQ号

2023-11-01

 新学期开始了,小哈是小哼的新同桌(小哈是个小美女哦~),小哼向小哈询问QQ号,小哈当然不会直接告诉小哼啦,原因嘛你懂的。所以小哈给了小哼一串加密过的数字,同时小哈也告诉了小哼解密规则。规则是这样的:首先将第1个数删除,紧接着将第2个数放到这串数的末尾,再将第3个数删除并将第4个数再放到这串数的末尾,再将第5个数删除……直到剩下最后一个数,将最后一个数也删除。按照刚才删除的顺序,把这些删除的数连在一起就是小哈的QQ啦。现在你来帮帮小哼吧。小哈给小哼加密过的一串数是“6 3 1 75 8 9 2 4”。


       OK,现在轮到你动手的时候了。快去找出9张便签或小纸片,将“6 3 1 75 8 9 2 4”这9个数分别写在9张便签上,模拟一下解密过程。如果你没有理解错解密规则的话,解密后小哈的QQ号应该是“6 1 5 94 7 2 8 3”。


       其实解密的过程就像是将这些数“排队”。每次从最前面拿两个,第1个扔掉,第2个放到尾部。具体过程是这样的:刚开始这串数是“6 3 1 75 8 9 2 4”,首先删除6并将3放到这串数的末尾,这串数更新为“1 7 5 89 2 4 3”。接下来删除1并将7放到末尾,即更新为“5 8 9 24 3 7”。再删除5并将8放到末尾即“9 2 4 3 7 8”,删除9并将2放到末尾即“4 3 7 8 2”,删除4并将3放到末尾即“7 8 2 3”,删除7并将8放到末尾即“2 3 8”,删除2并将3放到末尾即“8 3”,删除8并将3放到末尾即“3”,最后删除3。因此被删除的顺序是“6 1 5 9 4 7 2 8 3”,这就是小哈的QQ号码了,你可以加她试试看^_^。


       既然现在已经搞清楚了解密法则,不妨自己尝试一下去编程,我相信你一定可以写出来的。


       首先需要一个数组来存储这一串数即intq[101]。并初始化这个数组即intq[101]={0,6,3,1,7,5,8,9,2,4};(此处初始化是我多写了一个0,用来填充q[0],因为我比较喜欢从q[1]开始用,对数组初始化不是很理解的同学可以去看下我的上一本书《啊哈C!思考快你一步》)接下来就是模拟解密的过程了。


       解密的第一步是将第一个数删除,你可以想一下如何在数组中删除一个数呢?最简单的方法是将所有后面的数都往前面挪动一位,将前面的数覆盖。就好比我们在排队买票,最前面的人买好离开了,后面所有的人就需要全部向前面走一步,补上之前的空位,但是这样的做法很耗费时间。


       在这里,我将引入两个整型变量head和tail。head用来记录队列的队首(即第一位),tail用来记录队列的队尾(即最后一位)的下一个位置。你可能会问为什么tail不直接记录队尾,却要记录队尾的下一个位置呢?这是因为当队列当中只剩下一个元素时,队首和队尾重合会带来一些麻烦。我们这里规定队首和队尾重合时,队列为空。


       现在有9个数,9个数全部放入队列之后head=1;tail=10;此时head和tail之间的数就是目前队列中“有效”的数。如果要删除一个数的话,就将head++就OK了,这样仍然可以保持head和tail之间的数为目前队列中“有效”的数。这样做虽然浪费了一个空间,却节省了大量的时间,这是非常划算的。新增加一个数也很简单,把需要增加的数放到队尾即q[tail]之后再tail++就欧克啦。


       我们来小结一下,在队首删除一个数的操作是head++;

       在队尾增加一个数(假设这个数是x)的操作是q[tail]=x;tail++;
       整个解密过程,请看下面这个霸气外漏的图。
       最后的输出就是6 1 5 94 7 2 8 3,代码实现如下。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
#include <stdio.h>
int  main()
{
     int  q[102]={0,6,3,1,7,5,8,9,2,4},head,tail;
     int  i;
     //初始化队列
     head=1;
     tail=10;  //队列中已经有9个元素了,tail执向的队尾的后一个位置
     while (head<tail)  //当队列不为空的时候执行循环
     {
         //打印队首并将队首出队
         printf ( "%d " ,q[head]);
         head++;
                                                       
         //先将新队首的数添加到队尾
         q[tail]=q[head];
         tail++;
         //再将队首出队
         head++;
     }
                                                         
     getchar (); getchar ();
     return  0;
}


       怎么样上面的代码运行成功没有?现在我们再来总结一下队列的概念。队列是一种特殊的线性结构,它只允许在队列的首部(head)进行删除操作称之为“出队”,而在队列的尾部(tail)进行插入操作称之为“入队”。当队列中没有元素时(即head==tail),称为空队列。在我们的日常生活中有很多情况都符合队列的特性。比如我们之前提到过的买票,每个排队买票的窗口就是一个队列。在这个队列当中,新来的人总是站在队列的最后面,来的越早的人越靠前也就越早能买到票,就是先来的人先服务,我们称为“先进先出”(First InFirst Out,FIFO)原则。
       队列将是我们今后学习广度优先搜索以及队列优化的Bellman-Ford最短路算法的核心数据结构。所以现在将队列的三个基本元素(一个数组,两个变量)封装为一个结构体类型,如下。
1
2
3
4
5
6
struct  queue
{
     int  data[100]; //队列的主体,用来存储内容
     int  head; //队首
     int  tail; //队尾
};


       上面我们定义了一个结构体类型,我们通常将其放在main函数的外面,请注意结构体的定义末尾有个;号。struct是结构体的关键字,queue是我们为这个结构体起的名字。这个结构体有三个成员分别是:整型数组data、整型head和整型tail。这样我们就可以把这三个部分放在一起作为一个整体来对待。你可以这么理解:我们定义了一个新的数据类型,这个新类型非常强大,用这个新类型定义出的每一个变量可以同时存储一个整型数组和两个整数。
       有了新的结构体类型,如何定义结构体变量呢?很简单,这与我们之前定义变量的方式是一样,如下。
1
struct  queue q;


       请注意struct queue需要整体使用,不能直接写queue q;这样我们就定义了一个结构体变量q。这个结构体变量q就可以满足队列的所有操作了。那又该如何访问结构体变量的内部成员呢?可以使用.号,它叫做成员运算符或者点号运算符,如下:
1
2
3
q.head=1;
q.tail=1;
scanf ( "%d" ,&q.data[q.tail]);


       好了,下面这段代码就是使用结构体来实现的队列操作。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#include <stdio.h>
struct  queue
{
     int  data[100]; //队列的主体,用来存储内容
     int  head; //队首
     int  tail; //队尾
};
                                           
int  main()
{
     struct  queue q;
     int  i;
     //初始化队列
     q.head=1;
     q.tail=1;
     for (i=1;i<=9;i++)
     {
         //依次向队列插入9个数
         scanf ( "%d" ,&q.data[q.tail]);
         q.tail++;
     }
                                             
     while (q.head<q.tail)  //当队列不为空的时候执行循环
     {
         //打印队首并将队首出队
         printf ( "%d " ,q.data[q.head]);
         q.head++;
                                                  
         //先将新队首的数添加到队尾
         q.data[q.tail]=q.data[q.head];
         q.tail++;
         //再将队首出队
         q.head++;
     }
                                             
     getchar (); getchar ();
     return  0;
}


       上面的这种写法看起来虽然冗余了一些,但是可以加强你对队列这个算法的理解。C++的STL库已经有队列的实现,有兴趣的同学可以参看相关材料。队列的起源已经无法追溯。在还没有数字计算机之前,数学应用中就已经有对队列的记载了。我们生活中队列的例子也比比皆是,比如排队买票,又或者吃饭时候用来排队等候的叫号机,又或者拨打银行客服选择人工服务时,每次都会被提示“客服人员正忙,请耐心等待”,因为客服人员太少了,拨打电话的客户需要按照打进的时间顺序进行等候等等。这里表扬一下肯德基的宅急送,没有做广告的嫌疑啊,每次一打就通,基本不需要等待。但是我每次打银行的客服(具体是哪家银行就不点名了)基本都要等待很长时间,总是告诉我“正在转接,请稍后”,嘟嘟嘟三声后就变成“客服正忙,请耐心等待!”然后就给我放很难听的曲子。看来钱在谁那里谁就是老大啊……

http://bbs.ahalei.com/thread-4489-1-1.html
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【算法】队列——解密QQ号 的相关文章

  • 云计算概念详解

    1 云计算的定义 1 云计算是一种能够通过网络以便利的按需的方式获取云计算资源 网络 服务器 存储 应用和服务 的模式 2 这些资源来自一个共享的 可配置的资源池 并能够快速获取和释放 提供资源的网络称为云 3 云模式能够提高可用性 4 云

随机推荐

  • IDEA下载与安装,保姆级教程

    这里写自定义目录标题 1 搜索idea 2 选择官方网站 3 官网进入下载页面 4 版本选择问题 5 Ultimate和Community对比 6 下载 7 安装 1 搜索idea 2 选择官方网站 以前idea的官网后面有官网俩字 现在没
  • Open NMT-py 玩具模型使用说明

    前排提示 本文仅适合纯萌新玩家 算是官方指南的补档 大佬请直接关闭网页 避免浪费时间 截至到2023 3 15 最新的OpenNMT py环境要求 Python gt 3 7 PyTorch gt 1 9 0 如果是老版本的OpenNMT
  • https网络编程——使用openssl库自建根证书

    参考 如何自建根证书 使用openssl库自建根证书带图详解 地址 https qingmu blog csdn net article details 108217572 spm 1001 2014 3001 5502 目录 根证书的普通
  • spring boot最新教程(三):Spring Boot整合JdbcTemplate以及事务管理

    一 JdbcTemplate的使用 Spring对数据库的操作在jdbc上面做了深层次的封装 使用spring的注入功能 可以把DataSource注册到JdbcTemplate之中 JdbcTemplate 是在JDBC API基础上提供
  • python:日期时间处理

    目录 一 time模块 二 秒转换为时分秒 三 计算前后几天的日期 一 time模块 1 time strftime format t 格式 说明 a 本地 locale 简化星期名称 A 本地完整星期名称 b 本地简化月份名称 B 本地完
  • 【STM32Cube】学习笔记(六):DHT11温湿度传感器

    文章目录 摘要 一 简介 1 DHT11数字温湿度传感器 2 DHT11性能参数 2 DHT11数据结构 2 DHT11传输时序 二 硬件电路设计 1 模块内部电路 2 与单片机相连接电路 三 软件设计 1 CubeMX配置 2 CubeI
  • Spark Streaming(组件、updateStateByKey、Windows)总结

    Spark Streaming 1 SparkStreaming 是什么 2 实时计算框架对比 3 Spark Streaming组件 4 Spark Streaming 编码实战 无状态 4 1 Spark Streaming编码步骤 4
  • MiniMeters for Mac - 独立音频计量软件,创意音乐的最佳伙伴

    MiniMeters for Mac是一款专为Mac用户设计的音频计量软件 它提供了一套功能强大 直观易用的工具 帮助你更好地理解和处理音频 这款软件不仅具备高度的专业性 同时也极具创新性 它的出现将彻底改变你对音频处理的认知 安装 Min
  • 神经网络的几点思考

    2022 04 09 1 小卷积核和大卷积核有没有可能组合使用效果更好 比如在目标检测网络 人脸识别网络 2 人脸识别中共享卷积有效吗 共享卷积和局部卷积有没有可能组合使用效果会更好 人脸识别 人脸属性 人脸关键点 活体检测应该都可以用局部
  • PGCM-PostgreSQL备份工具 pgBackRest使用

    更多精彩内容 请登录 ke sandata com cn 前言 PGCM pgBackRest是一款开源的备份还原工具 目标旨在为备份和还原提供可靠易用的备份 特性 并行备份和还原 备份操作期间压缩通常是其瓶颈所在 pgBackRest通过
  • Zotero如何在word中引用跳转到参考文献/建立超链接

    省流目录 文章目录 问题 如标题 解决方案 1 打开word gt 视图 gt 宏 gt 点击 选查看宏 2 创建宏 3 将代码全部替换为下面这个 4 Ctrl s保存 左下角重命名为ZoteroLinkCitation 关闭页面 5 查看
  • clickhouse-server.service: main process exited, code=exited, status=232/ADDRESS_FA

    使用 systemctl start clickhouse server 启动失败 报错信息如下 root hantest mysql systemctl status clickhouse server clickhouse server
  • 密码学-传统加密技术

    传统加密技术 对称密码模型 明文 plaintext 加密算法 encryption 密钥 key 密文 cipher 解密算法 decryption 传统密码的要求 加密算法足够强 密钥安全 采用对称密码 首要的安全问题是密钥的保密性 密
  • web前端 --- 常见页面标签和语义化标签

    1 HTML5 1 含义 HTML Hypertext Markup Language 超文本标记语言 是一种用于创建网页的标准标记语言 您可以使用 HTML 来建立自己的 WEB 站点 HTML 运行在浏览器上 由浏览器来解析 声明为 H
  • Modbus 与 RS485 的区别与联系

    目前道长入坑了一家智能家居公司 以后会分享记录一些智能家居相关的知识 如果有问题 希望小伙伴交流指正 一 RS 485 1 1 RS 485 来源 RS485是美国电子工业协会 EIA 在1983年批准了一个新的平衡传输标准 balance
  • java线程状态

    1 新建 NEW 新创建了一个线程对象 2 可运行 RUNNABLE 调用了对象的start 方法 位于可运行线程池中 等待被线程调度选中 获取cpu 的使用权 3 运行 RUNNING 可运行状态 runnable 的线程获得了cpu 时
  • Django-全局配置文件&路由配置文件(二)

    一 全局配置文件 settings py 注意文件中注释 Django settings for mysite project Generated by django admin startproject using Django 2 2
  • 大数据Flink简介与架构剖析并搭建基础运行环境

    文章目录 前言 Flink 简介 Flink 集群剖析 Flink应用场景 Flink基础运行环境搭建 Docker安装 docker compose文件编写 创建并运行容器 访问Flink web界面 前言 前面我们分别介绍了大数据计算框
  • spring

    1 Spring简介 1 1 Spring概述 官网地址 https spring io Spring 是最受欢迎的企业级 Java 应用程序开发框架 数以百万的来自世界各地的开发人员使用 Spring 框架来创建性能好 易于测试 可重用的
  • 【算法】队列——解密QQ号

    新学期开始了 小哈是小哼的新同桌 小哈是个小美女哦 小哼向小哈询问QQ号 小哈当然不会直接告诉小哼啦 原因嘛你懂的 所以小哈给了小哼一串加密过的数字 同时小哈也告诉了小哼解密规则 规则是这样的 首先将第1个数删除 紧接着将第2个数放到这串数