电子科技大学链时代工作室招新题C语言部分---题号H

本文主要是介绍电子科技大学链时代工作室招新题C语言部分---题号H,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 题目

最有操作的一道题,有利于对贪心算法有个初步了解。

  这道题的开篇向我们介绍了一个叫汉明距离的概念。

汉明距离指的就是两个相同长度的字符串的不同字符的个数。

例如,abc和acd,b与c不同,c与d不同,所以这两个字符串的汉明距离就是2。

 这道题就要求我们找到一个字符串,使得该字符串与已知的两个字符串的汉明距离相同,并且该字符串在字典序上要尽可能地小。(字典序上最小,就是在符合条件的字符串中,用strcmp比较出来的最小的那个)。

题目言简意赅,但却是最考验思路的一道题。

输入

第一行输入一个整数T(1 <= T <= 100),表示接下来要输入几个案例。

接下来输入T个测试案例,对每个案例,输入两行长度相同的字符串(单个字符串长度不超过10^{4},字符串总长度不超过10^{6})。

输出

输出T行,每行为其对应案例的所求字符串。


2. 第一版解法

前面说过,这道题十分考验思路,我们第一版时,我们没有找到很好的办法,解法比较暴力,思路也比较混乱。

2.1 思路

1. 我们要比较目标字符串与两个原字符串的汉明距离,那我们势必要先生成一个目标字符串,然后再根据汉明距离的差异来调整目标字符串。

2. 既然要目标字符串在字典序上最小,那我们一开始生成的目标字符串就尽可能小,并且只在必要时调整,这样才能保证得到我们想要的结果。

3. 所以,我们决定将目标字符串初始化为每个元素都是'a',并从后往前调整(越靠前的字符权重越高,所以优先调整靠后的字符)。

4. 初始时,两个原字符串与目标字符串的汉明距离分别就是两个字符串中不等于'a'的字符的数量,然后根据不同的情况,我们对字符做不同的调整。

2.2 代码

