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

【算法设计与分析】实验——改写二分搜索算法,众数问题(算法分析:主要算法思路),有重复元素的排列问题,整数因子分解问题(算法实现:过程,分析,小结)

说明:博主是大学生,有一门课是算法设计与分析,这是博主记录课程实验报告的内容,题目是老师给的,其他内容和代码均为原创,可以参考学习,转载和搬运需评论吱声并注明出处哦。

要求:2.1和2.2为算法分析题,写出主要的算法即可。2.3-2.5为算法实现题,要求写出:实验名称、实验过程描述(包括程序代码、测试用例、实验结果分析)、实验小结(包括实验过程中遇到的问题及体会)。

2.1

a[0:n-1]是一个有n个元素的数组,k(0<=k<=n-1)是一个非负整数。试设计一个算法将子数组a[0:k]与a[k+1,n-1]换位。要求算法在最坏情况下耗时O(n),且只用到O(1)的辅助空间。

主要算法思路:

原地将a[0:k]转置,然后将a[k+1,n-1]转置,最后将a[0:n-1]转置,得到的结果与换位结果一样。

def reverse(a, m, n):f, g = m, nwhile m < n:a[m], a[n] = a[n], a[m]m += 1n -= 1a = list(map(int, input().split()))
k, n = map(int, input().split())
reverse(a, 0, k)
reverse(a, k + 1, n - 1)
reverse(a, 0, n - 1)
print(a)

2.2 改写二分搜索算法

设a[0 : n - 1]是已排好序的数组。请改写二分搜索算法,使得当搜索元素 x 不在数组中时,返回小于 x 的最大元素的位置 i 和大于 x 的最小元素位置 j。当搜索元素在数组中时,i 和 j 相同,均为 x 在数组中的位置。

主要算法思路:

正常进行二分查找算法。但不同的是函数的返回值为查找的边界,注意在最后一轮查找后左边界会比右边界大一个(如果没找到的话),所以要交换两个变量的位置即可满足题目要求。

def search(x, a):left, right = 1, len(a) - 1while left <= right:mid = (left + right) // 2if a[mid] == x:left, right = mid, midbreakelif a[mid] > x:right = mid - 1else:left = mid + 1return [right, left]a = list(map(int, input().split()))
x = int(input())
print(search(x, a))

2.3 众数问题

问题描述:

给定含有n个元素的多重集合S,每个元素在S中出现的次数称为该元素的重数。多重集S中重数最大的元素称为众数。

例如,S={1,2,2,2,3,5}。

多重集S的众数是2,其重数为3。

编程任务:

对于给定的由n个自然数组成的多重集S,编程计算S 的众数及其重数。

数据输入:

输入数据由文件名为input.txt的文本文件提供。

文件的第1行多重集S中元素个数n;接下来的n 行中,每行有一个自然数。

结果输出:

程序运行结束时,将计算结果输出到文件output.txt中。输出文件有2 行,第1行给

出众数,第2行是重数。

程序代码,测试用例:

with open('input.txt', 'r') as file:Dict = {}lines = file.readlines()[1:]  # 读取所有行,并跳过第一行for line in lines:x = int(line.strip())  # 去除行尾换行符if x in Dict.keys():Dict[x] += 1else:Dict[x] = 1y = max(Dict.values())with open('output.txt', 'w') as file:for key in Dict.keys():if Dict[key] == y:file.write('%d\n' % key)file.write('%d' % y)

题目中给出的两组测试用例分别得到的结果是[1,5]和[3,8],如下图所示

实验结果分析:

本次实验取得了预期的实验效果,正确的从input.txt读取数据并将结果写入output.txt。

本次算法和代码设计思路:用python中的一个字典来存放,字典的键为出现过的数字,字典的值为这个数字出现过的次数。读取文件时采用一行一行的读取,每次读取一个就在字典里面进行记录一次。后用max函数找出出现次数最大的值,最后遍历字典进行输出,并将结果写入文件。

