2014哈商大ICPC/ACM校赛解题报告
被debug邀请去參加校赛,哎,被虐。。我对不起工大。。
由于本人不搞ACM,算法处于HelloWorld水准。。
虽然题目除了鸟不拉屎星人之外都非常水,但我能做到这个程度,全然是超水平发挥了。。
数据:点此下载
==============================================================
a:逆序数组+删除特定元素
小伙伴们好像非常多没接触过ICPC,那先来一道水题尝尝鲜,给出一个数组,和一个特征值。将这个数组中特征值值删除后逆序输出。EG: 数组,1 2 3 4 5 6 7特征值4那么输出是:7 6 5 3 2 1
输入:
输入包括3行,第一行输入N(N< 100)表示数组长度
第二行输入 N个数
第三行 输入特征值T
输出:
删除特征值后的逆序数组,每组输出要换行。
例子输入:
7
1 2 3 4 5 6 7
4
6
例子输出:
765321
#include <iostream>
#include <vector> using namespace std; int main()
{
int n,tmp,delNum; while(cin >> n){
vector<int> a;
while(n--){
cin >> tmp;
a.push_back(tmp);
}
cin >> delNum; vector<int>::reverse_iterator a_rIter;
for(a_rIter = a.rbegin();a_rIter != a.rend(); ++a_rIter){
if(*a_rIter == delNum)
continue;
cout << *a_rIter;
}
cout << endl;
}
return 0;
}
b:括号匹配
在公式中常常遇到大量的括号,可是这些括号匹配的对不正确呢?由于不同意一个括号里包括不成对的括号.不同类型的括号能够互相包括比方这样{()},([]),[{}], {[][]()};可是一种类型的括号不同意包括其它类型不成对的括号,比方这样就是不同意的:{(} [{] {)};)(,}{,][这样也是不同意的。如今给出一个有括号组成的字符串,推断其是否合法。
输入:每组输入包括一个括号字符串
输出:推断括号是字符串是否合法,合法输出yes不合法输出no每次输出要换行
例子输入:{}{}()()(()){[()]}{}{}(([])){[}]
例子输出:YesNo
代码:
#include <iostream>
#include <stack> using namespace std; int main()
{
stack<char> s;
string str;
while(cin >> str){
char tmp;
bool ans = true;
for(unsigned int i = 0 ; i < str.size();i++){
if(str[i] == '(' || str[i] == '[' || str[i] == '{')
s.push(str[i]);
else{
if(s.empty()){ //假设栈为空,并且还遇到这样的字符就直接No了
ans = false;
break;
}
tmp = s.top();
s.pop();
if((str[i] == ')' && tmp == '(') ||
(str[i] == ']' && tmp == '[') ||
(str[i] == '}' && tmp == '{')){
continue;
}else{
ans = false;
break;
}
}
}
if(ans)
cout<<"Yes"<<endl;
else
cout<<"No"<<endl;
}
return 0;
}
c: 再多一天
输入一个日期,你来算算第二天的日期
输入:每组输入包括一个日期,格式如 2014-4-5
输出:输出第二天的日期 2014-4-6;每次输出要换行;
例子输入:2014-4-52014-4-6
例子输出2014-4-62014-4-7
#include <iostream>
#include <string>
#include <vector>
#include "stdio.h" using namespace std; class Time{
public:
int year;
int month;
int day; Time(int year,int month,int day)
{
this->day = day;
this->month = month;
this->year = year;
} bool isRunYear(){
if((this->year % 4 == 0 && this->year % 100 != 0) || this->year % 400 == 0)
return true;
return false;
} void addOneDay(){
this->day += 1; //do with day
if(this->day == 32){
this->day = 1;
this->month += 1;
}
else if(this->day == 31 && (this->month == 4 || this->month == 6 || this->month == 9 || this->month == 11)){
this->day = 1;
this->month += 1;
}
else if(this->day == 30 && this->month == 2 && isRunYear()){
this->day = 1;
this->month += 1;
}
else if(this->day == 29 && this->month == 2 && !isRunYear()){
this->day = 1;
this->month += 1;
} //do with month
if(month == 13){
this->month = 1;
this->year += 1;
}
} }; int main()
{
int y,m,d;
while(scanf("%d-%d-%d",&y,&m,&d)==3){
Time *t = new Time(y,m,d);
t->addOneDay();
cout<<t->year<<"-"<<t->month<<"-"<<t->day<<endl;
}
return 0;
}
/*改动于2014-4-23:9:23
vector<int> split(string str, string patternStr)
{
vector<int> res;
for(unsigned int i = 0 ; i < str.size() ; i++){
int pos = str.find(patternStr,i);
string s;
if(pos < str.size() && pos >= 0){
s = str.substr(i,pos-i);
i = pos;
}else{
s = str.substr(i);
}
stringstream ss; //string 转 int
ss << s;
int t;
ss >> t;
res.push_back(t);
}
return res;
}
d:相似字符串
如今字符串相似有个法则:假设字符串A能通过有次的变换能和字符串B同样那么说明他们是相似的。这样的变换法则是:将串头字母插入到串尾,然后总体迁移一位。比方:abcde cdeababcde 通过将 a 置尾变成 bcdeabcdea 通过将 b 置尾变成 cdeab那么说明abcde 和 cdeab 是相似的。
输入:输入包括两个等长字符串
输出:若两字符串相似输出yes否则输出no
例子输入:abcde cdeababc acb
例子输出:YesNo
#include <iostream>
#include <string> using namespace std; int main()
{
string str1,str2;
while(cin>>str1>>str2){
string str3 = str2+str2;
int length = str1.size(); if(str3.find(str1)<length)
cout<<"Yes"<<endl;
else
cout<<"No"<<endl;
}
return 0;
}
在str2+str2中怎样能找到str1,就是相似。。
找不时,返回值是string::npos。
npos是这样定义的: static const size_type npos = -1;
但输出是做无符号整数型别输出,即-1的补码4294967295。
反之,str.find("哦")!=string::npos则说明字符串str中存在“哦”这个字符
e:迷宫
小伙伴们钻进迷宫寻找宝藏,你是否能从(左边)入口開始进去,寻找到宝物呢。
输入:给出一个迷宫矩阵图,0表示通过,1表示墙壁。有左上角进入迷宫開始寻宝。
输出:输出寻找到的宝物符号,每次输出要换行
Sample In:111111111111111111111100000000000000000000011111111111111111111101111111111111111111110111000000000000*1111101111111111111111111110111111111111111111111011+1111111111111111110110000000000000000000011111111111111111111111
Sample Out:+
#include <iostream>
#include "stdio.h" using namespace std; char m[100][100]; int r = 0; //row
int c = 0; //column void dfs(int x,int y)
{
if(m[x][y] != '0')
cout<<m[x][y]<<endl;
m[x][y] = '1';//标记訪问过
//down
if(x+1 < r && m[x+1][y] != '1'){
dfs(x+1,y);
}
//up
if(x-1 >= 0 && m[x-1][y] != '1'){
dfs(x-1,y);
}
//right
if(y+1 < c && m[x][y+1] != '1'){
dfs(x,y+1);
}
//left
if(y-1 >= 0 && m[x][y-1] != '1'){
dfs(x,y-1);
} } int main()
{
//freopen("in.txt","r",stdin);
//freopen("out.txt","w",stdout); string t;
int index = 0 ;
while(cin>>t){
for(unsigned int i=0;i<t.size();i++)
m[index][i] = t[i];
c = t.size();
index ++;
}
r = index; //入口在左边,找入口
for(int i = 0;i<r;i++){
if(m[i][0] == '0')
dfs(i,0);
} return 0;
}
f:小胖子厨师
大家都知道小胖子是个吃货,可是都不知道小胖子还是个非常厉害的厨师,他的拿手功夫是切墩><.有一天他切肉时陷入了沉思:我怎么才干用最少的刀数把肉切的更碎呢(咱们先不考虑3维问题,仅仅在平面上思考吧) 。比方,我一刀下去能切成两块,两刀下去能切成四块,小胖子一气之下切了N多刀,肉最多被切成了多少块(预计不能吃了) ?
In:输入小胖子切了多少刀N
Out:输出肉最多能被切成多少块,每次输出要换行
Sample in:12
Sample out:24
#include <iostream> using namespace std; int main()
{
int n;
while(cin>>n){
int ans = (1+n)*n/2 + 1;
cout<<ans<<endl;
}
return 0;
}
这道题不得不停下来吐槽下,平面切割问题。。。
=f(n-2)+(n-1)+n
……
=f(1)+1+2+……+n
=n(n+1)/2+1
g:小气的自来水公司
如今有N个村庄,村庄与村庄之间有自来水管联通,如今自来水公司为了节省耗材将拆除多余耗材, 为了使拆卸得到的耗材最多而且要保证村庄之间水路想通。
输入:首先输入m n, (m,n< 26)m表示有m个村庄,n表示村庄之间有多少水路。接下来有n行,每行包括 两个大写字符,和一个整形数Eg.AB 10,表示村庄A B之间存在水路且长度是10
输出:拆卸结束后,村庄之间水路的总长度。
Sample in:3 2AB 1B C 1
Sample out:2
#include <iostream>
#include <algorithm>
#include <queue>
#include <vector>
#include "stdio.h" using namespace std; #define MAXN 1000
#define INF 1<<30 int closest[MAXN],lowcost[MAXN],m;// m为节点的个数
int G[MAXN][MAXN];//邻接矩阵 int prim(int m)
{
for(int i=0;i<m;i++)
{
lowcost[i]=INF;
}
for(int i=0;i<m;i++)
{
closest[i]=0;
} closest[0]=-1;//增加第一个点,-1表示该点在集合U中,否则在集合V中
int num=0,ans=0,e=0;//e为最新增加集合的点
while(num<m-1)//增加m-1条边
{
int micost=INF,miedge=-1;
bool flag = false;
for(int i=0;i<m;i++){ //找出最小边
if(closest[i]!=-1)
{
int temp=G[e][i];
if(temp<lowcost[i])
{
lowcost[i]=temp;
closest[i]=e;
}
if(lowcost[i]<micost){
micost=lowcost[i];
miedge = i;
flag = true;
}
}
}
if(!flag){
return -1;
}
ans+=micost;
closest[miedge]=-1;
e = miedge;
num++;
} return ans;
} int main()
{
int m,n;
while(scanf("%d %d",&m,&n) == 2){
for(int i=0;i<m;i++) //清空数组
for(int j=0;j<m;j++)
G[i][j] = INF; char a,b;
int w;
for(int i = 0 ; i < n;i++){
cin>>a>>b>>w;
G[a-'A'][b-'A'] = w;
G[b-'A'][a-'A'] = w;
} int ans = prim(m);
if(ans == -1)
cout<<"Can't do it"<<endl;
else
cout<<"Total:"<<ans<<endl;
}
return 0;
}
h:吐槽星人大战鸟不拉屎大王
HUC要举办ACM/ICPC校赛的消息已经传遍了整个宇宙,位于M77星云的鸟不拉屎大王为了赢得这次比赛, 他决定派出自己最为自傲的队伍。而在宇宙的深处,宇宙间的和平使者,总是破坏鸟不拉屎大王毁灭宇宙大阴谋,继呕吐曼之后成为全宇宙大英雄的吐槽星人,在得知了这个消息之后,为了阻止鸟不拉屎大王,也决定选出对应的队伍去參加比赛。 (他们总是这样,不管鸟不拉屎大王在做什么,阻止已经成为了习惯)可是宇宙间总是充斥着各种邪恶的存在,吐槽星人当然不可能将所有的力量放在鸟不拉屎大王身上 (比方说位于古老地球的蛇精,女王大人,在俘获了福禄小金刚之后也开展了妄图统治宇宙的邪恶计划)所以吐槽星人在得知了鸟不拉屎大王派出的队伍之后, 也派出了队伍。在出发比赛的当天,因为星际虫洞发生了严重的交通事故,造成了交通大堵塞。鸟不拉屎大王的队伍就和他们宿世的仇敌,吐槽星人的队伍坐在了同一辆星际飞船上開始了旅行 (不要问我他们为什么没有自己的旅行飞船, 要知道就算鸟不拉屎大王这样的宇宙高富帅的存在也会与资金紧张的时候,恰恰吐槽星人也受到了相同的困扰) 。我们知道每一个吐槽星人都有自己生而拥有的吐槽之力, 而鸟不拉屎大王的手下也有属于自己的翔之力 (能力的数值都在1 ~ n ^ 2之间的整数) 。如今正在飞往比赛地点的路程上发现了翔之力的秘密。 (这个是个大消息)他们决定展开自己的秘密计划,详细的内容就是当时在飞船之中,鸟不拉屎大王的手下坐成一排编号为0~p,吐槽星人坐一排编号为0~q,两排平行,吐槽星人在位置不变的情况下同一时候发动吐槽之力,各自能力会形成各自的一道能量波,能量波不能互相交叉, 发向对面的鸟不拉屎大王的手下, 这样就能够将对方成功吸收成为自己的人,为比赛加大胜利的筹码。我们会给出两方相应的翔之力和吐槽之力,你要计算出能够吸收的最大人数,当然因为基因的不同,每一个人的能力不可能呈现出同样的大小。 而可以成功计算出最大吸收人数的会有幸成为幸得吐槽星人, 维护宇宙的和平, 真是个有意义的工作。比方这两支队伍的能力各自是1 7 5 4 8 3 91 4 3 5 6 2 8 9我们能够选择1 -> 4 ->8 -> 9 或者 1 -> 5 -> 8 -> 9 或者 1 -> 4 -> 3-> 9 这三种吸收模式而1->5->3->9这样的模式就不能够
INPUT:输入的第一行为数据组数Case (Case <= 10),接下来每组数据包括3行,第一行为3个整数n,p,q ( 2 <= n <= 250, 0 <= q, p <= n ^ 2 ),接下来第二行是鸟不拉屎大王的手下的能量值,都是1 ~ n^2之间的整数,第三行是吐槽星人,格式同上。
OUTPUT:对于每组数据,输出“Case x: y” ,x从1開始一直到Case,而y表示吸收的最大人数。
sample input:23 6 71 7 5 4 8 3 91 4 3 5 6 2 8 94 15 510 16 3 0 1 8 14 15 13 6 9 5 7 12 11 25 15 11 7 2 0
sample output:Case 1: 4Case 2: 3
/*改动于2014-4-28
这道题用到了:偏序集的Dilworth定理!
样例例如以下:
求一个序列的最长上升子序列。其最大值即等于不上升子序列的最小划分数。
这就涉及到组合数学中偏序集的 Dilworth定理,代码:
#include<cstdio>
#include<cstring>
#include<algorithm> using namespace std; int a[100],f[100],g[100]; int main(){
freopen("lmis.in","r",stdin);
freopen("lmis.out","w",stdout);
int n,i;
scanf("%d",&n);
memset(g,0x7f,sizeof(g));
memset(f,0,sizeof(f));
for(i=0;i<n;i++)
scanf("%d",&a[i++]); for(i=0;i<n;i++){
int k=lower_bound(g+1,g+n,a[i])-g;
f[i]=k+1;
g[k+1]=a[i];
} printf("%d\n",*max_element(f,f+n));
return 0;
}
*/
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int MAXN=250*250+10;
const int INF=9999999;
int b[MAXN],n,p,q;
int g[MAXN],id[MAXN],x[MAXN];//g[i]:ICS为i时候的最小下标,x[i]:lis值
int main()
{
//freopen("1.txt","r",stdin);
int T,kase=1;
while(scanf("%d",&T)==1){
kase=1;
while(T--)
{
int temp,len=0; scanf("%d%d%d",&n,&p,&q);
memset(id,0,sizeof(id));
for(int i=1;i<=p+1;i++)
{
scanf("%d",&temp);
id[temp]=i;
}
for(int i=0;i<q+1;i++)
{
scanf("%d",&temp);
if(id[temp])
b[len++]=id[temp];
} for(int i=0;i<=len;i++)
g[i]=INF; int ans=0;
for(int i=0;i<len;i++)
{
int k=lower_bound(g+1,g+len+1,b[i])-g;
x[i]=k;
g[k]=b[i];
ans=max(ans,x[i]);
}
printf("Case %d: %d\n",kase++,ans); }
}
return 0;
}
2014哈商大ICPC/ACM校赛解题报告的更多相关文章
- 2013 ACM/ICPC 成都网络赛解题报告
第三题:HDU 4730 We Love MOE Girls 传送门:http://acm.hdu.edu.cn/showproblem.php?pid=4730 水题~~~ #include < ...
- hdu 4762 && 2013 ACM/ICPC 长春网络赛解题报告
这次的答案是猜出来的,如果做得话应该是应该是一个几何概型的数学题: 答案就是:n/(m^(n-1)); 具体的证明过程: 1.首先枚举这M个点中的的两个端点,概率是:n*(n-1); 2.假设这个蛋糕 ...
- hdu 4763 && 2013 ACM/ICPC 长春网络赛解题报告
一个KMP的简单题 不过好久没用过这个东东了,今天写的时候花了很多时间: 只需要花点时间判断下所有的元素都相同的的情况就行了! #include<cstdio> #include<c ...
- ZROIDay4-比赛解题报告
ZROIDay4-比赛解题报告 扯闲话 感觉这个出题人的题做起来全都没感觉啊,今天又凉了,T1完全不知道什么意思,T2只会暴力,T3现在还不懂什么意思,真的太菜了 A 题意半天没搞懂爆零GG了,讲了一 ...
- ZROIDay3-比赛解题报告
ZROIDay3-比赛解题报告 瞎扯 从今天开始考试有点不在状态,可能是因为不太适应题目的原因,T1已经接近了思想但是没有想到状态转移,T2思考方向错误,T3不会打LCT,还是太菜了 A 考场上想到要 ...
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- 10.30 NFLS-NOIP模拟赛 解题报告
总结:今天去了NOIP模拟赛,其实是几道USACO的经典的题目,第一题和最后一题都有思路,第二题是我一开始写了个spfa,写了一半中途发现应该是矩阵乘法,然后没做完,然后就没有然后了!第二题的暴力都没 ...
- CH Round #56 - 国庆节欢乐赛解题报告
最近CH上的比赛很多,在此会全部写出解题报告,与大家交流一下解题方法与技巧. T1 魔幻森林 描述 Cortana来到了一片魔幻森林,这片森林可以被视作一个N*M的矩阵,矩阵中的每个位置上都长着一棵树 ...
- 2019 西电ACM校赛网络赛 题解
今年题目难度有较大提升,总体与往年类似,数学题居多.以下为我通过的部分题解. 赛题链接:http://acm.xidian.edu.cn/contest.php?cid=1053 A - 上帝视角 我 ...
随机推荐
- HDU 1061.Rightmost Digit-规律题 or 快速幂取模
Rightmost Digit Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)T ...
- 学习LSM(Linux security module)之三:Apparmor的前世今生和基本使用
感冒了,感觉一脑子浆糊,真是蛋疼. 先粗略讲一些前置知识. 一:MAC和DAC DAC(Discretionary Access Control),自主访问控制,是最常用的一类访问控制机制,意思为主体 ...
- Codeforces 920 G List Of Integers
题目描述 Let's denote as L(x,p)L(x,p) an infinite sequence of integers yy such that gcd(p,y)=1gcd(p,y)=1 ...
- [LOJ6433]最大前缀和
深刻感受到自己的水平和机房里的其他人相差甚远,他们都是随手秒这个题的... $n$很小,考虑状压DP 当一个序列在某个位置取到最大前缀和后,意味着如果把后面的数抽出来单独成序列,那么它的每个前缀和都$ ...
- 【tarjan求割顶】BZOJ2730-[HNOI2012]矿场搭建
[题目大意] 煤矿工地可以看成是由隧道连接挖煤点组成的无向图.为安全起见,希望在工地发生事故时所有挖煤点的工人都能有一条出路逃到救援出口处.于是矿主决定在某些挖煤点设立救援出口,使得无论哪一个挖煤点坍 ...
- Java高级架构师(一)第30节:把应用部署到Linux服务器上
- java前后端加密(转载)
最近做一个项目的安全渗透测评,测评人员发来一份测试报告,报告明确提出不允许明文参数传输,因为数据在传输的过程中可能被拦截,被监听,所以在传输数据的时候使用数据的原始内容进行传输的话,安全隐患是非常大的 ...
- Metesploit使用随笔
平时在工作中真正用到metesploit机会不多,偶尔也会用来做漏洞验证,但是每次使用的时候都需要花点时间回忆一下具体是怎么用的,因此索性记下来方便自己,以使用Nessus扫描YS的某个硬件设备发现的 ...
- 在sublime执行自定义脚本
[背景] 一般项目都会有一个预处理的脚本, 在发布,或者预览效果的时候,往往要先执行脚本. 想法来自editplus 习惯了editplus的同学,都知道,可以配置自定义执行的脚本. 一般我会把它配置 ...
- sql-server-on-linux-how-i-think-they-did-it : Anthony Nocentino's Blog
http://www.sqlservercentral.com/blogs/anthony-nocentinos-blog/2016/11/21/sql-server-on-linux-how-i-t ...