【Week 11 作业】A - 必做题11-1、B - 必做题11-2、C - 必做题11-3、D - 必做题11-4、E - 选做题11-1 东东与 ATM、F - 选做题11-2 东东开车了

A - 必做题11-1

题意:

蒜头君从现在开始工作,年薪 NN 万。他希望在蒜厂附近买一套 6060 平米的房子,现在价格是 200200 万。假设房子价格以每年百分之 KK 增长,并且蒜头君未来年薪不变,且不吃不喝,不用交税,每年所得 NN 万全都积攒起来,问第几年能够买下这套房子?(第一年年薪 NN 万,房价 200200 万)

Input

一行,包含两个正整数N(10≤N≤50),K(1≤K≤20),中间用单个空格隔开。

Output

如果在第 2020 年或者之前就能买下这套房子,则输出一个整数 MM,表示最早需要在第 MM 年能买下;否则输出"Impossible"。

输出时每行末尾的多余空格,不影响答案正确性

Sample Input

50 10

Sample Output

8

思路做法:

用while循环,每次更新存款和房价,并比较大小,在20次循环内若存款超过房价,则输出循环轮数,否则超过20次循环,输出不可能

总结:

输出的循环轮数从1开始

代码:

#include <stdio.h>

int main(){
	int n, k; scanf("%d%d", &n, &k);
	int money = n, y = 1;  // 存款 年 
	double price = 200;  // 房价 
	bool isOk = true;
	while(money < price){
		money += n;
		price += price*k/100;
		y++;
		if(y > 20){
			isOk = false;
			break;
		}
//		printf("%d %lf %d\n", money, price, y);
	}
	if(!isOk) printf("Impossible\n");
	else printf("%d\n", y);
	return 0;
}

B - 必做题11-2

题意:

蒜头君的班级里有 n 2 n^2 n2 个同学,现在全班同学已经排列成一个 n∗n 的方阵,但是老师却临时给出了一组新的列队方案

为了方便列队,所以老师只关注这个方阵中同学的性别,不看具体的人是谁

这里我们用 0 表示男生,用 1 表示女生

现在蒜头君告诉你同学们已经排好的方阵是什么样的,再告诉你老师希望的方阵是什么样的

他想知道同学们已经列好的方阵能否通过顺时针旋转变成老师希望的方阵

不需要旋转则输出 0

顺时针旋转 90° 则输出 1

顺时针旋转 180° 则输出 2

顺时针旋转 270° 则输出 3

若不满足以上四种情况则输出 −1

若满足多种情况,则输出较小的数字

Input

第一行为一个整数 n

接下来的 n 行同学们已经列好的 01 方阵;

再接下来的 n 行表示老师希望的的 01 方阵。

Output

输出仅有一行,该行只有一个整数,如题所示。

数据范围

对于 100% 的数据中,1≤n≤20

输出时每行末尾的多余空格,不影响答案正确性

Sample Input

4
0 0 0 0
0 0 0 0
0 1 0 0
0 0 0 0
0 0 0 0
0 1 0 0
0 0 0 0
0 0 0 0

Sample Output

1

思路做法:

循环,每次先判断是否以转变为希望阵型,若没有则继续转换,直到转换成希望阵型或循环超过3次
旋转阵型用2层循环,由外到内为一层,在每一层把每个元素顺时针旋转

总结:

主要是rot函数内4个角上的元素如何同时位移,以及坐标的选取

代码:

#include <stdio.h>

const int N = 22;
int n, a[N][N], b[N][N];

void rot(){
	// 顺时针旋转90度
	int m = n / 2;
	for(int i = 0; i < m; ++i){
		for(int j = i; j < n-1-i; ++j){
			int temp = a[i][j];
			a[i][j] = a[n-j-1][i];
			a[n-j-1][i] = a[n-i-1][n-j-1];
			a[n-i-1][n-j-1] = a[j][n-i-1];
			a[j][n-i-1] = temp;
		}
	}
}

bool equal(){
	for(int i = 0; i < n; ++i)
		for(int j = 0; j < n; ++j)
			if(a[i][j] != b[i][j]) return false;
	return true;
}

