程序设计思维与实践 Week2 实验

A 化学

1.题目大意

化学很神奇,以下是烷烃基。在这里插入图片描述
假设如上图,这个烷烃基有6个原子和5个化学键,6个原子分别标号1~6,然后用一对数字 a,b 表示原子a和原子b间有一个化学键,这样通过5行a,b可以描述一个烷烃基。

你的任务是甄别烷烃基的类别。

输入:
输入第一行为数据的组数T(1≤T≤200000)。每组数据有5行,每行是两个整数a, b(1≤a,b≤6,a≤b)
数据保证,输入的烷烃基是以上5种之一

输出:
每组数据,输出一行,代表烷烃基的英文名

样例:

2
1 2
2 3
3 4
4 5
5 6
1 4
2 3
3 4
4 5
5 6
n-hexane
3-methylpentane

2.解题思路

  • 这道题主要考虑烷羟基的表示方式,并且找到不同烷羟基区别特征。
  • 烷羟基结构可以用图来表示,顶点表示原子,边表示化学键
  • 判断烷羟基类型可以转化为判断点的邻接边数量。
  • 烷羟基中原子编号无序,因此判断时需考虑邻接边条数分别为1~4的点的个数,不能考虑指定编号的点邻接边条数。
  • 观察可得每种烷羟基的区分方法:
  1. n-hexane没有邻接边条数大于2的顶点
  2. 2,3-dimethylbutane有2个邻接边条数等于3的顶点
  3. 2,2-dimethylbutane有1个邻接边条数等于4的顶点
  4. 3-methylpentane有1个邻接边条数等于3的顶点,该顶点的邻接点中只有1个邻接边条数等于2的顶点
  5. 2-methylpentane也有1个邻接边条数等于3的顶点,而该顶点的邻接点中有2个邻接边条数等于2的顶点

3.代码

#include <iostream>
using namespace std;

struct chainNode
{
    int element;
    chainNode *next;
    
    chainNode() {}
    chainNode(const int& element) {this->element=element;}
    chainNode(const int& element, chainNode *next) {this->element=element; this->next=next;}
};

class graphChain
{
protected:
    chainNode* firstNode;
    int listSize;
    
public:
    graphChain() {firstNode=NULL; listSize=0;}
    ~graphChain()
    {
        while (firstNode!=NULL)
        {
            chainNode* nextNode=firstNode->next;
            delete firstNode;
            firstNode=nextNode;
        }
    }
    int size() const {return listSize;}
    chainNode* first() const {return firstNode;}
    void insert(const int& theElement)
    {
        chainNode *p=firstNode, *tp=NULL;
        while (p!=NULL)
        {
            tp=p;
            p=p->next;
        }
        chainNode *newNode=new chainNode(theElement,p);
        if (tp==NULL) firstNode=newNode;
        else tp->next=newNode;
        
        listSize++;
    }
};

class linkedGraph
{
private:
    int n;
    int e;
    graphChain *aList;
    
public:
    linkedGraph(int ver=0)
    {
        n=ver;
        e=0;
        aList=new graphChain[n+1];
    }
    ~linkedGraph() {delete [] aList;}
    
    void insertEdge(int u, int v)
    {
        aList[u].insert(v);
        aList[v].insert(u);
        e++;
    }
    
    void judge()
    {
        int *a=new int[n];
        for (int i=1;i<=n;i++)
            a[i]=0;
        for (int i=1;i<=n;i++)
            a[aList[i].size()]++;
        
        if (a[4])
            cout<<"2,2-dimethylbutane"<<endl;
        else if (a[3]==2)
            cout<<"2,3-dimethylbutane"<<endl;
        else if (!a[3])
            cout<<"n-hexane"<<endl;
        else if (a[3]==1)
        {
            int count=0;
            for (int i=1;i<=n;i++)
            {
                if (aList[i].size()==3)
                {
                    for (chainNode *w=aList[i].first();w!=NULL;w=w->next)
                        if (aList[w->element].size()==2)
                            count++;
                    
                    if (count==2) cout<<"3-methylpentane"<<endl;
                    else cout<<"2-methylpentane"<<endl;
                }
            }
        }
        
    }
};

