#include <iostream>
#include <vector>
#include <cstdint>
#include <algorithm>
#include <random>
#include <iterator>
typedef std::vector<std::uint32_t> DType;
typedef std::vector<DType> Grid;
typedef std::vector<Grid> RType;
typedef std::vector < std::vector<DType>> Cube;
#ifdef _WIN32
#include <Windows.h>//ms windows os only.
void gotoxy(int X, int Y) {//カーソルを移動させる関数です。環境によってはエスケープシーケンスが使えるのでそれに置き換えることもできます。
COORD Pos = { static_cast<SHORT>(X),static_cast<SHORT>(Y) };//WINDOWS定義の構造体です。
SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), Pos);//windows定義の関数です。カーソルを移動します。
return;
}
#else
void gotoxy(int X, int Y) {
return;
}
#endif
Grid MakeProblem() {
Grid R{
{ 1,5,0,0,8,6,0,7,0, },
{ 7,0,8,9,0,0,0,0,0, },
{ 0,9,0,5,0,0,0,8,1, },
{ 9,0,0,0,0,8,1,0,6, },
{ 0,0,1,0,2,0,0,9,0, },
{ 0,7,5,6,0,0,2,0,0, },
{ 5,0,6,0,4,0,0,2,0, },
{ 3,0,0,1,0,2,5,0,0, },
{ 0,4,0,8,0,0,0,0,9, },
};
return R;
}
Grid MakeGrid() {
Grid G(9);
for (auto& o : G) {
o.assign(9, 0);
}
return G;
}
Cube MakeCube() {
DType D{ 1,2,3,4,5,6,7,8,9 };
Cube C(9);
for (auto& oo : C) {
oo.resize(9);
for (auto& o : oo) {
o = D;
}
}
return C;
}
Cube ReduceCube(const Cube& Cu, const Grid& P) {
Cube C = Cu;
for (std::size_t i = 0; i < P.size(); i++) {
for (std::size_t j = 0; j < P[i].size(); j++) {
if (P[i][j] != 0) {
C[i][j].clear();
C[i][j].push_back(P[i][j]);
}
}
}
for (std::size_t i = 0; i < C.size(); i++) {
for (std::size_t j = 0; j < C[i].size(); j++) {
if (C[i][j].size() != 1)continue;
for (std::size_t k = 0; k < P.size(); k++) {
if (C[i][k].size() == 1) continue;
auto it = std::find(C[i][k].begin(), C[i][k].end(), C[i][j][0]);
if (it != C[i][k].end()) {
C[i][k].erase(it);
}
}
for (std::size_t k = 0; k < P.size(); k++) {
if (C[k][j].size() == 1) continue;
auto it = std::find(C[k][j].begin(), C[k][j].end(), C[i][j][0]);
if (it != C[k][j].end()) {
C[k][j].erase(it);
}
}
}
}
std::size_t X = 0;
std::size_t Y = 0;
for (std::size_t Y = 0; Y < 3; Y++) {
for (std::size_t X = 0; X < 3; X++) {
for (std::size_t i = 0; i < 3; i++) {
for (std::size_t j = 0; j < 3; j++) {
if (C[Y * 3 + i][X * 3 + j].size() != 1)continue;
for (std::size_t k = 0; k < 3; k++) {
for (std::size_t l = 0; l < 3; l++) {
if (C[Y * 3 + k][X * 3 + l].size() == 1)continue;
auto it = std::find(C[Y * 3 + k][X * 3 + l].begin(), C[Y * 3 + k][X * 3 + l].end(), C[Y * 3 + i][X * 3 + j][0]);
if (it != C[Y * 3 + k][X * 3 + l].end()) {
C[Y * 3 + k][X * 3 + l].erase(it);
}
}
}
}
}
}
}
return C;
}
bool IsAnswer(Grid P) {
DType D;
for (std::size_t Y = 0; Y < 3; Y++) {
for (std::size_t X = 0; X < 3; X++) {
for (std::size_t i = 0; i < 3; i++) {
for (std::size_t j = 0; j < 3; j++) {
D.push_back(P[Y * 3 + i][X * 3 + j]);
}
}
std::sort(D.begin(), D.end());
D.erase(std::unique(D.begin(), D.end()), D.end());
if (D.size() != 9)return false;
}
}
for (std::size_t i = 0; i < P.size(); i++) {
D.clear();
for (std::size_t j = 0; j < P[i].size(); j++) {
D.push_back(P[i][j]);
}
std::sort(D.begin(), D.end());
D.erase(std::unique(D.begin(), D.end()), D.end());
if (D.size() != 9)return false;
}
for (std::size_t i = 0; i < P.size(); i++) {
D.clear();
for (std::size_t j = 0; j < P[i].size(); j++) {
D.push_back(P[j][i]);
}
std::sort(D.begin(), D.end());
D.erase(std::unique(D.begin(), D.end()), D.end());
if (D.size() != 9)return false;
}
return true;
}
Grid CubeToGrid(const Cube& C, const Grid G) {
Grid R = MakeGrid();
for (std::size_t i = 0; i < G.size(); i++) {
for (std::size_t j = 0; j < G[i].size(); j++) {
R[i][j] = C[i][j][G[i][j]];
}
}
return R;
}
Grid CubeToProblem(const Cube& C) {
Grid G = MakeGrid();
for (std::size_t i = 0; i < G.size(); i++) {
for (std::size_t j = 0; j < G[i].size(); j++) {
if (C[i][j].size() == 1) {
G[i][j] = C[i][j][0];
}
else {
G[i][j] = 0;
}
}
}
return G;
}
bool ShowList(Cube& C) {
std::size_t i = 1;
double D = 0;
for (auto& ooo : C) {
for (auto& oo : ooo) {
std::cout << '[' << i << ':';
i++;
D += oo.size();
for (auto& o : oo) {
std::cout << o << ',';
}
std::cout << ']';
}
std::cout << std::endl;
}
std::cout << D / 81.0 << std::endl;
return true;
}
bool Show(const Grid& G) {
for (auto& oo : G) {
for (auto& o : oo) {
std::cout << o;
}
std::cout << std::endl;
}
std::cout << std::endl;
std::cout << std::endl;
return true;
}
DType GetAntiIntersection(DType A ,const DType& B){//積集合を取り除く??
for (auto& o : B) {
auto it = std::find(A.begin(), A.end(), o);
if(it != A.end())A.erase(it);
}
return A;
}
bool IsHorizonValid(const Grid& G,std::size_t V) {//横方向検査
DType D;
for (std::size_t i = 0; i < G[V].size(); i++) {
if(G[V][i]!=0)D.push_back(G[V][i]);
}
std::sort(D.begin(), D.end());
D.erase(std::unique(D.begin(), D.end()), D.end());
return D.size() == 9 ? true : false;
}
DType GetHorizonUsing(const Grid& G,std::size_t V) {//横方向検査
DType D;
for (std::size_t i = 0; i < G[V].size(); i++) {
if(G[V][i]!=0) D.push_back(G[V][i]);
}
return D;
}
bool HaveHorizonZero(const Grid& G,std::size_t V) {//横方向検査
DType D;
for (std::size_t i = 0; i < G[V].size(); i++) {
if (G[V][i] == 0)return true;
}
return false;
}
bool IsVerticalValid(const Grid& G,std::size_t H) {//縦方向検査
DType D;
for (std::size_t i = 0; i < G.size(); i++) {
if(G[i][H]!= 0)D.push_back(G[i][H]);
}
std::sort(D.begin(), D.end());
D.erase(std::unique(D.begin(), D.end()), D.end());
return D.size() == 9 ? true : false;
}DType GetVerticalUsing(const Grid& G,std::size_t H) {//横方向検査
DType D;
DType T{ 0,1,2,3,4,5,6,7,8 };
for (std::size_t i = 0; i < G.size(); i++) {
if(G[i][H]!= 0)D.push_back(G[i][H]);
}
return D;
}
bool HaveVerticalZero(const Grid& G,std::size_t H) {//縦方向検査
DType D;
for (std::size_t i = 0; i < G.size(); i++) {
if (G[i][H] == 0)return true;
}
return false;
}
bool IsBlockValid(const Grid& G, std::size_t X, std::size_t Y) {//3x3検査
DType D;
for (std::size_t i = 0; i < 3; i++) {
for (std::size_t j = 0; j < 3; j++) {
if(G[Y*3+i][X*3+j] !=0)D.push_back(G[Y*3+i][X*3+j]);
}
}
std::sort(D.begin(), D.end());
D.erase(std::unique(D.begin(), D.end()), D.end());
return D.size() == 9 ? true : false;
}
DType GetBlockUsing(const Grid& G,std::size_t X,std::size_t Y) {//横方向検査
DType D;
for (std::size_t i = 0; i < 3; i++) {
for (std::size_t j = 0; j < 3; j++) {
if(G[Y*3+i][X*3+j] !=0) D.push_back(G[Y*3+i][X*3+j]);
}
}
return D;
}
bool HaveBlockNumber(const Grid& G, std::size_t X, std::size_t Y, std::size_t N) {//横方向検査
DType D;
for (std::size_t i = 0; i < 3; i++) {
for (std::size_t j = 0; j < 3; j++) {
if (G[Y * 3 + i][X * 3 + j] == N)return true;
}
}
return false;
}
bool HaveBlockZero(const Grid& G, std::size_t X, std::size_t Y) {//3x3検査
DType D;
for (std::size_t i = 0; i < 3; i++) {
for (std::size_t j = 0; j < 3; j++) {
if (G[Y * 3 + i][X * 3 + j] == 0)return true;
}
}
return false;
}
bool Search_r(Grid& G, int Deep, RType& R, std::size_t& C) {
C++;
gotoxy(11, 0);
// std::cout << C << std::endl;
if (Deep >= 81) {
gotoxy(0, 0);
//Show(G);
if (IsAnswer(G)) {
R.push_back(G);
std::cout << "find!!" << std::endl;
return true;
}
return false;
}
static const DType T{ 1,2,3,4,5,6,7,8,9 };
std::size_t MX = (Deep % 9) % 3;
std::size_t MY = (Deep % 9) / 3;
std::size_t BX = (Deep / 9) % 3;
std::size_t BY = (Deep / 9) / 3;
auto BU = GetBlockUsing(G, BX, BY);
auto HU = GetHorizonUsing(G, BY*3+MY);
auto VU = GetVerticalUsing(G, BX*3 + MX);
DType AI;
if (G[BY*3 + MY][BX*3 + MX] != 0) {
return Search_r(G, Deep + 1, R,C);
}
std::copy(BU.begin(), BU.end(), std::back_inserter(AI));
std::copy(HU.begin(), HU.end(), std::back_inserter(AI));
std::copy(VU.begin(), VU.end(), std::back_inserter(AI));
std::sort(AI.begin(), AI.end());
AI.erase(std::unique(AI.begin(), AI.end()), AI.end());
DType D = GetAntiIntersection(T, AI);
auto WG = G;
if (D.size() == 0) return false;
for (std::size_t i = 0; i < D.size(); i++) {
WG[BY*3 + MY][BX*3 + MX] = D[i];
gotoxy(0, 0);
// Show(WG);
Search_r(WG, Deep + 1, R,C);
}
return false;
}
bool Search(Grid& G, RType& R) {
int Deep = 0;
std::size_t C=0;
return Search_r(G, Deep, R, C);
}
RType MakeHoge(Grid G) {
Cube Cu = MakeCube();
Cube C = ReduceCube(Cu, G);
Grid V = MakeGrid();
RType R;
Grid B = G;
Grid D;
std::uint64_t Co = 0;
std::uint64_t Find = 1;
do {
D = B;
B = CubeToProblem(C);
C = ReduceCube(C, B);
} while (D != B);
gotoxy(0, 11);
// Show(B);
// ShowList(C);
auto A = CubeToProblem(C);
gotoxy(0, 11);
// Show(A);
Search(A, R);
return R;
}
int main() {
Grid G = MakeProblem();
RType R;
Show(G);
R = MakeHoge(G);
for (auto& o : R) {
gotoxy(0, 0);
Show(o);
}
return 0;
}
#include <iostream>
#include <vector>
#include <cstdint>
#include <algorithm>
#include <random>
#include <iterator>

