TFIELD spoj – VOI 2015 Day 1 Ruộng bậc thang

Nguồn đề bàihttp://vn.spoj.com/problems/TFIELD/

1. Đề bài TFIELD spoj

Ở các vùng cao hiếm đất cùng mặt bằng để canh tác, khi tiến hành trồng trọt trên các sườn đồi núi có đất màu, người ta phải bạt tam cấp để tạo thành những vạt đất bằng. Khu vực đất dốc dùng để canh tác như vậy gọi là ruộng bậc thang. Hình ảnh các khu ruộng bậc thang vẫn luôn là một hình ảnh đẹp ở các vùng cao khiến du khách và các nhà nhiếp ảnh đam mê và tốn không ít phim ảnh. Gia đình Hoàng có một khu ruộng bậc thang bao quanh một ngọn đồi được chia thành các khoang bậc thang, mỗi khoang trồng một loại cây. Khi nhìn từ trên cao xuống, ta thấy các khoang bậc thang này có hình dạng của các đa giác lồi lồng nhau. Ngoại trừ khoang chứa đỉnh đồi có biên là một đa giác lồi chứa đỉnh đồi, mỗi khoang còn lại được xác định bởi hai đa giác lồng nhau: đa giác có diện tích lớn hơn được gọi là biên ngoài của khoang còn đa giác có diện tích nhỏ hơn được gọi là biên trong của khoang. Mỗi khoang có màu đặc trưng của loại cây được trồng ở khoang đó. Vốn là một người say mê chụp ảnh, muốn có một bức ảnh đẹp, Hoàng tìm cách thay đổi không quá k loại cây được trồng ở k khoang để khi nhìn từ trên cao xuống sẽ thấy một vùng cùng màu có diện tích lớn nhất. Hoàng đã ghi nhận được danh sách mđa giác lồi mô tả biên ngoài của m khoang và màu tương ứng của chúng. Do sơ xuất, Hoàng đã để các thông tin về các khoang trong danh sách bị xáo trộn, không còn được liệt kê theo đúng trình tự từ khoang trong đến khoang ngoài.

Yêu cầu

Cho biết thông tin về danh sách mà Hoàng đã ghi nhận và số nguyên k, hãy tìm cách thay đổi không quá k loại cây được trồng ở k khoang để khi nhìn từ trên cao xuống sẽ thấy một vùng cùng màu có diện tích lớn nhất.

Dữ liệu vào

  • Dòng đầu chứa hai số nguyên dương m, k (km);
  • Dòng thứ i trong số m dòng tiếp theo chứa thông tin về khoang thứ i trong danh sách mà Hoàng ghi nhận bao gồm:
    • Đầu tiên là số nguyên ni là số đỉnh của đa giác lồi mô tả biên ngoài của khoang;
    • Tiếp theo là số nguyên ci thể hiện màu của khoang (1 ≤ cim);
    • Cuối cùng là ni cặp số nguyên, mỗi số có trị tuyệt đối không quá 109, là tọa độ của một đỉnh của đa giác. Các đỉnh của đa giác được liệt kê theo thứ tự ngược chiều kim đồng hồ.

Hai số liên tiếp trên cùng dòng được ghi cách nhau bởi dấu cách.

Dữ liệu ra

  • Ghi ra một số thực là diện tích vùng cùng màu lớn nhất sau khi thay đổi không quá kloại cây được trồng ở k khoang (kết quả đưa ra với độ chính xác 1 chữ số sau dấu chấm thập phân).

Ràng buộc

  • Có 40% số test ứng với 40% số điểm của bài thỏa mãn điều kiện: m ≤ 10; k = 1; các đa giác mô tả biên ngoài của các khoang là hình chữ nhật;
  • Có 40% số test khác ứng với 40% số điểm của bài thỏa mãn điều kiện: m ≤ 10; các đa giác mô tả biên ngoài của các khoang là tam giác;
  • Có 20% số test còn lại ứng với 20% số điểm của bài thỏa mãn điều kiện: mni ≤ 1000.

Ví dụ

Input:

3 1

4 1 0 0 1 0 1 1 0 1

4 1 -2 -3 5 -3 5 5 -2 5

3 2 -1 -1 4 -1 -1 4

Output:

56.0

2. Gợi ý TFIELD spoj – VOI 2015

– Sort tăng dần các đa giác theo diện tích

– Tìm dãy các đa giác dài nhất cùng màu.

3. Code Tham Khảo TFIELD spoj – VOI 2015

const fi='';
      fo='';
      maxn=1000;
      maxm=1000;
type  point=record
        x,y:int64;
      end;
      poly=record
        p:array[1..maxn+1] of point;
        n:longint;
        s:int64;
        c:longint;
      end;
var   f:text;
      d:array[0..maxm] of poly;
      dem:array[0..maxm] of longint;
      kq:int64;
      m,k,i,j,max:longint;

procedure input;
  begin
    assign(f,fi);
    reset(f);
    readln(f,m,k);
    for i:=1 to m do
      begin
        read(f,d[i].n);
        read(f,d[i].c);
        for j:=1 to d[i].n do read(f,d[i].p[j].x,d[i].p[j].y);
        d[i].p[d[i].n+1].x:=d[i].p[1].x;
        d[i].p[d[i].n+1].y:=d[i].p[1].y;
        readln(f);
      end;
    close(f);
  end;

function tinhS(d:poly):int64;
var s:int64;
    var i:longint;
  begin
    s:=0;
    for i:=1 to d.n do
      s:=s+(d.p[i+1].x-d.p[i].x)*(d.p[i+1].y+d.p[i].y);
    exit(abs(s));
  end;

procedure sort(l,r: longint);
var i,j: longint;
    x:int64;
    y:poly;
  begin
    i:=l;
    j:=r;
    x:=d[(l+r) div 2].s;
    repeat
      while d[i].s<x do inc(i);
      while x<d[j].s do dec(j);
      if not(i>j) then
        begin
           y:=d[i];
           d[i]:=d[j];
           d[j]:=y;
           inc(i);
           j:=j-1;
        end;
    until i>j;
    if l<j then sort(l,j);
    if i<r then sort(i,r);
  end;

procedure solve;
  begin
    for i:=1 to m do
      d[i].s:=tinhS(d[i]);
    sort(1,m);
    for i:=1 to m do
      begin
        fillchar(dem,sizeof(dem),0);
        max:=0;
        for j:=i to m do
          begin
            inc(dem[d[j].c]);
            if max<dem[d[j].c] then max:=dem[d[j].c];
            if j-i+1-max<=k then
              if d[j].s-d[i-1].s>kq then kq:=d[j].s-d[i-1].s;
          end;
      end;
  end;

procedure output;
  begin
    assign(f,fo);
    rewrite(f);
    write(f,kq div 2);
    if kq mod 2 = 0 then write(f,'.',0)
    else write(f,'.',5);
    close(f);
  end;

begin
  input;
  solve;
  output;
end.

Để lại một bình luận

Email của bạn sẽ không được hiển thị công khai. Các trường bắt buộc được đánh dấu *