-
Notifications
You must be signed in to change notification settings - Fork 1k
/
Copy pathRectangle_cutting.cpp
112 lines (95 loc) · 2.88 KB
/
Rectangle_cutting.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
/*
Given an a×b rectangle, your task is to cut it into squares. On each move you can select a rectangle and cut it into two
rectangles in such a way that all side lengths remain integers. What is the minimum possible number of moves?
Input
The only input line has two integers a and b.
Output
Print one integer: the minimum number of moves.
Constraints
1≤a,b≤500
Example
Input:
3 5
Output:
3
*/
/*
APPROACH
----------------------------------------------------
We will make all the horizontal cuts and try to find
after which move we have to make minimum moves to make
all squares,
Similarly,
We do the same with vertical moves,
We will divide the recetanle in two parts and add the
result for both parts and find minimum among all moves
which will be our ans.
*/
#include <iostream>
using namespace std;
int dp[501][501]; //declaring dp array for memorization.
int cutting(int a,int b)
{
//if side lengths are equal then no move required.
if(a==b)
return 0;
//for a by 1 or 1 by b rectangle we have to make a-1 and b-1 cuts respectively.
if(a==1 || b==1)
return max(a,b)-1;
//if the result is stored already return it.
if(dp[a][b]!=-1)
return dp[a][b];
int ans=INT_MAX;
//recursivly making cuts.
//loop to make all the horizontal cuts
/*
iterate upto a/2 only wwwww
*/
for(int i=1;i<=a/2;i++)
{
//making every position move and taking the minimum out of it.
ans=min(ans,cutting(i,b)+cutting(a-i,b));
}
//loop to make all the vertical cuts
for(int i=1;i<=b/2;i++)
{
//making every position move and taking the minimum out of it.
ans=min(ans,cutting(a,i)+cutting(a,b-i));
}
return dp[a][b]=1+ans; //this 1 corressponds to one of the minimum move we have done above.
}
int cutting_iterative(int a,int b)
{
int dp[a+1][b+1];
//These 2 loops are for base case.
for(int i=1;i<=a;i++)
dp[i][1]=i-1;
for(int i=1;i<=b;i++)
dp[1][i]=i-1;
for(int i=2;i<a+1;i++)
{
for(int j=2;j<b+1;j++)
{
if(i==j)
dp[i][j]=0;
else
{
int ans=INT_MAX;
for(int k=1;k<=i/2;k++)
ans=min(ans,dp[k][j]+dp[i-k][j]);
for(int k=1;k<=j/2;k++)
ans=min(ans,dp[i][k]+dp[i][j-k]);
dp[i][j]=1+ans;
}
}
}
return dp[a][b];
}
int main(){
int a,b;
cin>>a>>b;
for(int i=0;i<=a;i++) for(int j=0;j<=b;j++) dp[i][j]=-1; //initialising the DP Array for Memorization.
cout<<cutting(a,b)<<endl;//call to the function to find the answer using memorization.
cout<<cutting_iterative(a,b)<<endl;//call to function to find the answer using iteration.
return 0;
}