int main(){
	scanf("%d", &n);
	for(int i = 0; i < n; ++i)
		for(int j = 0; j < n; ++j)
			scanf("%d", &a[i][j]);
	for(int i = 0; i < n; ++i)
		for(int j = 0; j < n; ++j)
			scanf("%d", &b[i][j]);
	int ans = 0;
	while(1){
		if(equal()){
			printf("%d\n", ans); return 0;
		}else{
			if(ans == 3) break;
			rot(); ans++;
		}
	}
	printf("-1\n");
	return 0;
}

C - 必做题11-3

题意:

Julius Caesar 曾经使用过一种很简单的密码。对于明文中的每个字符,将它用它字母表中后 55 位对应的字符来代替,这样就得到了密文。比如字符’A’用’F’来代替。如下是密文和明文中字符的对应关系。

密文A B C D E F G H I J K L M N O P Q R S T U V W X Y Z

明文V W X Y Z A B C D E F G H I J K L M N O P Q R S T U

你的任务是对给定的密文进行解密得到明文。

你需要注意的是,密文中出现的字母都是大写字母。密文中也包括非字母的字符,对这些字符不用进行解码。

Input

一行,给出密文,密文不为空,而且其中的字符数不超过 200200。

Output

输出一行,即密文对应的明文。

输出时每行末尾的多余空格,不影响答案正确性

Sample Input

NS BFW, JAJSYX TK NRUTWYFSHJ FWJ YMJ WJXZQY TK YWNANFQ HFZXJX

Sample Output

IN WAR, EVENTS OF IMPORTANCE ARE THE RESULT OF TRIVIAL CAUSES

思路做法:

对给定的一个字符,若它不属于字母,则原样输出,否则,计算明文字母再输出

总结:

代码中计算了密文到明文字母的映射,或者不用也可以

代码:

#include <string>
#include <iostream>
#include <map>
using namespace std;

string code;
map<char, char> dict;  // 由密文转明文的字典 

int main(){
	getline(cin, code);
	for(int i = 0; i < 26; ++i) dict['A'+i] = (i+21)%26+'A';
	for(int i = 0; i < code.size(); ++i){
		if(code[i] < 'A' || code[i] > 'Z') cout << code[i];
		else cout << dict[code[i]];
	}
	cout << endl;
	return 0;
} 

D - 必做题11-4

题意:

东东和他的女朋友(幻想的)去寿司店吃晚餐(在梦中),他发现了一个有趣的事情,这家餐厅提供的 n 个的寿司被连续的放置在桌子上 (有序),东东可以选择一段连续的寿司来吃

东东想吃鳗鱼,但是东妹想吃金枪鱼。核 平 起 见,他们想选择一段连续的寿司(这段寿司必须满足金枪鱼的数量等于鳗鱼的数量,且前一半全是一种,后一半全是另外一种)我们用1代表鳗鱼,2代表金枪鱼。

比如,[2,2,2,1,1,1]这段序列是合法的,[1,2,1,2,1,2]是非法的。因为它不满足第二个要求。

东东希望你能帮助他找到最长的一段合法寿司,以便自己能吃饱。

Input

输入:
第一行:一个整数n(2≤n≤100000),寿司序列的长度。
第二行:n个整数(每个整数不是1就是2,意义如上所述)

Output

输出:一个整数(代表东东可以选择的最长的一段连续的且合法的寿司)

Sample Input

7
2 2 2 1 1 2 2

6
1 2 1 2 1 2

9
2 2 1 1 1 2 2 2 2

Sample Output

4

2

6

思路做法:

要想找到最长寿司,理论上要遍历全部可能。从左到右开始遍历,对于一个位置,从该位置同时向两边探索,用2个变量le,ri记录区间的左右端点,并尽可能地更新(如果可以的话),用计算的区间长度来更新ans,遍历完所有可能的区间都找过了,ans即为答案

总结:

这种做法比较简单易懂,复杂度为 O ( n 2 ) O(n^2) O(n2)

代码:

#include <stdio.h>

const int N = 1e5+50;
int n, a[N];

int getLen(int le, int ri){
	int i = le, j = ri;
	if(a[le] == a[ri]) return 0;
	while(i >= 0 && j < n && a[i] == a[le] && a[j] == a[ri]){
		i--; j++;
	}
	return j - i - 1;
}