int main()
{
    int n,u,v;
    cin>>n;
    for (int i=0;i<n;i++)
    {
        linkedGraph graph(6);
        for (int j=0;j<5;j++)
        {
            cin>>u>>v;
            graph.insertEdge(u,v);
        }
        graph.judge();
    }
    return 0;
}

4.实现细节

  • 利用了数据结构课程中图类的代码,构造链表描述的图,已存在插入边函数insertEdge用于初始化烷羟基。
  • judge函数判断烷羟基类别。
  1. 用a[k]数组记录每个图中邻接边条数为k的顶点个数为a[k],遍历图的顶点,每次进行a[aList[i].size()]++实现。
  2. 再根据解题思路中烷羟基的区别方法进行区别,分类a[4]>0或a[3]==2或a[3]==0或a[3]==1等。
  3. a[3]==1有两种情况,进一步考虑该顶点的邻接点的情况。

B 爆零

1.题目大意

实时评测系统具有及时获得成绩排名的特点,那它的功能是怎么实现的呢?做题过程中,通过的题数虽然越来越多,但通过每题时所共花去的时间(从最开始算起,直至通过题目时的这段时间)都会被记录下来。特别的,对于通过的题目,曾经的关于这题的每次错误提交都会被算上一定的单位时间罚时,这样一来,在做出的题数上可能领先,但是在做出同样题数的人中,可能会因为罚时过高而处于排名上的劣势。
例如某次考试一共八道题(A,B,C,D,E,F,G,H),每个人做的题都在对应的题号下有个数量标记,负数表示该学生在该题上有过的错误提交次数但到现在还没有AC,正数表示AC所耗的时间,如果正数a跟上了一对括号,里面有个正数b,则表示该学生AC了这道题,耗去了时间a,同时曾经错误提交了b次。例子可见下方的样例输入与输出部分。

输入:
输入数据包含多行,第一行是共有的题数n(1≤n≤12)以及单位罚时m(10≤m≤20),之后的每行数据描述一个学生的信息,首先是学生的用户名(不多于10个字符的字串)其次是所有n道题的得分现状,其描述采用问题描述中的数量标记的格式,见上面的表格。

输出:
根据这些学生的得分现状,输出一个实时排名。实时排名显然先按AC题数的多少排,多的在前,再按时间分的多少排,少的在前,如果凑巧前两者都相等,则按名字的字典序排,小的在前。每个学生占一行,输出名字(10个字符宽),做出的题数(2个字符宽,右对齐)和时间分(4个字符宽,右对齐)。名字、题数和时间分相互之间有一个空格。数据保证可按要求的输出格式进行输出。

样例:

8 20
GuGuDong  96     -3    40(3) 0    0    1      -8    0
hrz       107    67    -3    0    0    82     0     0
TT        120(3) 30    10(1) -3   0    47     21(2) -2
OMRailgun 0      -99   -8    0    -666 -10086 0     -9999996
yjq       -2     37(2) 13    -1   0    113(2) 79(1) -1
Zjm       0      0     57(5) 0    0    99(3)  -7    0
TT          5  348
yjq         4  342
GuGuDong    3  197
hrz         3  256
Zjm         2  316
OMRailgun   0    0

2.解题思路

  • 这道题主要考虑准确地进行格式化输入和输出。
  • 先根据输入内容判断学生答题信息:
  1. 得分为0,表示该题未提交未通过
  2. 得分为-a (a > 0),表示该题已提交a次都未通过
  3. 得分为a (a > 0),表示该题在a时刻提交一次性通过
  4. 得分为a(b) (a > 0 && b > 0)表示该题在a时刻提交通过前错误提交b次未通过
  • 需要操作符重载或定义比较函数,用于根据升降序要求进行多条件排序。

3.代码

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int MAXN=10000;

struct Competitor
{
    char name[11];
    int score=0;
    int penalty=0;
    
    bool operator < (const Competitor &com) const
    {
        if (score!=com.score) return score>com.score;
        if (penalty!=com.penalty) return penalty<com.penalty;
        return strcmp(name,com.name)<0;
    }
}cpts[MAXN];