typedef std::vector<std::uint32_t> DType;
typedef std::vector<DType> Grid;
typedef std::vector<Grid> RType;
typedef std::vector < std::vector<DType>> Cube;
#ifdef _WIN32
#include <Windows.h>//ms windows os only.
void gotoxy(int X, int Y) {//カーソルを移動させる関数です。環境によってはエスケープシーケンスが使えるのでそれに置き換えることもできます。
	COORD Pos = { static_cast<SHORT>(X),static_cast<SHORT>(Y) };//WINDOWS定義の構造体です。
	SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), Pos);//windows定義の関数です。カーソルを移動します。
	return;
}
#else
void gotoxy(int X, int Y) {
	return;
}
#endif


Grid MakeProblem() {
	Grid R{
		{ 1,5,0,0,8,6,0,7,0, },
		{ 7,0,8,9,0,0,0,0,0, },
		{ 0,9,0,5,0,0,0,8,1, },
		{ 9,0,0,0,0,8,1,0,6, },
		{ 0,0,1,0,2,0,0,9,0, },
		{ 0,7,5,6,0,0,2,0,0, },
		{ 5,0,6,0,4,0,0,2,0, },
		{ 3,0,0,1,0,2,5,0,0, },
		{ 0,4,0,8,0,0,0,0,9, },
	};

	return R;

}

