求面积 (坐标叉积公式+凹多边形面积-坐标公式)

求面积(AREA

给出一个简单多边形(没有缺口),它的边要么是垂直的,要么是水平的。要求计算多边形的面积。

多边形被放置在一个X-Y的卡笛尔平面上,它所有的边都平行于两条坐标轴之一。然后按逆时针方向给出各顶点的坐标值。所有的坐标值都是整数(因此多边形的面积也为整数)。

 

输入

输入文件第一行给出多边形的顶点数n(n≤100)。接下来的几行每行给出多边形一个顶点的坐标值X和Y(都为整数并且用空格隔开)。顶点按逆时针方向逐个给出。并且多边形的每一个顶点的坐标值-200≤x,y≤200。多边形最后是靠从最后一个顶点到第一个顶点画一条边来封闭的。

 

输出

输出文件仅有一行包含一个整数,表示多边形的面积。

 

样例

AREA.IN

10

0 0

4 0

4 1

3 1

3 3

2 3

2 2

1 2

1 3

0 3

 

AREA.OUT

9

叉积公式 A X B= x1y2-x2y1=S(平行四边形)=2S(三角形)=2*|a|*|b|*sinaC




#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<cmath>
#include<functional>
#include<algorithm>
#include<queue>
using namespace std;
#define MAXN (100+10)
class _vector
{
public:
	int x,y;
	_vector():x(0),y(0){}
	_vector(int _x,int _y):x(_x),y(_y){}
	friend int operator*(const _vector a,const _vector b)
	{
		return a.x*b.y-a.y*b.x;
	}

}node[MAXN];

istream& operator>>(istream& in,_vector& a)
{
	in>>a.x>>a.y;
	return in;
}
int n;
int main()
{
	freopen("area.in","r",stdin);
	freopen("area.out","w",stdout);
	scanf("%d",&n);
	for (int i=1;i<=n;i++) cin>>node[i];
	node[n+1]=node[1];
	int ans=0;
	for (int i=1;i<=n;i++)
		ans+=node[i]*node[i+1];
	ans=abs(ans);

	printf("%dn",int(round(double(ans)/2)));
//	while (1);
	return 0;


}

连续背包 (背包套背包)

连续背包(bag

【问题描述】

从T组物品中选出一些物品,放入背包中,求剩余空间的最小值。

限制条件:从每组物品中挑选物品必须要选取连续的一段。就是说,如果这组物品共有n个: 物品1、物品2、物品3、…、物品n,那么只能选取物品i、物品i+1、…、物品j,其中1<=i<=j<=n,或者不选。

【输入】

第一行为两个用空格隔开的正整数v和T。表示背包的空间和物品的组数。

接下来有T行,每行先是一个正整数ni,表示这组物品有ni个,然后ni个正整数,表示每个物品的大小。

【输出】

 仅一个数,表示剩余空间的最小值。

【输入输出样例】

bag.in

100 3

3 7 6 8

2 80 70

4 101 108 103150

 

bag.out

6

 

【输入输出样例解释】

 第1组选6、8,第2组选80,第3组不选。

【限制】

60%的数据满足:1 <= ni <= 10

100%的数据满足:1 <= ni <= 100,1<=v<=5000,1<=T<=10

 

对每一段背包

再对各段背包

注意单步中Vmax=V[i-1]+sum[ni]

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<cmath>
#include<functional>
#include<algorithm>
using namespace std;
#define MAXNI (100+10)
#define MAXV (500000+10)
#define MAXN (10+10)
int T,n,v,sum[MAXNI]={0},a[MAXNI],A[MAXN][MAXV]={0};
bool f[MAXN][MAXV]={0};
int main()
{
	freopen("bag.in","r",stdin);
	freopen("bag.out","w",stdout);
	scanf("%d%d",&v,&T);
//	v=v*10;
	f[0][0]=1;
	for (int i=1;i<=T;i++)
	{
		scanf("%d",&n);
		sum[0]=0;
		for (int j=1;j<=n;j++) {scanf("%d",&a[j]);if (a[j]>v/*/10*/) a[j]=0; sum[j]=sum[j-1]+a[j];}
		//for (int j=1;j<=n;j++) printf("%d ",sum[j]);
		for (int k=0;k<=v+sum[n];k++)
			for (int j=n;j>=0;j--)
			{
				if (k-sum[j]<0) continue;
				if (f[i-1][k-sum[j]])
				{
					f[i][k]=1;
					A[i][k]=j;
					break;
				}
			}
		for (int k=0;k<=v+sum[n];k++)
			for (int j=A[i][k]-1;j>=1;j--)
			{
				f[i][k-sum[j]]=f[i][k]||f[i][k-sum[j]];
			}



		//cout<<endl;
	}
//	v/=10;
	int i=v;
	while (!f[T][i])
	{
		i--;
	}
	printf("%dn",v-i);
/*
	for (int i=0;i<=T;i++)
	{
		for (int j=0;j<=v;j++)
			if (f[i][j]) cout<<i<<','<<j<<' ';
		cout<<'n';
	}

	while (1);
*/	return 0;
}

水灾 (BFS-先洪水后寻路)

水灾(sliker

大雨应经下了几天雨,却还是没有停的样子。ksy刚从外地回来,知道不久除了自己家,其他的地方都将会被洪水淹没。

ksy的老家可以用一个N*M的地图表示,地图上有五种符号:“. * X D S”。其中“X”表示石头,水和人都不能从上面经过。“.”表示平原,ksy和洪水都可以经过。“*”表示洪水开始地方(可能有多个地方开始发生洪水)。“D”表示ksy的家。“S”表示ksy现在的位置。

ksy每分钟可以向相邻位置移动,而洪水将会在ksy移动之后把相邻的没有的土地淹没(从已淹没的土地)。

求ksy回答家最少时间。如他回不了家,被淹死输出KAKTUS。

Input

3 3

D.*

...

.S.

Output

3

Input

3 3

D.*
...

..S

Output

KAKTUS

Input

3 6

D...*.

.X.X..

....S.

Output

6

因为第i秒走后,所到达的点不能有Flood

所以必须在之前Flood,然后再往下找

显然柯黑再同一个地方停留不优

故只要存储到达一个点的最短时间

注意C++中构造函数的写法

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<cmath>
#include<functional>
#include<algorithm>
#include<queue>
using namespace std;
#define MAXN (50+10)

struct node
{
	int x,y,t;
	node():x(0),y(0),t(0){}
	node(int _x,int _y,int _t):x(_x),y(_y),t(_t){/*cout<<x<<' '<<y<<' '<<t<<endl;*/}
}start,end;
/*
node made_node(int i,int j,int t)
{
	node now;
	now.x=i;
	now.y=j;
	now.t=t;
	return now;
}
*/

int n,m;
bool map[MAXN][MAXN],b[MAXN][MAXN];
char s[MAXN];
queue<node> flood,q;
bool inside(int x,int y)
{
	if (x>=1&&x<=n&&y>=1&&y<=m) return true;
	return false;
}
bool bfs()
{
	int l=-1;
	while (!q.empty())
	{
		node now=q.front();
//		cout<<now.x<<' '<<now.y<<endl;
		q.pop();
		if (now.t>l)
		{
			int size=flood.size();
			while (size)
			{
				node nowf=flood.front();
				flood.pop();
				int x=nowf.x,y=nowf.y;
				if (x>1&&b[x-1][y])
				{
					flood.push(node(x-1,y,now.t));
					map[x-1][y]=b[x-1][y]=0;
				}
				if (x<n&&b[x+1][y])
				{
					flood.push(node(x+1,y,now.t));
					map[x+1][y]=b[x+1][y]=0;
				}
				if (y>1&&b[x][y-1])
				{
					flood.push(node(x,y-1,now.t));
					map[x][y-1]=b[x][y-1]=0;
				}
				if (y<m&&b[x][y+1])
				{
					flood.push(node(x,y+1,now.t));
					map[x][y+1]=b[x][y+1]=0;
				}

				size--;
			}
			l++;
		}
		int x=now.x,y=now.y;
//		if (!map[x][y]) continue;
		if (x>1&&map[x-1][y])
		{
			if (x-1==end.x&&y==end.y){end.t=now.t+1; return true;}
			q.push(node(x-1,y,now.t+1));
			map[x-1][y]=0;
		}
		if (x<n&&map[x+1][y])
		{
			if (x+1==end.x&&y==end.y){end.t=now.t+1; return true;}
		 	q.push(node(x+1,y,now.t+1));
			map[x+1][y]=0;
		}
		if (y>1&&map[x][y-1])
		{
			if (x==end.x&&y-1==end.y){end.t=now.t+1; return true;}
			q.push(node(x,y-1,now.t+1));
			map[x][y-1]=0;
		}
		if (y<m&&map[x][y+1])
		{
			if (x==end.x&&y+1==end.y){end.t=now.t+1; return true;}
			q.push(node(x,y+1,now.t+1));
			map[x][y+1]=0;
		}





	}
	return false;
}

int main()
{
	freopen("sliker.in","r",stdin);
	freopen("sliker.out","w",stdout);
	scanf("%d%d",&n,&m);
	memset(map,1,sizeof(map));
	memset(b,1,sizeof(b));

	for (int i=1;i<=n;i++)
	{
		scanf("%s",s);
		for (int j=0;j<m;j++)
		{
			if (s[j]=='S')
			{
				start=node(i,j+1,0);
				q.push(start);
			}
			if (s[j]=='D')
			{
				end=node(i,j+1,0);
				b[i][j+1]=0;
			}
			if (s[j]=='X')
			{
				map[i][j+1]=0;
				b[i][j+1]=0;
			}
			if (s[j]=='*')
			{
				map[i][j+1]=0;
				b[i][j+1]=0;
				flood.push(node(i,j+1,0));
			}

		}
	}
/*	for (int i=1;i<=n;i++)
		for (int j=1;j<=m;j++)
			if (map[i][j]) cout<<"map "<<i<<' '<<j<<endl;
	cout<<"end"<<end.x<<' '<<end.y;
*/

	if (bfs()) printf("%dn",end.t);
	else printf("KAKTUSn");


//	while (1);
	return 0;
}

旅行 (分别考虑不同方向曼哈顿距离和)

第二题:旅行(journey)

时间限制:1秒

内存限制:256MB

输入:journey.in

输出:journey.out

问题描述

给定一个n行m列的字符矩阵,’.’代表空地,’X’代表障碍。移动的规则是:每秒钟以上下左右四个方向之一移动一格,不能进入障碍。计算:在空地中随机选择起点和终点(可以重合,此时最短耗时为0),从起点移动到终点最短耗时的平均值。

每一行每一列至多有1个障碍,并且障碍不在对角线方向相邻。以下矩阵是不合法的:

.X

X.

输入

第一行两个整数n, m。。

接下来n行,每行m个字符’.’或’X’。在50%的数据中,保证每个字符都是’.’。

输出

平均耗时,保留4位小数,四舍五入。数据保证在范围之内输出一致,Ans是标准答案。

样例输入

2 2

..

.X

样例输出

0.8889

 

 

 

 

 

yle�� sth�p�ive;top:3.0pt;mso-text-raise:-3.0pt'>。所有信息按照从左到右的顺序给出。在50%的数据中,。

输出

第一行:出列的总对数k。接下来输出k行,每行是两个整数。按跳舞顺序输出,两个整数代表这一对舞伴的编号(按输入顺序从左往右1至n编号)。请先输出较小的整数,再输出较大的整数。

样例输入

4

BGBG

4 2 4 3

样例输出

2

3 4

1 2

样例输入

4

BGBB

1 1 2 3

样例输出

1

1 2

解1:

Dp出所有平地曼哈顿距离和(全T)

#include<cstdio>
#include<cstring>
#include<queue>
#include<cmath>
#include<cstdlib>
#include<iostream>
#include<functional>
#include<algorithm>
#include<stack>
using namespace std;
#define MAXN (1000+10)
#define DELTA (0.00000001)
#define F (10000000)
int n,m;
char s[2000];
int a[MAXN],b[MAXN];
long long f[MAXN][MAXN];
int calc(int l,int r,int n)
{
	return 2*(l-1)*(n-r);
}


int main()
{
	freopen("journey.in","r",stdin);
	freopen("journey.out","w",stdout);

/*	long long ans=0;
	scanf("%d%d",&n,&m);
	for (int n=1;n<=10;n++)
	{
	for (int m=1;m<=10;m++)
	{
	long long ans=0;
	for (int i=1;i<=n;i++)
		for (int j=1;j<=m;j++)
			for (int k=1;k<=n;k++)
				for (int l=1;l<=m;l++)
					ans+=abs(i-k)+abs(j-l);

	printf("%d ",ans);
	}
	cout<<endl;
}
*/

	memset(f,0,sizeof(f));
	memset(a,0,sizeof(a));
	memset(b,0,sizeof(b));

	long long tot=0;
	scanf("%d%d",&n,&m);
	for (int i=1;i<=n;i++)
	{
		scanf("%s",s);
		for (int j=0;j<m;j++)
			if (s[j]=='X')
			{
				tot++;
				a[i]=j+1;
				b[j+1]=i;
				break;
			}
	}
	tot=n*m-tot;

//	for (int i=1;i<=n;i++) cout<<a[i]<<' ';


	for (int i=1;i<=n;i++)
		for (int j=1;j<=m;j++)
		{
			f[i][j]=f[i-1][j]+f[i][j-1]-f[i-1][j-1]+i*j*(i+j-2)+(i-1)*(j-1)*(i+j);
			if (a[i]==j) f[i][j]-=+i*j*(i+j-2);
			else
			{
				for (int k=1;k<=i;k++)
				 if (a[k]&&a[k]<=j)
				 f[i][j]-=2*(i-k+j-a[k]);


				if (a[i]&&a[i]<j)
					 f[i][j]-=(i-1)*(i+2*(j-a[i]));
				if (b[j]&&b[j]<i)
					 f[i][j]-=(j-1)*(j+2*(i-b[j]));
				if (a[i]&&a[i]<j&&b[j]&&b[j]<i)
					f[i][j]+=2*((j-a[i])+(i-b[j]));

			}
//			cout<<f[i][j]<<endl;
		}
/*
	for (int i=0;i<=n;i++){
		for (int j=0;j<=m;j++)
			cout<<f[i][j]<<' ';
		cout<<'n';
	}*/

	long long ans=f[n][m];
//	cout<<ans<<endl;
	for (int i=1;i<=n;i++)
		if (a[i])
		{
			ans+=2*calc(a[i],a[i],m);
			if (i>1&&a[i-1])
			{
				int j=i-1;
				if (a[j]<a[j+1])
				{
					while (j>0&&a[j]&&a[j]<a[j+1]) {ans+=2*calc(a[j],a[i],m);j--;}
				}/*
				else if (a[j]>a[j+1])
				{
					while (j>0&&a[j]>a[j+1]) {ans+=2*calc(a[i],a[j],m);j--;}
				}	*/
			}
			if (i<n&&a[i+1])
			{
				int j=i+1;
				if (a[j]<a[j-1])
				{
					while (j<=n&&a[j]&&a[j]<a[j-1]) {ans+=2*calc(a[j],a[i],m);j++;}
				}/*
				else if (a[j]>a[j-1])
				{
					while (j<=n&&a[j]>a[j-1]) {ans+=2*calc(a[i],a[j],m);j++;}
				}*/
			}
		}
	for (int i=1;i<=m;i++)
		if (b[i])
		{
			ans+=2*calc(b[i],b[i],n);
			if (i>1&&b[i-1])
			{
				int j=i-1;
				if (b[j]<b[j+1])
				{
					while (j>0&&b[j]&&b[j]<b[j+1]) {ans+=2*calc(b[j],b[i],n);j--;}
				}/*
				else if (b[j]>b[j+1])
				{
					while (j>0&&b[j]>b[j+1]) {ans+=2*calc(b[i],b[j],n);j--;}
				}*/
			}
			if (i<m&&b[i+1])
			{
				int j=i+1;
				if (b[j]<b[j-1])
				{
					while (j<=m&&b[j]&&b[j]<b[j-1]) {ans+=2*calc(b[j],b[i],n);j++;}
				}/*
				else if (b[j]>b[j-1])
				{
					while (j<=m&&b[j]>b[j-1]) {ans+=2*calc(b[i],b[j],n);j++;}
				}*/
			}
		}
//	cout<<ans<<endl;
	double _ans=double(ans)/double(tot)/double(tot);
	printf("%.4lfn",_ans);


	while (1);

	return 0;


}

正解

找一格,向前后搜 曼哈顿距离和分别考虑纵横方向

#include<cstdio>
#include<cstring>
#include<queue>
#include<cmath>
#include<cstdlib>
#include<iostream>
#include<functional>
#include<algorithm>
#include<stack>
using namespace std;
#define MAXN (1000+10)
#define DELTA (0.00000001)
#define F (10000000)
int n,m;
char s[2000];
int a[MAXN],b[MAXN];
//long long f[MAXN][MAXN];
int calc(int l,int r,int n)
{
	return 2*(l-1)*(n-r);
}


int main()
{
	freopen("journey.in","r",stdin);
	freopen("journey.out","w",stdout);

//	memset(f,0,sizeof(f));
	memset(a,0,sizeof(a));
	memset(b,0,sizeof(b));

	long long tot=0;
	scanf("%d%d",&n,&m);
	for (int i=1;i<=n;i++)
	{
		scanf("%s",s);
		for (int j=0;j<m;j++)
			if (s[j]=='X')
			{
				tot++;
				a[i]=j+1;
				b[j+1]=i;
				break;
			}
	}
	tot=n*m-tot;

//	for (int i=1;i<=n;i++) cout<<a[i]<<' ';

	/*
	for (int i=1;i<=n;i++)
		for (int j=1;j<=m;j++)
		{
			f[i][j]=f[i-1][j]+f[i][j-1]-f[i-1][j-1]+i*j*(i+j-2)+(i-1)*(j-1)*(i+j);
			if (a[i]==j) f[i][j]-=+i*j*(i+j-2);
			else
			{
				for (int k=1;k<=i;k++)
				 if (a[k]&&a[k]<=j)
				 f[i][j]-=2*(i-k+j-a[k]);


				if (a[i]&&a[i]<j)
					 f[i][j]-=(i-1)*(i+2*(j-a[i]));
				if (b[j]&&b[j]<i)
					 f[i][j]-=(j-1)*(j+2*(i-b[j]));
				if (a[i]&&a[i]<j&&b[j]&&b[j]<i)
					f[i][j]+=2*((j-a[i])+(i-b[j]));

			}
//			cout<<f[i][j]<<endl;
		}
/*
	for (int i=0;i<=n;i++){
		for (int j=0;j<=m;j++)
			cout<<f[i][j]<<' ';
		cout<<'n';
	}*/

	long long ans=0;
	for (int i=1;i<n;i++)
		for (int j=i+1;j<=n;j++)
			ans+=(m-(a[i]>0))*(m-(a[j]>0))*(j-i);
	for (int i=1;i<m;i++)
		for (int j=i+1;j<=m;j++)
			ans+=(n-(b[i]>0))*(n-(b[j]>0))*(j-i);
	ans*=2;





//	long long ans=f[n][m];
//	cout<<ans<<endl;
	for (int i=1;i<=n;i++)
		if (a[i])
		{
			ans+=2*calc(a[i],a[i],m);
			if (i>1&&a[i-1])
			{
				int j=i-1;
				if (a[j]<a[j+1])
				{
					while (j>0&&a[j]&&a[j]<a[j+1]) {ans+=2*calc(a[j],a[i],m);j--;}
				}
			}
			if (i<n&&a[i+1])
			{
				int j=i+1;
				if (a[j]<a[j-1])
				{
					while (j<=n&&a[j]&&a[j]<a[j-1]) {ans+=2*calc(a[j],a[i],m);j++;}
				}
			}
		}
	for (int i=1;i<=m;i++)
		if (b[i])
		{
			ans+=2*calc(b[i],b[i],n);
			if (i>1&&b[i-1])
			{
				int j=i-1;
				if (b[j]<b[j+1])
				{
					while (j>0&&b[j]&&b[j]<b[j+1]) {ans+=2*calc(b[j],b[i],n);j--;}
				}
			}
			if (i<m&&b[i+1])
			{
				int j=i+1;
				if (b[j]<b[j-1])
				{
					while (j<=m&&b[j]&&b[j]<b[j-1]) {ans+=2*calc(b[j],b[i],n);j++;}
				}
			}
		}
//	cout<<ans<<endl;
	double _ans=double(ans)/double(tot)/double(tot);
	printf("%.4lfn",_ans);


//	while (1);

	return 0;


}

CF 237C (质数区间)

给定区间[a,b] 求l的最小值使[a,b]中任意长度为l的一段包含至少k个Prime

二分l

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<functional>
#include<iostream>
using namespace std;
#define MAXN (1000000+10)
int a[MAXN],tot=0,x,y,k;
bool b[MAXN]={0};
void work()
{
	b[1]=1;
	for (int i=2;i<=y;i++)
	{
		if (!b[i])
		{
			tot++;
			a[tot]=i;
		}
		for (int j=1;j<=tot;j++)
		{
			if (a[j]*i>y) break;
			b[a[j]*i]=1;
			if (!i%a[j]) break;
		}
	}
}
bool is_ok(int l)
{
	int tot=0;
	for (int i=x;i<=x+l-1;i++)
		if (!b[i]) tot++;
	if (tot<k) return false;
	for (int j=x+l;j<=y;j++)
	{
		tot=tot+(!b[j])-(!b[j-l]);
		if (tot<k) return false;
	}
	return true;
}

int main()
{
	scanf("%d%d%d",&x,&y,&k);
	work();
	int l=k,r=y-x+1;
	if (l>r||!is_ok(r))
	{
		printf("-1n");
		return 0;
	}
	for (int i=1;i<=60;i++)
	{
		if (l==r) break;
		int m=(l+r)>>1;
//		if (r-l==1) m++;
		if (is_ok(m)) r=m;
		else l=m+1;
	}
	printf("%dn",l);
//	while (1);
	return 0;
}

CF 237A (Cash)

题目大意:是一堆人来h点m分来超市买东西,同时可以有an位顾客买单,买单可认为1分钟以内完成,问至少有几位售货员才能使所有顾客不等待

直接统计……

#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<algorithm>
#include<functional>

using namespace std;
#define MAXN (100000+10)
int n,i;
int a[MAXN]={0};
int main()
{
	int ans=0;
	scanf("%d",&n);
	for (int i=1;i<=n;i++)
	{
		int h,m;
		scanf("%d%d",&h,&m);
		a[h*60+m]++;
		ans=max(ans,a[h*60+m]);
	}
	printf("%dn",ans);

//	while (1);
	return 0;
}