int main()
{
    int n,m,res,t,f,i;
    char record[10];
    cin>>n>>m;
    for (i=0;cin>>cpts[i].name;i++)
    {
        for (int j=0;j<n;j++)
        {
            scanf("%s",record);
            res=sscanf(record,"%d(%d)",&t,&f);
            if (res==2)
            {
                cpts[i].score++;
                cpts[i].penalty+=t+f*m;
            }
            else if (res==1 && t>0)
            {
                cpts[i].score++;
                cpts[i].penalty+=t;
            }
        }
    }
    sort(cpts,cpts+i);
    for (int j=0;j<i;j++)
        printf("%-10s %2d %4d\n",cpts[j].name,cpts[j].score,cpts[j].penalty);
    
    return 0;
}

4.实现细节

  • 创建Competitor类用于记录每个学生的答题情况,其中重载操作符 < 用于后续多条件排序。
    (此处得知若升序排序则return xxx<a.xxx,符号与重载的符号相同,降序排序则符号相反,终于不再这种小逻辑上纠结了!)
  • 这道题的重点在于如何正确读入并判断每个学生的得分现状,这里用sscanf函数用于格式化读入最合适。
  1. sscanf函数返回值为读入数据的个数,可以判断得分现状是否有()的数据
  2. 返回值为2表示通过但非一次性,得分增加,罚时增加时间和错误次数
  3. 返回值为1且a<=0表示未通过,不做处理;返回值为1且a>0表示一次性通过,得分增加,罚时仅增加时间
  • 输出时也要注意格式。这里记录一些不熟悉的printf输出格式:
  1. “%-xxx”:以特定宽度左对齐
  2. “%+xxx”:以特定宽度右对齐
  3. “% xxx”:输出值为正显示空格,为负时显示负号
  4. “%#xxx”:对c、s、d、u类无影响;对o类,在输出时加前缀o;对x类,在输出时加前缀0x;对e、g、f 类当结果有小数时才给出小数点

C 瑞神打牌

1.题目大意

牌局由四个人构成,围成一圈。我们称四个方向为北、东、南、西。对应的英文是North,East,South,West。游戏一共由一副扑克,也就是52张构成。开始,我们指定一位发牌员(东南西北中的一个,用英文首字母标识)开始发牌,发牌顺序为顺时针,发牌员第一个不发自己,而是发他的下一个人(顺时针的下一个人)。这样,每个人都会拿到13张牌。
现在我们定义牌的顺序,首先,花色是(梅花)<(方片)<(黑桃)<(红桃),(输入时,我们用C, D, S, H分别表示梅花,方片,黑桃,红桃,即其单词首字母)。对于牌面的值,我们规定2 < 3 < 4 < 5 < 6 < 7 < 8 < 9 < T < J < Q < K < A。
现在你作为上帝,你要从小到大排序每个人手中的牌,并按照给定格式输出。(具体格式见输出描述和样例输出)。

输入:
输入包含多组数据
每组数据的第一行包含一个大写字符,表示发牌员是谁。如果该字符为‘#’则表示输入结束。
接下来有两行,每行有52个字符,表示了26张牌,两行加起来一共52张牌。每张牌都由两个字符组成,第一个字符表示花色,第二个字符表示数值。

输出:
输出多组数据发牌的结果,每组数据之后需要额外多输出一个空行
每组数据应该由24行的组成,输出按照顺时针方向,始终先输出South Player的结果,每位玩家先输出一行即玩家名称(东南西北),接下来五行,第一行和第五行输出固定格式(见样例),第二行和第四行按顺序和格式输出数值(见样例),第三行按顺序和格式输出花色(见样例)。

样例:

N
CTCAH8CJD4C6D9SQC7S5HAD2HJH9CKD3H6D6D7H3HQH4C5DKHKS9
SJDTS3S7S4C4CQHTSAH2D8DJSTSKS2H5D5DQDAH7C9S8C8S6C2C3
#
South player:
+---+---+---+---+---+---+---+---+---+---+---+---+---+
|6 6|A A|6 6|J J|5 5|6 6|7 7|9 9|4 4|5 5|7 7|9 9|T T|
| C | C | D | D | S | S | S | S | H | H | H | H | H |
|6 6|A A|6 6|J J|5 5|6 6|7 7|9 9|4 4|5 5|7 7|9 9|T T|
+---+---+---+---+---+---+---+---+---+---+---+---+---+
West player:
+---+---+---+---+---+---+---+---+---+---+---+---+---+
|2 2|5 5|9 9|K K|5 5|7 7|9 9|4 4|T T|J J|A A|8 8|A A|
| C | C | C | C | D | D | D | S | S | S | S | H | H |
|2 2|5 5|9 9|K K|5 5|7 7|9 9|4 4|T T|J J|A A|8 8|A A|
+---+---+---+---+---+---+---+---+---+---+---+---+---+
North player:
+---+---+---+---+---+---+---+---+---+---+---+---+---+
|3 3|4 4|J J|2 2|3 3|T T|Q Q|K K|8 8|Q Q|K K|2 2|3 3|
| C | C | C | D | D | D | D | D | S | S | S | H | H |
|3 3|4 4|J J|2 2|3 3|T T|Q Q|K K|8 8|Q Q|K K|2 2|3 3|
+---+---+---+---+---+---+---+---+---+---+---+---+---+
East player:
+---+---+---+---+---+---+---+---+---+---+---+---+---+
|7 7|8 8|T T|Q Q|4 4|8 8|A A|2 2|3 3|6 6|J J|Q Q|K K|
| C | C | C | C | D | D | D | S | S | H | H | H | H |
|7 7|8 8|T T|Q Q|4 4|8 8|A A|2 2|3 3|6 6|J J|Q Q|K K|
+---+---+---+---+---+---+---+---+---+---+---+---+---+

2.解题思路

  • 这道题主要考虑把输入的牌发给每位玩家,并且比较牌的大小。
  • 先根据发牌员的位置和发牌顺序(顺时针)确定每个玩家手中的牌:
  1. 将四个位置S, W, N, E以顺时针为升序表示为数字,则可以通过下标取模运算,每隔4张牌将牌发到相同的玩家
  2. 由于发牌员先给自己的下一个玩家发牌,因此表示四个位置的数字从1开始升序而非从0开始(这样从i=0开始遍历发牌时下标不用+1)
  • 需要操作符重载或定义比较函数,用于根据规定进行牌的排序。
    这里的排序涉及到数字和字母的混合排序,因此把字母转化为对应大小的数字会更加方便

3.代码

#include <iostream>
#include <string.h>
#include <algorithm>
#include <vector>
using namespace std;

char input[10000];
int banker;

struct Card
{
    char col,num;
    Card();
    Card(int theCol,int theNum)
    {
        col=theCol;
        num=theNum;
    }
};

vector<Card> player[4];

int getCol(char col)
{
    switch (col)
    {
        case 'C': return 0;
        case 'D': return 1;
        case 'S': return 2;
        case 'H': return 3;
    }
    return -1;
}

int getNum(char num)
{
    switch (num)
    {
        case 'T': return 10;
        case 'J': return 11;
        case 'Q': return 12;
        case 'K': return 13;
        case 'A': return 14;
    }
    return num-'0';
}

bool cmp(const Card &card1,const Card& card2)
{
    if (card1.col==card2.col)
        return getNum(card1.num)<getNum(card2.num);
    return getCol(card1.col)<getCol(card2.col);
}

int main()
{
    while (scanf("%s",input) && input[0]!='#')
    {
        switch (input[0])
        {
            case 'S': banker=1; break;
            case 'W': banker=2; break;
            case 'N': banker=3; break;
            case 'E': banker=0; break;
        }
        for (int i=0;i<4;i++)
            player[i].clear();
        scanf("%s%s",input,input+52);
        for (int i=0;i<104;i+=2,banker=(banker+1)%4)
            player[banker].push_back(Card(input[i],input[i+1]));
        
        for (int i=0;i<4;i++)
            sort(player[i].begin(),player[i].end(),cmp);
        
        for (int i=0;i<4;i++)
        {
            switch (i)
            {
                case 0: cout<<"South player:"<<endl; break;
                case 1: cout<<"West player:"<<endl; break;
                case 2: cout<<"North player:"<<endl; break;
                case 3: cout<<"East player:"<<endl; break;
            }
            cout<<"+---+---+---+---+---+---+---+---+---+---+---+---+---+"<<endl;
            int j;
            for (j=0;j<13;j++)
            {
                cout<<"|"<<player[i][j].num<<" "<<player[i][j].num;
                if (j==12) cout<<"|"<<endl;
            }
            for (j=0;j<13;j++)
            {
                cout<<"| "<<player[i][j].col<<" ";
                if (j==12) cout<<"|"<<endl;
            }
            for (j=0;j<13;j++)
            {
                cout<<"|"<<player[i][j].num<<" "<<player[i][j].num;
                if (j==12) cout<<"|"<<endl;
            }
            cout<<"+---+---+---+---+---+---+---+---+---+---+---+---+---+"<<endl;
        }
        cout<<endl;
    }
    return 0;
}