时间复杂度:最好和最坏情况下的时间复杂度都是O(n),n为文件里数据的行数。

空间复杂度:最好情况下为O(1),所有元素都相同,最坏情况下为O(n),所有元素都不相同。

实验小结:

本次实验主要的困难在于对python语言中文件的读写操作不够熟悉;查询相关资料后掌握了python中文件读写操作的相关代码。

实验中还有两个需要注意的细节,一个是读取文件中的第一行代表的是接下来会有几行数据这可能在C或C++编程中有些用处,但python并不需要,所以可以采用切片的方式从第二行开始读取数据;还有一个点是读取到每一行的数据会包含一个换行符,可以用strip()函数去除,不然没办法正常转换为整型。

2.4 有重复元素的排列问题

问题描述:

设 R={ r1 , r2, …, rn }是要进行排列的n个元素。其中元素  r1 , r2 , , rn 可能相同。试设计一个算法,列出R的所有不同排列。

编程任务:

给定n以及待排列的n个元素。计算出这n个元素的所有不同排列。数据输入:

由文件input.txt 提供输入数据。文件的第1 行是元素个数n,1£n£500。接下来的1 行是待排列的n个元素。

结果输出:

程序运行结束时,将计算出的n 个元素的所有不同排列输出到文件output.txt 中。文件最后1 行中的数是排列总数。

程序代码,测试用例:

def sorting(a, flag):if len(a) == len(num):# print(a)res.append(a.copy())returnfor i in range(0, len(num)):# 当前元素被访问过if flag[i]:continue# 剪枝:相同元素需要按顺序加入elif i > 0 and num[i] == num[i - 1] and not flag[i - 1]:continue# 正常添加a.append(num[i])flag[i] = True# 进行递归sorting(a, flag)# 回溯a.pop()flag[i] = Falseres = []
with open('input.txt', 'r') as file:next(file)    # 从第二行开始读num = list(file.readline().strip())# print(num)num.sort()# print(num)sorting([], [False]*len(num))
with open('output.txt', 'w') as file:for j in res:Str = ''.join(j)# print(Str)file.write(Str + '\n')file.write('%d' % len(res))

题目中给出的测试用例得到的结果是2520,如下图所示.代码如下图两段红框中所示

实验结果分析:

本次实验正确实现了列出有重复元素的所有无重复排序。

算法:本次算法涉及到两个重点,一是通过剪枝来避免有重复排序的情况,一是通过回溯来列出所有可能排列。主要是通过递归来生成可能序列,同时用回溯的方法来生成所有可能排列。剪枝是指,如果有两个相同的元素,一定确保第一个元素使用之后再使用第二个元素,也就是相同元素一定有一个出现的顺序,如果没有顺序就会有,[a(第一个),a(第二个)],[a(第二个),a(第一个)]这样重复的情况,所以在排序之前就要先按一定顺序进行排列,确保相同元素放在一起。

程序设计:本题我用python进行实现,传入函数两个数组,一个是生成的排序数组,一个是标志数组,标志该元素是否被引用。

时间复杂度:最坏情况下,每个元素都不相同,递归树有n!个叶子节点,每个叶子节点要O(n)时间复制到结果中,所以是O(n!×n)

实验小结:

我认为本次算法难度较大,我也是在上网查询相关算法题目和思路后理解掌握,编写本题代码。递归思路理解方面难度最大。

算法方面主要学到了用递归和回溯来遍历所有可能结果,用剪枝来避免重复情况。程序涉及方面复习到了用next()函数来跳过第一行,使用join()函数将列表转换为字符串,输出函数中可以使用“+”来添加内容。

25 整数因子分解问题问题描述:

大于1的正整数n可以分解为:n=x1*x2**xm

例如,当n=12 时,共有8 种不同的分解式:

12=12

12=6*2

12=4*3

12=3*4

12=3*2*2

12=2*6

12=2*3*2

