#include using namespace std; class UnionFind{ private: int ok = 0; vector par,siz; vector> S; public: UnionFind(int N,const vector> &ST){ par.resize(N,-1); siz.resize(N,1); S.resize(N); ok = 0; for(int i=0; ifalse,した->trueを返す. u = root(u),v = root(v); if(u == v) return false; if(S.at(u).size() < S.at(v).size()) swap(u,v); //Union by size. par.at(v) = u; siz.at(u) += siz.at(v); for(auto p : S.at(v)){ if(S.at(u).count(p)) ok++,S.at(u).erase(p); else S.at(u).insert(p); } return true; } bool issame(int u, int v){ //同じ連結成分ならtrue. if(root(u) == root(v)) return true; else return false; } int size(int pos){return siz.at(root(pos));} //posの連結成分の大きさを返す. int query(){return ok;} }; int main() { ios_base::sync_with_stdio(false); cin.tie(nullptr); int N,M; cin >> N >> M; vector> edge(N-1),ST(M); for(auto &[u,v] : edge) cin >> u >> v,u--,v--; for(auto &[u,v] : ST) cin >> u >> v,u--,v--; UnionFind Z(N,ST); vector answer = {0}; for(int i=N-2; i>0; i--){ auto [u,v] = edge.at(i); Z.unite(u,v); answer.push_back(Z.query()); } while(answer.size()) cout << answer.back() << "\n",answer.pop_back(); }