/// read all the words into a trie, marking who has what word
// minimax like strategy
import java.util.Scanner;
import java.util.StringJoiner;
class COCI20_vlak {
    static class Trie {
        TrieNode head;
        public void add(char[] s, boolean belongsToNina) {
            TrieNode cur = head;
            int i = 0;
            if (belongsToNina) {
                cur.num_ninaswords_on_or_below++;
            } else {
                cur.num_emiljaswords_on_or_below++;
            }
            while (i < s.length) {
                if (cur.leaves[s[i]-'a'] == null) {
                    cur.leaves[s[i]-'a'] = new TrieNode();
                }
                cur = cur.leaves[s[i]-'a'];
                if (belongsToNina) {
                    cur.num_ninaswords_on_or_below++;
                } else {
                    cur.num_emiljaswords_on_or_below++;
                }
                
                i++;
            }
            cur.isTerminal=true;
            /*if (belongsToNina) {
                cur.num_ninaswords_on_or_below++;
            } else {
                cur.num_emiljaswords_on_or_below++;
            }*/
        }
        public Trie() {
            head = new TrieNode();
        }
        public String toString() {
            return this.head.toString();
        }
    }
    static class TrieNode {
        long num_ninaswords_on_or_below;
        long num_emiljaswords_on_or_below;
        boolean isTerminal;
        TrieNode[] leaves;
        public TrieNode() {
            num_emiljaswords_on_or_below=0;
            num_ninaswords_on_or_below=0;
            isTerminal=false;
            //this.belongsToNina = belongsToNina;
            leaves = new TrieNode[26];
        }
        public String toString() {
            StringBuilder sb = new StringBuilder();
            sb.append("<TrieNode nina="+num_ninaswords_on_or_below+" emilja="+num_emiljaswords_on_or_below+" leaves=[");
            StringJoiner sj = new StringJoiner(", ");
            for (int i = 0; i<26; i++) {
                if (leaves[i]!=null) {
                    sj.add((char) (i+'a') + ": " + leaves[i].toString());
                }
            }
            sb.append(sj.toString());
            sb.append("]>");
            return new String(sb);
        }
        public boolean hasNoChildren() {
            for (int i = 0; i<leaves.length; i++) {
                if (leaves[i] != null) return false;
            }
            return true;
        }
    }
    public static void main(String[] args) {
        System.err.println("I am ready");
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        Trie trie = new Trie();
        for (int i = 0; i<n; i++) {
            trie.add(sc.next().toCharArray(), true);
        }
        int m = sc.nextInt();
        for (int i = 0; i<m; i++) {
            trie.add(sc.next().toCharArray(), false);
        }
        sc.close();
       // System.out.println("THIS IS MY TRIE: " + trie + "\n\n\n");
        boolean ans = minimax(trie.head, true);
        System.out.println(ans ? "Nina" : "Emilja");
    }
    public static boolean minimax(TrieNode node, boolean maximizingPlayer) { // returns whether i win
        if (node==null) return true;
        if (maximizingPlayer && node.num_ninaswords_on_or_below == 0) return false;
        if (!maximizingPlayer && node.num_emiljaswords_on_or_below == 0) return false;
        /*boolean canMakeMove = false;
        for (int i = 0; i<node.leaves.length; i++) {
            if (node.leaves[i] != null) {
                if (
                    (maximizingPlayer && node.leaves[i].num_ninaswords_below>0)
                    ||(!maximizingPlayer && node.leaves[i].num_emiljaswords_below>0)) 
                    canMakeMove=true;
            }
        }
        if (!canMakeMove) return false;*/
        for (int i = 0; i<node.leaves.length; i++) {
            if (node.leaves[i] != null) {
                if (!minimax(node.leaves[i], !maximizingPlayer)) return true;
            }
        }
        return false;
    }
}
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... | 
| # | Verdict | Execution time | Memory | Grader output | 
|---|
| Fetching results... |