12=2*2*3

编程任务:

对于给定的正整数n,编程计算n共有多少种不同的分解式。

数据输入:

由文件input.txt给出输入数据。第一行有1 个正整数n (1≤n≤2000000000)。

结果输出:将计算出的不同的分解式数输出到文件output.txt

程序代码,测试用例:

# 记录所有可能
# num = 1
#  x表示要分解的数
def resolve(x, num):global resultif num == x:result += 1returnif num > x:returnfor i in range(x, 1, -1):if x % i == 0 and num * i <= x:num *= i# 递归resolve(x, num)# 回溯num %= iresult = 0
n = int(input())
resolve(n, 1)
print(result)

题目中给出的测试用例得到的结果是2520,如下图所示.代码如下图两段红框中所示

实验结果分析:

本次实验正确实现了列出有重复元素的所有无重复排序。

算法:本次算法涉及到两个重点,一是通过剪枝来避免有重复排序的情况,一是通过回溯来列出所有可能排列。主要是通过递归来生成可能序列,同时用回溯的方法来生成所有可能排列。剪枝是指,如果有两个相同的元素,一定确保第一个元素使用之后再使用第二个元素,也就是相同元素一定有一个出现的顺序,如果没有顺序就会有,[a(第一个),a(第二个)],[a(第二个),a(第一个)]这样重复的情况,所以在排序之前就要先按一定顺序进行排列,确保相同元素放在一起。

程序设计:本题我用python进行实现,传入函数两个数组,一个是生成的排序数组,一个是标志数组,标志该元素是否被引用。

时间复杂度:最坏情况下,每个元素都不相同,递归树有n!个叶子节点,每个叶子节点要O(n)时间复制到结果中,所以是O(n!×n)

实验小结:

我认为本次算法难度较大,我也是在上网查询相关算法题目和思路后理解掌握,编写本题代码。递归思路理解方面难度最大。

算法方面主要学到了用递归和回溯来遍历所有可能结果,用剪枝来避免重复情况。程序涉及方面复习到了用next()函数来跳过第一行,使用join()函数将列表转换为字符串,输出函数中可以使用“+”来添加内容。

相关文章:

  • Java递归编程中的StackOverflowError问题分析与解决方案
  • 从“敲窗“到“推门“:用Requests库解锁网络数据的“读心术“——Python爬虫入门实战指南
  • SCAU8640--希尔排序
  • 通俗理解“高内聚,低耦合”
  • ipfs下载和安装(windows)
  • 50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Sound Board(音响控制面板)
  • 回文字符串
  • 【烧脑算法】不定长滑动窗口:从动态调整到精准匹配以灵活特性实现高效破题
  • 小目标检测:YOLOV7改进之双坐标注意力(DCA)
  • YOLOv10改进|爆改模型|涨点|C2F引入空间和通道注意力模块暴力涨点(附代码+修改教程)
  • 【C盘瘦身】给DevEco Studio中HarmonyOSEmulator(鸿蒙模拟器)换个地方,一键移动给C盘瘦身
  • FreeRTOS实时操作系统学习笔记
  • 文件索引:数组、二叉树、二叉排序树、平衡树、红黑树、B树、B+树
  • 【LeetCode 热题100】BFS/DFS 实战:岛屿数量 腐烂的橘子(力扣200 / 994 )(Go语言版)
  • vscode中的markdown表格列宽
  • js的时间循环的讲解
  • Y1——链式前向星
  • mybatisplus的总结
  • 计算机网络 : 应用层自定义协议与序列化
  • 2894. 分类求和并作差
  • 学做网站设计/搜索推广
  • 如何做好专业类网站/自己开平台怎么弄啊
  • 手机怎么打开微信网站/快速排名优化公司
  • 如何用dw做php网站代码/武汉百度信息流广告
  • 教学网站在线自测功能怎么做/360关键词排名推广
  • 淄博哪有培训做网站的/百度网络推广怎么收费