山东大学程序设计思维与实践作业-Week2

山东大学程序设计思维与实践作业-Week2

仅作参考,切莫照搬,有查重

  • 代码题目

    • 1
    git clone https://github.com/W1412X/cxsj_sduoj.git
    
    • 2
      前往https://github.com/W1412X/cxsj_sduoj.git获取
  • 简答题目

1

f1

  • set使用红黑树实现,对应的insert()函数的时间复杂度是 log(n),所以整体的时间复杂度为:
    O ( n log ⁡ ( n ) ) O(n\log(n)) O(nlog(n))

f2

  • O(n)

2

  • sort函数的时间复杂度为O(nlogn),所以整体的时间复杂度为
    O ( n log ⁡ ( n ) ) O(n\log(n)) O(nlog(n))

3

  • sort函数中的迭代器要求是随机访问迭代器,但是list由双向链表实现,不支持随机存取

4

  • std:find
    -std:find函数是顺序查找,对n个元素,其查找的时间复杂度为 O ( n ) O(n) O(n)
  • map,set
    • 两者均基于红黑树实现,两者的find的函数相比于std:find有更好的性能,时间复杂度为
      log ⁡ ( n ) \log(n) log(n)

5

  • 首先需要自定义比较函数来改变set的自动排序顺序
  • 之后直接使用迭代器输出即可
#include<bits/stdc++.h>
using namespace std;
class Compare{
public:
    bool operator()(const int&a,const int&b)const{
        return a>b;
    }
};
int main(){
    set<int,Compare>a;
    for(int i=0;i<10;i++){
        a.insert(i);
    }
    set<int>::iterator it=a.begin();
    while(it!=a.end()){
        cout<<*it<<' ';
        it++;
    }
}

6

  • 解法

    • 使用数组与优先队列解决此问题,数组的每个元素均为一个优先队列,用于模拟对应的桶
    • 将各个桶的元素放到对应的优先队列中
    • 最后遍历数组,从优先队列弹出元素
  • 时间复杂度
    O ( n + m log ⁡ ( n ) ) O(n+m\log(n)) O(n+mlog(n))

    • 解释
      • 创建大小为 n+1 的 vector为O(n)
      • 对于 m 次输入操作,每次插入操作的时间复杂度为 O(log n),总为O(mlog(n))
      • 对于输出,pop的时间复杂度也为log(n),则对应的复杂度为O(mlog(n))
  • 代码

#include<iostream>
#include<vector>
#include<algorithm>
#include<queue>
using namespace std;
int main(){
    int n,m;
    cin>>n>>m;
    vector<priority_queue<int,vector<int>,greater<int>>>a(n+1,priority_queue<int,vector<int>,greater<int>>());//创建对应的优先级队列数组
    int p,q;
    for(int i=0;i<m;i++){
        cin>>p>>q;
        a[q].push(p);//将q桶的元素放入a[q]对应的堆中
    }
    for(int i=1;i<a.size();i++){
        while(a[i].size()>0){//对每一个桶,输出
            cout<<a[i].top()<<' ';
            a[i].pop();
        }
        cout<<endl;
    }
}

7

  • 解法

    • 利用set的特性,在插入元素时自动按照字典序排列
    • 分别定义操作1-3
    • 根据每一行首个字符的不同进行操作
  • 时间复杂度

    O ( n log ⁡ ( n ) ) O(n\log(n)) O(nlog(n))

    • 解释
    • 查找,插入,删除操作的时间复杂度为log(n)
    • 按照字典序输出的操作使用迭代器完成,时间复杂度为O(n)
  • 代码

#include<bits/stdc++.h>
using namespace std;
set<string>s;//利用set可以自动按照字典序排列
void op1(string str){//查找单词的函数
    if(s.find(str)!=s.end()){
        cout<<"found"<<endl;
    }else{
        cout<<"write"<<endl;
        s.insert(str);
    }
}
void op2(string str){//删除单词的操作
    if(s.find(str)!=s.end()){
        cout<<"erased"<<endl;
        s.erase(str);
    }else{
        cout<<"not found"<<endl;
    }
}
void op3(){//按字典序通读笔记本
    
    for(const auto&e:s){
        cout<<e<<' ';
    }
    cout<<endl;
}
int main(){
    int n;
    cin>>n;
    char op;
    string opstr;
    for(int i=0;i<n;i++){
        cin>>op;
        switch (op)
        {
        case '1'://查找
            cin>>opstr;
            op1(opstr);
            break;
        case '2'://删除
            cin>>opstr;
            op2(opstr);
            break;
        case '3'://通读
            op3();
            break;
        default:
            break;
        }
    }
}
  • 10
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值