#include <stdio.h>
#include <string.h>int main()
{int T = 0;scanf("%d", &T);char arr1[10001] = {0};char arr2[10001] = {0};char ret[101][10001] = {0};for(int k = 0; k < T; k++){char* tmp = &ret[k][0];scanf("%s", arr1);scanf("%s", arr2);int len = strlen(arr1);int hm1 = len, hm2 = len;//记录汉明数tmp[len] = '\0';for(int i = 0; i < len; i++){tmp[i] = 'a';if(arr1[i] == 'a')hm1--;if(arr2[i] == 'a')hm2--;}int op = len - 1;while(hm1 != hm2){if(arr1[op] == arr2[op])//两个相同,改变无意义且会增大字符{op--;}else if(hm1 - hm2 >= 2&&'a' == arr2[op])//汉明数相差超过二{tmp[op] = arr1[op];hm1--;hm2++;op--;}else if(hm2 - hm1 >= 2&&'a' == arr1[op]){tmp[op] = arr2[op];hm2--;hm1++;op--;}else if(op >= 1&&hm1 - hm2 == 2&&'a' == arr2[op - 1]&&arr1[op - 1] == 'b')//下一次能改两,且刚好差两{tmp[op - 1] = arr1[op - 1];hm1--;hm2++;op--;}else if(op >= 1&&hm2 - hm1 == 2&&'a' == arr1[op - 1]&&arr2[op - 1] == 'b'){tmp[op - 1] = arr2[op - 1];hm2--;hm1++;op--;}else if(hm1 > hm2&&'a' != arr1[op]&&'a' != arr2[op]){tmp[op] = arr1[op];hm1--;op--;}else if(hm1 < hm2&&'a' != arr1[op]&&'a' != arr2[op]){tmp[op] = arr2[op];hm2--;op--;}else if(hm1 - hm2 == 1&&'a' == arr2[op]){for(char j = 'a'; j <= 'z'; j++){if(arr1[op] != j&&arr2[op] != j){tmp[op] = j;break;}}hm2++;op--;}else if(hm2 - hm1 == 1&&'a' == arr1[op]){for(char j = 'a'; j <= 'z'; j++){if(arr1[op] != j&&arr2[op] != j){tmp[op] = j;break;}}hm1++;op--;}else{op--;}}}for(int i = 0; i < T; i++){printf("Case %d: %s\n", i + 1, ret[i]);}return 0;
}

2.3 总结

这一版中,我们考虑了许多种情况,可以看到if和else语句写了一长串,以告诉程序在遇到不同情况时该怎么做。

感兴趣的小伙伴可以仔细看看第4和5个分支,这种情况是最难想到的。比如输入aab和bbc时,正确答案应该是aba,而没有这两个分支时会输出acc。

虽然这段代码看起来天衣无缝,万无一失,能讨论的情况都考虑了(我至今都想不通到底是什么测试用例会导致其失败)。但是在oj上跑时,始终会提示“wrong answer on test 2!”。

还记得我们总结的经验教训吗,当你的代码开始用if语句处理特殊输入时,就要开始考虑改改了。

用if和else语句来告诉计算机该怎么处理不同的情况,这样的算法是远比不上一个通用的算法的。

3. 最终版解法

这一版解法是我在走投无路的情况下向大佬求教,得到耐心指点之后才搞定的。

3.1 思路

这一版的思路将我们之前的做法全盘推翻了,这一次我们采用从前向后操作的顺序,并且不是调整出目标字符串,而是直接生成出目标字符串。

在写出第一版之前,我也考虑过从前往后生成目标字符串的写法。但是,这么做会有两个问题:

1. 目标字符串逐步生成,那么我在生成过程中如何知道当前位置该放什么,才能保证之后有办法使得目标字符串与两个原字符串的汉明距离相同(之后称这个目标为“汉明距离差值为0”)?

2. 如何确保我生成的字符串是字典序最小的?

为了解决这两个问题,我们需要深入地去挖掘一些隐藏的数学关系:

导致目标字符串与两个原字符串的汉明距离不同的,只能是两个原字符串字符不同的位置。也就是说,在目标字符串生成到某个位置时,它与两个原字符串的汉明距离的差值一定小于未生成部分,两个原字符串的汉明距离。我们只需要确保这一点成立,就可以将尽可能小的字符放在该位置。

 好像这段话不长,但我就是看不懂?

没关系,好好理解一下,很快就会有一种恍然大悟的感觉。

那么,为什么要从前往后操作呢?因为随着我操作的进行,需要满足的条件会越来越苛刻。前面的字符更高贵,苦了后面也不能苦前面。

这个思路的妙处在于,它要求要满足的条件是一个必要条件,将条件限制到最苛刻的情况(必要条件不满足,结果一定不成立),我再考虑放置较大的字符,这样就可以确保我得到的是最小字符串。这也就是贪心算法的基本思想。

而且,随着我操作的进行,两个原字符串的汉明距离会逐渐减小,那么目标字符串与两个原字符串的汉明距离之差也会逐渐减小,直到最后归于零。

3.2 代码

#include <stdio.h>
#include <string.h>int HanMingDistance(char* str1, char* str2, int len)
{int distance = 0;for(int i = 0; i < len; i++){if(str1[i] != str2[i])distance++;}return distance;
}int main()
{int T = 0;scanf("%d", &T);char arr1[10001] = {0};char arr2[10001] = {0};char ret[101][10001] = {0};for(int k = 0; k < T; k++){char* tmp = &ret[k][0];scanf("%s", arr1);scanf("%s", arr2);int len = strlen(arr1);tmp[len] = '\0';int dice = 0;//输入的两个字符串当前位置起的汉明距离int hm1 = 0, hm2 = 0;for(int i = 0; i < len; i++){if(arr1[i] == arr2[i]){tmp[i] = 'a';continue;}dice = HanMingDistance(arr1+i, arr2+i, len-i);int tem1 = 0, tem2 = 0;for(char j = 'a'; j <= 'z'; j++){if(arr1[i] != j)tem1 = 1;elsetem1 = 0;if(arr2[i] != j)tem2 = 1;elsetem2 = 0;if(hm1 + tem1 - hm2 - tem2 < dice&&hm2 + tem2 - hm1 -tem1 < dice){hm1 = hm1 + tem1;hm2 = hm2 + tem2;tmp[i] = j;break;}}}}for(int i = 0; i < T; i++){printf("Case %d: %s\n", i + 1, ret[i]);}return 0;
}

3.3 总结

这一版仅65行,差不多只有第一版的一半,但是却能顺利通过oj的考验。

算法,真是一个神奇的东西,通过这道题,我们也能看见研究算法的重要性。

这篇关于电子科技大学链时代工作室招新题C语言部分---题号H的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/834377

相关文章

C语言中位操作的实际应用举例

《C语言中位操作的实际应用举例》:本文主要介绍C语言中位操作的实际应用,总结了位操作的使用场景,并指出了需要注意的问题,如可读性、平台依赖性和溢出风险,文中通过代码介绍的非常详细,需要的朋友可以参... 目录1. 嵌入式系统与硬件寄存器操作2. 网络协议解析3. 图像处理与颜色编码4. 高效处理布尔标志集合

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服

C 语言中enum枚举的定义和使用小结

《C语言中enum枚举的定义和使用小结》在C语言里,enum(枚举)是一种用户自定义的数据类型,它能够让你创建一组具名的整数常量,下面我会从定义、使用、特性等方面详细介绍enum,感兴趣的朋友一起看... 目录1、引言2、基本定义3、定义枚举变量4、自定义枚举常量的值5、枚举与switch语句结合使用6、枚

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

Go 语言中的select语句详解及工作原理

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语... 目录Go 语言中的 select 是做什么的基本功能语法工作原理示例示例 1:监听多个通道示例 2:带

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T

C语言中的数据类型强制转换

《C语言中的数据类型强制转换》:本文主要介绍C语言中的数据类型强制转换方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C语言数据类型强制转换自动转换强制转换类型总结C语言数据类型强制转换强制类型转换:是通过类型转换运算来实现的,主要的数据类型转换分为自动转换

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时