Grid MakeGrid() {
	Grid G(9);
	for (auto& o : G) {
		o.assign(9, 0);
	}
	return G;
}
Cube MakeCube() {
	DType D{ 1,2,3,4,5,6,7,8,9 };
	Cube C(9);
	for (auto& oo : C) {
		oo.resize(9);
		for (auto& o : oo) {
			o = D;
		}
	}

	return C;
}

Cube ReduceCube(const Cube& Cu, const Grid& P) {
	Cube C = Cu;
	for (std::size_t i = 0; i < P.size(); i++) {
		for (std::size_t j = 0; j < P[i].size(); j++) {
			if (P[i][j] != 0) {
				C[i][j].clear();
				C[i][j].push_back(P[i][j]);
			}

		}
	}
	for (std::size_t i = 0; i < C.size(); i++) {
		for (std::size_t j = 0; j < C[i].size(); j++) {
			if (C[i][j].size() != 1)continue;


			for (std::size_t k = 0; k < P.size(); k++) {
				if (C[i][k].size() == 1) continue;
				auto it = std::find(C[i][k].begin(), C[i][k].end(), C[i][j][0]);
				if (it != C[i][k].end()) {
					C[i][k].erase(it);
				}
			}
			for (std::size_t k = 0; k < P.size(); k++) {
				if (C[k][j].size() == 1) continue;
				auto it = std::find(C[k][j].begin(), C[k][j].end(), C[i][j][0]);
				if (it != C[k][j].end()) {
					C[k][j].erase(it);
				}
			}
		}
	}
	std::size_t X = 0;
	std::size_t Y = 0;
	for (std::size_t Y = 0; Y < 3; Y++) {
		for (std::size_t X = 0; X < 3; X++) {

			for (std::size_t i = 0; i < 3; i++) {
				for (std::size_t j = 0; j < 3; j++) {

					if (C[Y * 3 + i][X * 3 + j].size() != 1)continue;

					for (std::size_t k = 0; k < 3; k++) {
						for (std::size_t l = 0; l < 3; l++) {
							if (C[Y * 3 + k][X * 3 + l].size() == 1)continue;
							auto it = std::find(C[Y * 3 + k][X * 3 + l].begin(), C[Y * 3 + k][X * 3 + l].end(), C[Y * 3 + i][X * 3 + j][0]);
							if (it != C[Y * 3 + k][X * 3 + l].end()) {
								C[Y * 3 + k][X * 3 + l].erase(it);

							}
						}
					}
				}
			}
		}
	}
	return C;
}


