#pragma GCC optimize ("O3") #pragma GCC target ("avx") #include "bits/stdc++.h" // define macro "/D__MAI" using namespace std; typedef unsigned int uint; typedef long long int ll; typedef unsigned long long int ull; #define debugv(v) printf("L%d %s => ",__LINE__,#v);for(auto e:v){cout< ",__LINE__,#m);for(int x=0;x<(w);x++){cout<<(m)[x]<<" ";}cout<=l;--cnt) #define BIGINT 0x7FFFFFFF #define MD 1000000007ll #define PI 3.1415926535897932384626433832795 template ostream& operator <<(ostream &o, const pair p) { o << "(" << p.first << ":" << p.second << ")"; return o; } #define TIME chrono::system_clock::now() #define MILLISEC(t) (chrono::duration_cast(t).count()) namespace { std::chrono::system_clock::time_point ttt; void tic() { ttt = TIME; } void toc() { fprintf(stderr, "TIME : %lldms\n", MILLISEC(TIME - ttt)); } std::chrono::system_clock::time_point tle = TIME; #ifdef __MAI void safe_tle(int msec) { assert(MILLISEC(TIME - tle) < msec); } #else #define safe_tle(k) ; #endif } #ifdef __MAI //_getchar_nolock #define getchar_unlocked getchar #endif namespace { class MaiScanner { public: template void input_integer(T& var) { var = 0; T sign = 1; int cc = getchar_unlocked(); for (; cc<'0' || '9'>(int& var) { input_integer(var); return *this; } inline MaiScanner& operator>>(long long& var) { input_integer(var); return *this; } }; } MaiScanner scanner; // // // 存在判定であれば,全列挙する必要はなく O(M) で出来る ・・・ // // void yes() { cout << "YES" << endl; exit(0); } struct Vdata { int large = 0; int small = 0; }; int vertex[1010]; Vdata memo[1010]; int main() { int n, m; scanner >> n >> m; repeat(n) { scanner >> vertex[cnt]; } bool h = false; repeat(m) { int u, v; scanner >> u >> v; --u; --v; // ある辺の両端が同じ値なら,その辺は無視. if (vertex[u] == vertex[v])continue; if (vertex[u] > vertex[v]) { // もし,u が v 以外に vertex[u] より小さい頂点 x に隣接していて, // その値が vertex[v] と異なるならば yes() // 式で書くと,vertex[x] < vertex[u] > vertex[v] && vertex[x] != vertex[v] if (memo[u].small && memo[u].small != vertex[v]) yes(); // もし,v が u 以外に vertex[v] より大きい頂点 x に隣接していて, // その値が vertex[u] と異なるならば yes() // 式で書くと,vertex[x] > vertex[v] < vertex[u] && vertex[x] != vertex[v] if (memo[v].large && memo[v].large != vertex[u]) yes(); // u には,自分自身u より小さい値を持つ v と接続していることが分かった. memo[u].small = vertex[v]; // v には,自分自身v より大きい値を持つ u と接続していることが分かった. memo[v].large = vertex[u]; } else { // ほとんど同じ // vertex[x] > vertex[u] < vertex[v] && vertex[x] != vertex[v] if (memo[u].large && memo[u].large != vertex[v]) yes(); // vertex[x] < vertex[v] > vertex[u] && vertex[x] != vertex[v] if (memo[v].small && memo[v].small != vertex[u]) yes(); memo[u].large = vertex[v]; memo[v].small = vertex[u]; } } cout << "NO" << endl; return 0; }