int main(){
	int ans = 0; scanf("%d", &n);
	for(int i = 0; i < n; ++i) scanf("%d", &a[i]);
	for(int i = 0; i < n-1; ++i){
		int len = getLen(i, i+1);
		if(len > ans) ans = len;
	}
	printf("%d\n", ans);
	return 0;
}

E - 选做题11-1 东东与 ATM

题意:

一家银行计划安装一台用于提取现金的机器。
机器能够按要求的现金量发送适当的账单。
机器使用正好N种不同的面额钞票,例如D_k,k = 1,2,…,N,并且对于每种面额D_k,机器都有n_k张钞票。
例如,
N = 3,
n_1 = 10,D_1 = 100,
n_2 = 4,D_2 = 50,
n_3 = 5,D_3 = 10
表示机器有10张面额为100的钞票、4张面额为50的钞票、5张面额为10的钞票。
东东在写一个 ATM 的程序,可根据具体金额请求机器交付现金。
注意,这个程序计算程序得出的最大现金少于或等于可以根据设备的可用票据供应有效交付的现金。

Input

程序输入来自标准输入。 输入中的每个数据集代表特定交易,其格式为:Cash N n1 D1 n2 D2 … nN DN其中0 <= Cash <= 100000是所请求的现金量,0 <= N <= 10是 纸币面额的数量,0 <= nk <= 1000是Dk面额的可用纸币的数量,1 <= Dk <= 1000,k = 1,N。 输入中的数字之间可以自由出现空格。 输入数据正确。

Output

对于每组数据,程序将在下一行中将结果打印到单独一行上的标准输出中。

Sample Input

735 3  4 125  6 5  3 350
633 4  500 30  6 100  1 5  0 1
735 0
0 3  10 100  10 50  10 10

Sample Output

735
630
0
0

hint

第一个数据集指定一笔交易,其中请求的现金金额为 735。 机器包含3种面额的纸币:4张钞票 125、6张钞票 5和3张钞票 350。 机器可以交付所需现金的确切金额。

在第二种情况下,机器的票据供应不能满足所要求的确切现金数量。 可以交付的最大现金为 630。 请注意,在机器中组合钞票以匹配交付的现金有多种可能性。

在第三种情况下,机器是空的,没有现金交付。 在第四种情况下,请求的现金金额为 0,因此机器不交付现金。

思路做法:

多重背包问题,但是要用二进制拆分转化为01背包来优化,不然会超时

总结:

转化为01背包后,原本的二维数组f可以用一维存储(逆向)
二进制优化后的所需空间要变大,代码中按*10的比例变化

代码:

#include <stdio.h>
#include <algorithm>
using namespace std;

const int N = 1e6+50;

int f[N], w[15], c[15], ww[150];

int main(){
	int cash, n;
	while(~scanf("%d%d", &cash, &n)){
		for(int i = 1; i <= n; ++i){
			scanf("%d%d", &c[i], &w[i]);
		}
		for(int i = 0; i < N; ++i) f[i] = 0;
		int cnt = 0;
		for(int i = 1; i <= n; ++i){
			int t = c[i];
			for(int k = 1; k <= t; k <<= 1){
				cnt++;
				ww[cnt] = k * w[i];
				t -= k;
			}
			if(t > 0){
				cnt++;
				ww[cnt] = t * w[i];
			}
		}
		for(int i = 1; i <= cnt; ++i){
			for(int j = cash; j >= 0; j--){
				if(j-ww[i] >= 0)
					f[j] = max(f[j], f[j-ww[i]]+ww[i]);
			}
		}
		printf("%d\n", f[cash]);
	}
	return 0;
}

F - 选做题11-2 东东开车了

题意:

东东开车出去泡妞(在梦中),车内提供了 n 张CD唱片,已知东东开车的时间是 n 分钟,他该如何去选择唱片去消磨这无聊的时间呢

假设:

CD数量不超过20张
没有一张CD唱片超过 N 分钟
每张唱片只能听一次
唱片的播放长度为整数
N 也是整数
我们需要找到最能消磨时间的唱片数量,并按使用顺序输出答案(必须是听完唱片,不能有唱片没听完却到了下车时间的情况发生)

本题是 Special Judge

Input

多组输入