bool IsAnswer(Grid P) {
	DType D;
	for (std::size_t Y = 0; Y < 3; Y++) {
		for (std::size_t X = 0; X < 3; X++) {
			for (std::size_t i = 0; i < 3; i++) {
				for (std::size_t j = 0; j < 3; j++) {
					D.push_back(P[Y * 3 + i][X * 3 + j]);
				}
			}
			std::sort(D.begin(), D.end());
			D.erase(std::unique(D.begin(), D.end()), D.end());
			if (D.size() != 9)return false;
		}
	}

	for (std::size_t i = 0; i < P.size(); i++) {
		D.clear();
		for (std::size_t j = 0; j < P[i].size(); j++) {
			D.push_back(P[i][j]);
		}
		std::sort(D.begin(), D.end());
		D.erase(std::unique(D.begin(), D.end()), D.end());
		if (D.size() != 9)return false;
	}
	for (std::size_t i = 0; i < P.size(); i++) {
		D.clear();
		for (std::size_t j = 0; j < P[i].size(); j++) {
			D.push_back(P[j][i]);
		}
		std::sort(D.begin(), D.end());
		D.erase(std::unique(D.begin(), D.end()), D.end());
		if (D.size() != 9)return false;
	}

	return true;
}
Grid CubeToGrid(const Cube& C, const Grid G) {
	Grid R = MakeGrid();
	for (std::size_t i = 0; i < G.size(); i++) {
		for (std::size_t j = 0; j < G[i].size(); j++) {
			R[i][j] = C[i][j][G[i][j]];
		}
	}
	return R;
}
Grid CubeToProblem(const Cube& C) {
	Grid G = MakeGrid();
	for (std::size_t i = 0; i < G.size(); i++) {
		for (std::size_t j = 0; j < G[i].size(); j++) {
			if (C[i][j].size() == 1) {
				G[i][j] = C[i][j][0];
			}
			else {
				G[i][j] = 0;
			}
		}
	}
	return G;
}

