【无标题】百花2022年第十三周数据结构作业

注满水分的一次作业。。。。

A二叉排序树 - 文本输出

题目描述

给定一个序列,使用该序列生成二叉排序树(也叫二叉搜索树,BST),然后以本题规定方法输出该二叉排序树。
例:
给定一个序列:43 25 29 67 17 88 54 47 35 62
以第一个数字(43)为根节点,然后将后面的数字依输入次序逐个添加至该树中,得到一个二叉排序树,如下图所示。
在这里插入图片描述
然后先序遍历上面这个树,并按行输出数字。
其中每个子节点的输出前,需要相较于其父节点前多四个普通空格。
当某个节点为叶子节点(即无子节点),则该节点的左右叶子节点均不用输出。
而当某个节点仅有左叶子节点或右叶子节点时,另一个空缺的子节点用#占位。

以该图为例,其最终输出结果为:
在这里插入图片描述

输入格式
第一行为正整数n,表示接下来将输入的节点数量。(n<500)
第二行为n个正整数,每个数字以空格分隔(以第一个数字为根节点)
输出格式
以题目描述中的方法输出得到的二叉排序树。
以第一个数字为根节点,然后将后面的数字依输入次序逐个添加至该树中,得到一个二叉排序树。
然后先序遍历该树,并按行输出数字。
其中每个子节点的输出前,需要相较于其父节点前多四个普通空格。
当某个节点为叶子节点(即无子节点),则该节点的左右叶子节点均不用输出。
而当某个节点仅有左叶子节点或右叶子节点时,另一个空缺的子节点用#占位。

输入样例

10
43 25 29 67 17 88 54 47 35 62

输出样例

在这里插入图片描述

code:

#include<bits/stdc++.h>
using namespace std;
#define isw {ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);}
typedef pair<int,int>P;
#define inf 0x3f3f3f3f
#define maxn 100005
#define int long long 
struct bstnode{
    int data;
    bstnode *rchild,*lchild;
};
typedef bstnode* bst;
void init(bstnode* &t){
    t=NULL;
}
int Insert(bst &t,int x){//插入元素x
    bst p=(bst)malloc(sizeof(bstnode));
            p->data=x;
            p->lchild=NULL;
            p->rchild=NULL;
    if(!t){
        t=p;
        return 1;
    }
     else if(t->data>x){
        if(!t){
            t->lchild=p;
        }
        else{
            Insert(t->lchild,x);
        }
    }
   else{
        if(!t){
            t->rchild=p;
        }
        else{
            Insert(t->rchild,x);
        }
    }
    return 1;
}
void pretral(bst t,int depth){
    if(!t){
        for(int i=0;i<depth;i++){
            cout<<"    ";
        }
        cout<<"#"<<endl;
    }
    else{
        for(int i=0;i<depth;i++){
            cout<<"    ";
        }
        cout<<t->data<<endl;
        //cout<<t->data<<" "<<depth<<endl;
        // for(int i=0;i<depth;i++){
        //     cout<<"    ";
        // }
       if(!t->lchild&&!t->rchild){
           //cout<<"#"<<endl;
           return;
       }
       else{
           pretral(t->lchild,depth+1);
           pretral(t->rchild,depth+1);
       }
    }
}
signed main(){
    int n;
    cin>>n;
    bst t;
    init(t);
    for(int i=0;i<n;i++){
        int x;
        cin>>x;
        Insert(t,x);
    }
    pretral(t,0);
}

B销售排行榜

题目描述
你的任务是帮助淘宝网店店长整理销售数据,根据累计的销售记录,将所有商品按销售数量降序排列。
输入格式
输入包括多行数据(行数小于100000),每行数据包括4个信息,分别是商品名称、销售数量、单价、成交日期
商品名称由小写字母组成,且不超过100个字符,销售数量和单价都是正整数,且小于10000
输出格式
输出包括多行数据,将所有在输入中出现的商品按销售数量降序排列,每行数据包括3个信息,分别是商品名称、销售数量、销售额,如果两种商品销售数量一样,则按商品的字母顺序升序排列

输入样例

apple 1 20 2014-4-2
basketball 1 20 2014-4-2
computer 1 20 2014-4-2
shoe 1 20 2014-4-2
tv 1 20 2014-4-2
apple 1 18 2014-4-3

输出样例

apple 2 38
basketball 1 20
computer 1 20
shoe 1 20
tv 1 20

思路

按照题意统计然后排序就行

code

#include<bits/stdc++.h>
using namespace std;
#define isw {ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);}
typedef pair<int,int>P;
#define inf 0x3f3f3f3f
#define maxn 100005
#define int long long 
struct goods{
    string name;
    int num,price;
};
bool cmp(goods a,goods b){
    if(a.num!=b.num)return a.num>b.num;
    return a.name<b.name;
}
map<string,P>mp;
goods ans[100005];
int main(){
    string s;
    string a;
    int x,y;
    int p=0;
    while(cin>>s>>x>>y>>a){
        if(s=="#")break;
        if(mp.find(s)==mp.end()){
            mp[s]={x,x*y};
            ans[p].name=s;
            ans[p].num=x;
            ans[p].price=x*y;
            p++;
        }
        else{
            mp[s]={mp[s].first+x,mp[s].second+x*y};
            for(int i=0;i<p;i++){
                if(ans[i].name==s){
                    ans[i].num+=x;
                    ans[i].price+=x*y;
                }
            }
        }


    }
    sort(ans,ans+p,cmp);
    for(int i=0;i<p;i++){
        cout<<ans[i].name<<" "<<ans[i].num<<" "<<ans[i].price<<endl;
    }

}