4.实现细节

  • 创建Card类用于记录每张牌的花色和大小,4个Card型向量分别表示四个玩家的牌。
  • 先读入所有牌的信息,读入时注意scanf需要分为两部分input/input+52,防止读入中间的换行。
  • 遍历上一步读入的数据进行发牌:
  1. 由于花色和大小占用两个字符,以及根据解题思路中的下标取模运算,因此每个循环 i += 2 且 banker = (banker+1) % 4 为步长
  2. player[banker].push_back(Card(input[i],input[i+1]))发牌
  • getCol和getNum函数分别为花色的字符和数字转换以及大小的字符和数字转换函数,在比较函数cmp里调用,用于排序输出。
    (原来的思路是将字符串转化成数字后再输入Card,用结构体里操作符重载进行排序,就不需要cmp函数了,但是实现后发现这样在输出的时候还要重新转化成字符串,比较不方便)
  • 按照格式输出时注意行首和行末的空格与中间位置不同。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
东南亚位于我国倡导推进的“一带一路”海陆交汇地带,作为当今全球发展最为迅速的地区之一,近年来区域内生产总值实现了显著且稳定的增长。根据东盟主要经济体公布的最新数据,印度尼西亚2023年国内生产总值(GDP)增长5.05%;越南2023年经济增长5.05%;马来西亚2023年经济增速为3.7%;泰国2023年经济增长1.9%;新加坡2023年经济增长1.1%;柬埔寨2023年经济增速预计为5.6%。 东盟国家在“一带一路”沿线国家中的总体GDP经济规模、贸易总额与国外直接投资均为最大,因此有着举足轻重的地位和作用。当前,东盟与中国已互相成为双方最大的交易伙伴。中国-东盟贸易总额已从2013年的443亿元增长至 2023年合计超逾6.4万亿元,占中国外贸总值的15.4%。在过去20余年中,东盟国家不断在全球多变的格局里面临挑战并寻求机遇。2023东盟国家主要经济体受到国内消费、国外投资、货币政策、旅游业复苏、和大宗商品出口价企稳等方面的提振,经济显现出稳步增长态势和强韧性的潜能。 本调研报告旨在深度挖掘东南亚市场的增长潜力与发展机会,分析东南亚市场竞争态势、销售模式、客户偏好、整体市场营商环境,为国内企业出海开展业务提供客观参考意见。 本文核心内容: 市场空间:全球行业市场空间、东南亚市场发展空间。 竞争态势:全球份额,东南亚市场企业份额。 销售模式:东南亚市场销售模式、本地代理商 客户情况:东南亚本地客户及偏好分析 营商环境:东南亚营商环境分析 本文纳入的企业包括国外及印尼本土企业,以及相关上下游企业等,部分名单 QYResearch是全球知名的大型咨询公司,行业涵盖各高科技行业产业链细分市场,横跨如半导体产业链(半导体设备及零部件、半导体材料、集成电路、制造、封测、分立器件、传感器、光电器件)、光伏产业链(设备、硅料/硅片、电池片、组件、辅料支架、逆变器、电站终端)、新能源汽车产业链(动力电池及材料、电驱电控、汽车半导体/电子、整车、充电桩)、通信产业链(通信系统设备、终端设备、电子元器件、射频前端、光模块、4G/5G/6G、宽带、IoT、数字经济、AI)、先进材料产业链(金属材料、高分子材料、陶瓷材料、纳米材料等)、机械制造产业链(数控机床、工程机械、电气机械、3C自动化、工业机器人、激光、工控、无人机)、食品药品、医疗器械、农业等。邮箱:market@qyresearch.com

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值