202309-2 | |
试题名称: | 坐标变换(其二) |
时间限制: | 2.0s |
内存限制: | 512.0MB |
问题描述: | 问题描述对于平面直角坐标系上的坐标 (,),小 P 定义了如下两种操作:
设定好了包含 个操作的序列 (1,2,⋯,) 后,小 P 又定义了如下查询:
对于给定的操作序列,试计算 个查询的结果。 输入格式从标准输入读入数据。 输入共 ++1 行。 输入的第一行包含空格分隔的两个正整数 和 ,分别表示操作和查询个数。 接下来 行依次输入 个操作,每行包含空格分隔的一个整数(操作类型)和一个实数( 或 ),形如 1 (表示拉伸 倍)或 2 (表示旋转 )。 接下来 行依次输入 个查询,每行包含空格分隔的四个整数 、、 和 ,含义如前文所述。 输出格式输出到标准输出中。 输出共 行,每行包含空格分隔的两个实数,表示对应查询的结果。 样例输入 Data 样例输出 Data 样例说明第五个查询仅对输入坐标使用了操作八:拉伸 0.716 倍。 横坐标:159430×0.716=114151.88 纵坐标:−511187×0.716=−366009.892 由于具体计算方式不同,程序输出结果可能与真实值有微小差异,样例输出仅保留了三位小数。 评测用例规模与约定80% 的测试数据满足:,≤1000; 全部的测试数据满足:
评分方式如果你输出的浮点数与参考结果相比,满足绝对误差不大于 0.1,则该测试点满分,否则不得分。 提示
|
//334 80分朴素代码
#include
using namespace std;
const int N=1000010;
int main()
{
int n,m;
cin>>n>>m;
pairp[n+1];
for(int i=1;i<=n;i++)
{
// cin>>p[i].first>>p[i].second;
scanf("%d%lf",&p[i].first,&p[i].second);
}
while(m--)
{
int i,j;
double x,y;
cin>>i>>j>>x>>y;
for(int k=i;k<=j;k++)
{
if(p[k].first==1)
x*=p[k].second,y*=p[k].second;
else
{
double t=x;
x=x*cos(p[k].second)-y*sin(p[k].second);
y=t*sin(p[k].second)+y*cos(p[k].second);
}
}
printf("%.3f %.3f",x,y);
cout<
//335 100分代码前缀和思想
#include
using namespace std;
const int N=1000010;
double s[N],k[N]={1};//注意前缀乘积初始化为1;
int main()
{
int n,m;
cin>>n>>m;
pairp[n+1];
for(int i=1;i<=n;i++)
{
cin>>p[i].first>>p[i].second;
if(p[i].first==1)
{
k[i]=k[i-1]*p[i].second;
s[i]=s[i-1];//与上一步的和一样
}
else
{
k[i]=k[i-1];//与上一步乘积一样
s[i]=s[i-1]+p[i].second;
}
}
// for(int i=1;i<=n;i++) cout<>i>>j>>x>>y;
double a=s[j]-s[i-1];
double b=k[j]/k[i-1];
x*=b,y*=b;
double t=x;
x=x*cos(a)-y*sin(a);
y=t*sin(a)+y*cos(a);
printf("%.3lf %.3lf",x,y);
cout<