每行输入第一个数字N, 代表总时间,第二个数字 M 代表有 M 张唱片,后面紧跟 M 个数字,代表每张唱片的时长 例如样例一: N=5, M=3, 第一张唱片为 1 分钟, 第二张唱片 3 分钟, 第三张 4 分钟

所有数据均满足以下条件:

N≤10000
M≤20

Output

输出所有唱片的时长和总时长,具体输出格式见样例

Sample Input

5 3 1 3 4
10 4 9 8 4 2
20 4 10 5 7 4
90 8 10 23 1 2 3 4 5 7
45 8 4 10 44 43 12 9 8 2

Sample Output

1 4 sum:5
8 2 sum:10
10 5 4 sum:19
10 23 1 2 3 4 5 7 sum:55
4 10 12 9 8 2 sum:45

思路做法:

01背包问题,但是需要输出选择的背包,或者说路径
因此用二维数组b来记录, b [ i ] [ j ] b[i][j] b[i][j]表示第i个唱片在时长为j的时间里放不放(bool 虽然代码中用的int)
动态规划的循环中要更新b数组,最后反着找路径,从第m个唱片,时长为n开始,若 b [ i ] [ j ] b[i][j] b[i][j]为真,则说明唱片i要放,记录下来,时长相应的要减少 a [ i ] a[i] a[i],否则,唱片i并没有放,那么还是该时长不变,看i-1个唱片的情况。
遍历完,记录的唱片顺序是反的,逆序输出即可。

总结:

重点在于路径的记录与寻找,另外一组数据可能有多组解,都是允许的

代码:

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

const int M = 25, N = 1e4+50;
int f[N], a[M], b[M][N];
vector<int> v;

int main(){
	int n, m;
	while(~scanf("%d%d", &n, &m)){
		for(int i = 1; i <= m; ++i) scanf("%d", &a[i]);
		for(int i = 0; i <= n; ++i) f[i] = 0;
		for(int i = 1; i <= m; ++i)
			for(int j = 0; j <= n; ++j)
				b[i][j] = 0;
		v.clear();
		
		for(int i = 1; i <= m; ++i){
			for(int j = n; j >= 0; j--){
				if(j - a[i] >= 0){
					if(f[j-a[i]]+a[i] > f[j]){
						f[j] = f[j-a[i]]+a[i];
						b[i][j] = 1;
					}
				}
			}
		}
		// 回溯 
		for(int i = m, j = n; i >= 1 && j > 0; i--){
			if(b[i][j]){
//				printf("%d ", a[i]);
				v.push_back(a[i]);
				j -= a[i];
			}
		}
		for(int i = v.size()-1; i >= 0; i--) printf("%d ", v[i]);
		printf("sum:%d\n", f[n]);
	}
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在第4周的可重复研究项目中,我将继续探索如何使用开放源代码工具和技术来实现可重复性和透明度。 首先,我将继续使用版本控制系统(如Git),以便跟踪我研究项目中的所有更改和改进。这将确保我能够回溯到每个版本的数据和代码,并对项目进行可重复性验证。在本周内,我还将学习更多关于Git分支和合并的知识,以便更好地组织和管理我的项目。 另外,我还将使用Jupyter Notebook来记录我的实验过程和结果。Jupyter Notebook提供了一个互动环境,可以将代码、文档和图形化结果结合在一起,使得我的研究成果更加易于理解和重现。我会确保我的Notebook中包含了所有必要的步骤和解释,以便他人能够准确地复现我的研究。 为了进一步提高可重复性,我还将采取一些数据预处理和清洗的措施。这些措施包括去除异常值、处理缺失数据和标准化数据等。我将确保我的数据处理过程明确记录,并提供相应的代码和文档,以便他人能够按照相同的步骤进行处理。 最后,我还计划使用容器化技术(如Docker)来实现我的研究项目的可移植性。通过将我的环境和依赖项封装在一个容器中,我可以确保其他人能够在不同的计算机和操作系统上轻松地运行我的代码和分析。 综上所述,第4周的可重复研究项目将继续探索一系列工具和技术,旨在提高我的研究项目的可重复性和透明度。通过使用版本控制系统、Jupyter Notebook、数据处理和清洗措施以及容器化技术,我将确保我的研究成果可以被其他人准确地重现和验证。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

容嬷嬷当年一枝花

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值