bool ShowList(Cube& C) {
	std::size_t i = 1;
	double D = 0;
	for (auto& ooo : C) {
		for (auto& oo : ooo) {
			std::cout << '[' << i << ':';
			i++;
			D += oo.size();
			for (auto& o : oo) {
				std::cout << o << ',';
			}
			std::cout << ']';
		}
		std::cout << std::endl;
	}
	std::cout << D / 81.0 << std::endl;
	return true;
}

bool Show(const Grid& G) {
	for (auto& oo : G) {
		for (auto& o : oo) {
			std::cout << o;
		}
		std::cout << std::endl;
	}
	std::cout << std::endl;
	std::cout << std::endl;
	return true;
}
DType GetAntiIntersection(DType A ,const DType& B){//積集合を取り除く？？
	for (auto& o : B) {
		auto it = std::find(A.begin(), A.end(), o);
		if(it != A.end())A.erase(it);
	}
	return A;
}
bool IsHorizonValid(const Grid& G,std::size_t V) {//横方向検査
	DType D;
	for (std::size_t i = 0; i < G[V].size(); i++) {
		if(G[V][i]!=0)D.push_back(G[V][i]);
	}
	std::sort(D.begin(), D.end());
	D.erase(std::unique(D.begin(), D.end()), D.end());
	return D.size() == 9 ? true : false;
}
DType GetHorizonUsing(const Grid& G,std::size_t V) {//横方向検査
	DType D;
	for (std::size_t i = 0; i < G[V].size(); i++) {
		if(G[V][i]!=0) D.push_back(G[V][i]);
	}
	return  D;
}
bool HaveHorizonZero(const Grid& G,std::size_t V) {//横方向検査
	DType D;
	for (std::size_t i = 0; i < G[V].size(); i++) {
		if (G[V][i] == 0)return true;
	}
	return false;
}
bool IsVerticalValid(const Grid& G,std::size_t H) {//縦方向検査
	DType D;
	for (std::size_t i = 0; i < G.size(); i++) {
		if(G[i][H]!= 0)D.push_back(G[i][H]);
	}
	std::sort(D.begin(), D.end());
	D.erase(std::unique(D.begin(), D.end()), D.end());
	return D.size() == 9 ? true : false;
}DType GetVerticalUsing(const Grid& G,std::size_t H) {//横方向検査
	DType D;
	DType T{ 0,1,2,3,4,5,6,7,8 };
	for (std::size_t i = 0; i < G.size(); i++) {
		if(G[i][H]!= 0)D.push_back(G[i][H]);
	}

	return  D;
}
bool HaveVerticalZero(const Grid& G,std::size_t H) {//縦方向検査
	DType D;
	for (std::size_t i = 0; i < G.size(); i++) {
		if (G[i][H] == 0)return true;
	}
	return false;
}
bool IsBlockValid(const Grid& G, std::size_t X, std::size_t Y) {//3x3検査
	DType D;
	for (std::size_t i = 0; i < 3; i++) {
		for (std::size_t j = 0; j < 3; j++) {
			if(G[Y*3+i][X*3+j] !=0)D.push_back(G[Y*3+i][X*3+j]);
		}
	}
	std::sort(D.begin(), D.end());
	D.erase(std::unique(D.begin(), D.end()), D.end());
	return D.size() == 9 ? true : false;
}
DType GetBlockUsing(const Grid& G,std::size_t X,std::size_t Y) {//横方向検査
	DType D;
	for (std::size_t i = 0; i < 3; i++) {
		for (std::size_t j = 0; j < 3; j++) {
			if(G[Y*3+i][X*3+j] !=0) D.push_back(G[Y*3+i][X*3+j]);
		}
	}

	return D;
}
bool HaveBlockNumber(const Grid& G, std::size_t X, std::size_t Y, std::size_t N) {//横方向検査
	DType D;
	for (std::size_t i = 0; i < 3; i++) {
		for (std::size_t j = 0; j < 3; j++) {
			if (G[Y * 3 + i][X * 3 + j] == N)return true;
		}
	}

	return false;
}
bool HaveBlockZero(const Grid& G, std::size_t X, std::size_t Y) {//3x3検査
	DType D;
	for (std::size_t i = 0; i < 3; i++) {
		for (std::size_t j = 0; j < 3; j++) {
			if (G[Y * 3 + i][X * 3 + j] == 0)return true;
		}
	}

	return false;
}


