###### tags: `LeetCode` `Recursion` `DFS` `Hard`
# LeetCode #996 [Number of Squareful Arrays](https://leetcode.com/problems/number-of-squareful-arrays/)
### (Hard)
給定一個非負整數數組 A,如果該數組每對相鄰元素之和是一個完全平方數,則稱這一數組為正方形數組。
返回 A 的正方形排列的數目。兩個排列 A1 和 A2 不同的充要條件是存在某個索引 i,使得 A1[i] != A2[i]。
---
題目要求相鄰的兩數能成為完全平方數, 因此檢查的條件為:
* 輪到的數字沒有被使用過
* 輪到的數字和上一個數字能組成完全平方數
* 若輪到的數字為該輪的第一個數字, 則它必須是unique的, 不然會有重複排列
若不滿足上述任一條件, 則跳過該數
而因為是求排列(不存在後面的數不用考慮前面這種情況), 所以需要兩個for迴圈, 一個負責決定第一個數, 另一個負責遍歷所有的排列組合。
而每當暫存數組的大小等於給定數組時, 代表已遍歷完整個數組, 此時count加一, 最後回傳count即可。
---
```
class Solution {
public:
int count=0;
int numSquarefulPerms(vector<int>& nums) {
vector<int> used(nums.size(), 0);
vector<int> tmp;
sort(nums.begin(), nums.end());
for(int i=0;i<nums.size();i++){//first for loop
if(i!=0&&nums[i]==nums[i-1])continue;
used[i]=1;
tmp.push_back(nums[i]);
dfs(nums, used, tmp);
tmp.pop_back();
used[i]=0;
}
return count;
}
void dfs(vector<int>& nums, vector<int>& used, vector<int>& tmp){
if(tmp.size()!=1){
int square=tmp.back()+*(&tmp.back()-1);
if(!isint(sqrt(square)))return;
}
if(tmp.size()==nums.size()){
count++;
return;
}
for(int i=0;i<nums.size();i++){//second for loop.
if(used[i]==1)continue;
if(i!=0&&nums[i]==nums[i-1]&&used[i-1]==0)continue;
used[i]=1;
tmp.push_back(nums[i]);
dfs(nums, used, tmp);
tmp.pop_back();
used[i]=0;
}
}
bool isint(float val){
return val==(float)(int)val;
}
};
```