This is a port of the open source C implementation of word2vec (https://code.google.com/p/word2vec/). You can browse/contribute the repository via Github. Alternatively you can pull it from the central Maven repositories:
<dependency>
<groupId>com.medallia.word2vec</groupId>
<artifactId>Word2VecJava</artifactId>
<version>0.10.3</version>
</dependency>
For more background information about word2vec and neural network training for the vector representation of words, please see the following papers.
- http://ttic.uchicago.edu/~haotang/speech/1301.3781.pdf
- http://papers.nips.cc/paper/5021-distributed-representations-of-words-and-phrases-and-their-compositionality.pdf
- http://www-personal.umich.edu/~ronxin/pdf/w2vexp.pdf
Note that this isn't a completely faithful rewrite, specifically:
- The original version does a reduction step when learning the vocabulary from the file when the vocab size hits 21 million words, removing any words that do not meet the minimum frequency threshold. This Java port has no such reduction step.
- The original version injects a token into the vocabulary (with a word count of 0) as a substitute for newlines in the input file. This Java port's vocabulary excludes the token.
- The original version does a quicksort which is not stable, so vocabulary terms with the same frequency may be ordered non-deterministically. The Java port does an explicit sort first by frequency, then by the token's lexicographical ordering.
- The original version assumes that sentences are delimited by newline characters and injects a sentence boundary per 1000 non-filtered tokens, i.e. valid token by the vocabulary and not removed by the randomized sampling process. Java port mimics this behavior for now ...
- When the original version encounters an empty line in the input file, it re-processes the first word of the last non-empty line with a sentence length of 0 and updates the random value. Java port omits this behavior.
- The original C documentation indicates that the range should be between 0 and 1e-5, but the default value is 1e-3. This Java port retains that confusing information.
- The random value generated for comparison to determine if a token should be filtered uses a float. This Java port uses double precision for twice the fun.
- The original version includes an unnecessary normalization of the vector for the input query which may lead to tiny inaccuracies. This Java port foregoes this superfluous operation.
- The original version has an O(n * k) algorithm for finding top matches and is hardcoded to 40 matches. This Java port uses Google's lovely com.google.common.collect.Ordering.greatestOf(java.util.Iterator, int) which is O(n + k log k) and takes in arbitrary k.
Note: The k-means clustering option is excluded in the Java port
Please do not hesitate to peek at the source code. It should be readable, concise, and correct. Please feel free to reach out if it is not.
To verify that the project is building correctly, run
./gradlew build && ./gradlew test
It should run 7 tests without any error.
Note: this project requires gradle 2.2+, if you are using older version of gradle, please upgrade it and run:
./gradlew clean test
to have a clean build and re-run the tests.
Andrew Ko (wko@medallia.com)