ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

CCF-CSP考试2021年4月第2题(202104-2邻域均值)

2021-09-12 12:03:44  阅读:354  来源: 互联网

标签:202104 600 int 元素 邻域 twoArray 2021 CCF 范围


题目:计算机软件能力认证考试系统

特别注意评测用例规模与约定!

        70 的测试数据满足 n≤100、r≤10。

        全部的测试数据满足 0<n≤600、0<r≤100 且 2≤t<L≤256。

思路:

刚开始直接用四层for循环遍历,暴力求解,只得了70分,运行时间超时,需要对代码的循环进行优化。

通过分析可以知道,矩阵中同一行元素邻域的纵向范围是一样的,变化的是横向范围,并且同一行相邻的两个元素的邻域范围重合度非常高,所以我们可以用同一行的前一个元素的邻域来避免重复计算。

在同一行中,元素邻域的范围经历扩张、大小不变位置平移以及收缩这三个阶段。(j是元素在某一行的位置,r是邻域的范围)

j-r<=0时,(相比于前一个元素)邻域扩张,这时需要考虑j+r是否超过了矩阵的边界:

        若j+r超过了矩阵边界,邻域范围不变(特殊情况);否则,邻域将扩张。

j-r>0时,(相比于前一个元素)邻域有平移和收缩两种情况:

        当j+r<=n-1时,邻域大小保持不变位置向右平移;当j+r>n-1时,邻域开始收缩。

代码1:暴力遍历(70分)

#include<iostream>
using namespace std;
int twoArray[600][600];
int main(){
	// n是矩阵的边长,L是最大灰度值,r指明附近元素的具体范围,t是阈值 
	int n,L,r,t;
	cin>>n>>L>>r>>t;
	// count计数暗区域总数 
	int count=0;
	for(int i=0;i<n;i++){
		for(int j=0;j<n;j++){
			cin>>twoArray[i][j];
		}
	}
	
	for(int i = 0;i<n;i++){
		for(int j=0;j<n;j++){
			double temp = 0; // 记录某元素附近所有元素的像素和 
			int num = 0; // 记录某元素附近所有元素个数 
			// k1到k4表示某元素附近元素的边界 
			int k1=0,k2=0,k3=0,k4=0;
			if(j-r<0){ k1=0; }  // 横向左 
			else if(j-r>=0){ k1=j-r; }
			
			if(j+r>=n){ k2=n-1; } // 横向右 
			else if(j+r<n){ k2 = j+r; }
			
			if(i-r<0){ k3=0; } // 纵向上 
			else if(i-r>=0){ k3=i-r; }
			
			if(i+r>=n){ k4=n-1; } // 纵向下 
			else if(i+r<n){ k4=i+r; }
			
			for(int m=k3;m<=k4;m++){
				for(int n=k1;n<=k2;n++){
					temp+=twoArray[m][n];
					num++;
				}
			}
			if((double)(temp/num) <= t){
				count++;
			}
		}
	}
	cout<<count;
}

代码2:优化后(100分)

#include<iostream>
using namespace std;
int twoArray[600][600];
int main(){
	int n,L,r,t;
	cin>>n>>L>>r>>t;
	for(int i=0;i<n;i++){
		for(int j=0;j<n;j++){
			cin>>twoArray[i][j];
		}
	}
	int count = 0;
	// 计算均值
	for(int i=0;i<n;i++){
		// 先算出每行第一个元素的邻域值之和
		int start_y=i-r>0?i-r:0;
		int bound_y=i+r>n-1?n-1:i+r;
		int bound_x=0+r>n-1?n-1:r; 
		double sum = 0;
		int num = 0;
		for(int j=start_y;j<=bound_y;j++){
			for(int k=0;k<=bound_x;k++){
				sum+=twoArray[j][k];
				num++;
			}
		}
		if((sum/num)<=t){
			count++;
		}
		// 同一行元素的 start_y bound_y都是一样的,不一样的是横向的边界 
		for(int j=1;j<n;j++){
			if(j-r<=0){
				if(j+r<=n-1){
					// 邻域范围在扩张 
					for(int k=start_y;k<=bound_y;k++){
						sum+=twoArray[k][j+r];
						num++;
					}
				}
			}
			else if(j-r>0){
				if(j+r<=n-1){
					// 邻域范围不变 
					for(int k=start_y;k<=bound_y;k++){
						sum=sum+twoArray[k][j+r]-twoArray[k][j-r-1];
						// num值不变 
					}
				}
				else{
					// 邻域范围在收缩 
					for(int k=start_y;k<=bound_y;k++){
						sum-=twoArray[k][j-r-1];
						num--;
					}
				}
			}
			if((sum/num)<=t){
				count++;
			}
		}
	} 
	cout<<count;
}

标签:202104,600,int,元素,邻域,twoArray,2021,CCF,范围
来源: https://blog.csdn.net/ly999999/article/details/120248569

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有