bool Search_r(Grid& G, int Deep, RType& R, std::size_t& C) {
	C++;
	gotoxy(11, 0);
//	std::cout << C << std::endl;
	if (Deep >= 81) {
		gotoxy(0, 0);
		//Show(G);
		if (IsAnswer(G)) {
			R.push_back(G);
			std::cout << "find!!" << std::endl;
			return true;
		}

		return false;
	}	
	
	static const DType T{ 1,2,3,4,5,6,7,8,9 };
	std::size_t MX = (Deep % 9) % 3;
	std::size_t MY = (Deep % 9) / 3;
	std::size_t BX = (Deep / 9) % 3;
	std::size_t BY = (Deep / 9) / 3;
	auto BU = GetBlockUsing(G, BX, BY);
	auto HU = GetHorizonUsing(G, BY*3+MY);
	auto VU = GetVerticalUsing(G, BX*3 + MX);
	DType AI;
	
	if (G[BY*3 + MY][BX*3 + MX] != 0) {
		return Search_r(G, Deep + 1, R,C);
	}
	
	std::copy(BU.begin(), BU.end(), std::back_inserter(AI));
	std::copy(HU.begin(), HU.end(), std::back_inserter(AI));
	std::copy(VU.begin(), VU.end(), std::back_inserter(AI));
	std::sort(AI.begin(), AI.end());
	AI.erase(std::unique(AI.begin(), AI.end()), AI.end());
	DType D = GetAntiIntersection(T, AI);

	auto WG = G;
	if (D.size() == 0) return false;
	for (std::size_t i = 0; i < D.size(); i++) {
		WG[BY*3 + MY][BX*3 + MX] = D[i];
		gotoxy(0, 0);
//		Show(WG);
		Search_r(WG, Deep + 1, R,C);
	}

	return false;

}
bool Search(Grid& G, RType& R) {
	int Deep = 0;
	std::size_t C=0;
	return Search_r(G, Deep, R, C);
}


RType MakeHoge(Grid G) {
	Cube Cu = MakeCube();
	Cube C = ReduceCube(Cu, G);
	Grid V = MakeGrid();
	RType R;
	Grid B = G;
	Grid D;
	std::uint64_t Co = 0;
	std::uint64_t Find = 1;
	do {
		D = B;
		B = CubeToProblem(C);
		C = ReduceCube(C, B);
	} while (D != B);
	gotoxy(0, 11);
//	Show(B);
//	ShowList(C);
	auto A = CubeToProblem(C);
	gotoxy(0, 11);
//	Show(A);
	Search(A, R);

	return R;
}

int main() {
	Grid G = MakeProblem();
	RType R;
	Show(G);
	R = MakeHoge(G);
	for (auto& o : R) {
		gotoxy(0, 0);
		Show(o);
	}

	return 0;
}