C: 二叉排序树-平衡因子

题目描述

给定一个序列,使用该序列生成二叉排序树,然后以本题规定方法输出该二叉排序树。
例:
给定一个序列:43 25 29 67 17 88 54 47 35 62
以第一个数字(43)为根节点,然后将后面的数字依输入次序逐个添加至该树中,得到一个二叉排序树,如下图所示。
在这里插入图片描述
然后先序遍历上面这个树,并按行输出数字。
其中每个子节点的输出前,需要相较于其父节点前多四个普通空格。
当某个节点为叶子节点(即无子节点),则该节点的左右叶子节点均不用输出。
而当某个节点仅有左叶子节点或右叶子节点时,另一个空缺的子节点用#占位。
对于非空的节点,求出其平衡因子,并用括号括起来输出在结果中
以该图为例,其最终输出结果为:

在这里插入图片描述

输入样例

10
43 25 29 67 17 88 54 47 35 62

输出样例

在这里插入图片描述
写了一个pdf版本的,那里C题与A题一起分析

思路

对于A题,先建立二叉排序树,再根据题意输出,对于题目中的空格的输出,由于同一层前面的空格时一样的,可用一个depth记录一下当前层数。然而对于C题,需要输出平衡因子,即左子树高度减去右子树的高度,呃。。。直接算就好了,放心不会超时,我试过了,计算方法前面都写过.
Dep(T)=max(Dep(T->lchild)+1,Dep(T->rchild)+1)

code

#include<bits/stdc++.h>
using namespace std;
#define isw {ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);}
typedef pair<int,int>P;
#define inf 0x3f3f3f3f
#define maxn 100005
#define int long long 
struct bstnode{
     int data;
     bstnode *rchild,*lchild;
};
typedef bstnode* bst;
void init(bstnode* &t){
    t=NULL;
}
int Depth(bst t){//求树的高度,原理在前面的作业说过
    if(t==NULL)return 0;
    return max(Depth(t->lchild)+1,Depth(t->rchild)+1);
}
int Insert(bst &t,int x){//插入元素x
    bst p=(bst)malloc(sizeof(bstnode));
            p->data=x;
            p->lchild=NULL;
            p->rchild=NULL;
    if(!t){
        t=p;
    }
     else if(t->data>x){    
        Insert(t->lchild,x);
    }
   else{
        Insert(t->rchild,x);
    }
    return 1;
}
void pretral(bst t,int depth){
    for(int i=0;i<depth;i++){
            cout<<"    ";
        }
    if(!t){ 
        cout<<"#"<<endl;
    }
    else{
        cout<<t->data;
        printf("(%d)\n",Depth(t->lchild)-Depth(t->rchild));
       if(!t->lchild&&!t->rchild){
           //cout<<"#"<<endl;
           return;
       }
       else{
           pretral(t->lchild,depth+1);
           pretral(t->rchild,depth+1);
       }
    }
}
signed main(){
    int n;
    cin>>n;
    bst t;
    init(t);
    for(int i=0;i<n;i++){
        int x;
        cin>>x;
        Insert(t,x);
    }
    pretral(t,0);
}

D二分查找

题目描述

给定大小为N(0<N<1000),从小到大排列的整数数组A[],以及待查找的整数X,目的是找到X在数组A[]中第一次出现的位置并输出。如没有找到则输出-1;
输入格式
第一行 数组大小 N
第二行 数组A[]
第三行 带查找的整数X
输出格式
如找到X,则输出第一次出现的位置。如未找到,则输出-1。

输入样例

5
1 2 4 4 5
4

输出样例

3

思路

最简单的二分查找模板题,虽然不用二分也不会超时

code

#include<bits/stdc++.h>
using namespace std;
#define isw {ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);}
typedef pair<int,int>P;
#define inf 0x3f3f3f3f
#define maxn 100005
#define int long long
int a[1005]; 
signed main(){
    int n;
    int x;
    cin>>n;
    for(int i=1;i<=n;i++)cin>>a[i];
    cin>>x;
    int l=0,r=n;
    int ans=0;
    while(l<=r){
        int mid=(l+r)>>1;
        if(a[mid]<x){
            l=mid+1;
        }
        else {
            
            r=mid-1;
            ans=mid;
        }
    }
    if(a[ans]!=x)cout<<-1<<endl;
    else cout<<ans<<endl;
}

E两个有序序列的中位数

题目描述

已知有两个等长非降序序列S1,S2。先将S1,S2合并为S3,求S3的中位数。长度为N的非降序序列SN的中位数为第X个数,X=不超过(N+1)/2的最大整数。

输入格式
第一行,序列S1,S2的长度N
第二行,序列S1的N个整数
第三行,序列S2的N个整数
输出格式
输出两个序列合并后序列S3的中位数

输入样例

5
1 3 5 7 9
2 3 4 5 6

输出样例

4

思路

将两个序列合并后排序,中间的数就是中位数

code

#include<bits/stdc++.h>
using namespace std;
#define isw {ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);}
typedef pair<int,int>P;
#define inf 0x3f3f3f3f
#define maxn 100005
#define int long long 
signed main(){
    int n;
    cin>>n;
    int a[2*n+5];
    for(int i=0;i<2*n;i++)cin>>a[i];
    sort(a,a+2*n);
    cout<<a[n-1];
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值