Misplaced Pages

Feature hashing: Difference between revisions

Article snapshot taken from Wikipedia with creative commons attribution-sharealike license. Give it a read and then ask your questions in the chat. We can research this topic together.
Browse history interactively← Previous editContent deleted Content addedVisualWikitext
Revision as of 13:19, 6 February 2014 editQwertyus (talk | contribs)Extended confirmed users31,640 edits Motivating example: Heaps' law← Previous edit Latest revision as of 18:26, 13 May 2024 edit undoCitation bot (talk | contribs)Bots5,444,439 edits Add: date, arxiv, authors 1-1. Removed parameters. Some additions/deletions were parameter name changes. | Use this bot. Report bugs. | Suggested by Dominic3203 | Category:Machine learning | #UCB_Category 114/230 
(103 intermediate revisions by 53 users not shown)
Line 1: Line 1:
{{short description|Vectorizing features using a hash function}}
In ], '''feature hashing''', also known as the '''hashing trick'''<ref name="Weinberger"/> (by analogy to the ]), is a fast and space-efficient way of vectorizing ], i.e. turning arbitrary features into indices in a vector or matrix. It works by applying a ] to the features and using their hash values as indices directly, rather than looking the indices up in an ].
In ], '''feature hashing''', also known as the '''hashing trick''' (by analogy to the ]), is a fast and space-efficient way of vectorizing ], i.e. turning arbitrary features into indices in a vector or matrix.<ref name="Moody">{{cite journal |last1=Moody |first1=John |title=Fast learning in multi-resolution hierarchies |journal=Advances in Neural Information Processing Systems |date=1989 |url=http://papers.nips.cc/paper/175-fast-learning-in-multi-resolution-hierarchies.pdf}}</ref><ref name="Weinberger"/> It works by applying a ] to the features and using their hash values as indices directly (after a modulo operation), rather than looking the indices up in an ]. In addition to its use for encoding non-numeric values, feature hashing can also be used for ].<ref name="Weinberger" />


This trick is often attributed to Weinberger et al. (2009),<ref name="Weinberger" /> but there exists a much earlier description of this method published by John Moody in 1989.<ref name="Moody" />
==Motivating example==
In a typical ] task, the input to the machine learning algorithm (both during learning and classification) is free text. From this, a ] (BOW) representation is constructed: the individual ] are extracted and counted, and each distinct token in the training set defines a feature (dependent variable) of each of the documents in both the training and test sets.


==Motivation==
Machine learning algorithms, however, are typically defined in terms of numerical vectors. Therefore, the bags of words for a set of documents is regarded as a ] where each row is a single document, and each column is a single feature/word; the entry {{math|''i'', ''j''}} in such a matrix captures the frequency (or weight) of the {{mvar|j}}'th term of the ''vocabulary'' in document {{mvar|i}}. (An alternative convention swaps the rows and columns of the term-document matrix, but this difference is immaterial.)
Typically, these vectors are extremely ].


=== Motivating example ===
The common approach is to construct, at learning time or prior to that, a ''dictionary'' representation of the vocabulary of the training set, and use that to map words to indices. ]s and ]s are common candidates for dictionary implementation. E.g., the texts
In a typical ] task, the input to the machine learning algorithm (both during learning and classification) is free text. From this, a ] (BOW) representation is constructed: the individual ] are extracted and counted, and each distinct token in the training set defines a ] (independent variable) of each of the documents in both the training and test sets.


Machine learning algorithms, however, are typically defined in terms of numerical vectors. Therefore, the bags of words for a set of documents is regarded as a ] where each row is a single document, and each column is a single feature/word; the entry {{math|''i'', ''j''}} in such a matrix captures the frequency (or weight) of the {{mvar|j}}'th term of the ''vocabulary'' in document {{mvar|i}}. (An alternative convention swaps the rows and columns of the matrix, but this difference is immaterial.)
John likes to watch movies. Mary likes movies too.
Typically, these vectors are extremely ]—according to ].
John also likes to watch football games.
An apple a day keeps the doctor away.


The common approach is to construct, at learning time or prior to that, a ''dictionary'' representation of the vocabulary of the training set, and use that to map words to indices. ]s and ]s are common candidates for dictionary implementation. E.g., the three documents
can be converted, using the dictionary (in ] notation):


{"John": 1, "likes": 2, "to": 3, "watch": 4, "movies": 5, "also": 6, "football": 7, "games": 8, "Mary": 9, "too": 10, * ''John likes to watch movies. ''
* ''Mary likes movies too.''
"an": 11, "apple": 12, "a": 13, "day": 14, "keeps": 15, "the": 16, "doctor": 17, "away": 18}
* ''John also likes football.''


can be converted, using the dictionary
to the matrix


{| class="wikitable"
:<math>\begin{bmatrix}
|-
1 & 2 & 1 & 1 & 2 & 0 & 0 & 0 & 1 & 1 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 \\
! Term !! Index
1 & 1 & 1 & 1 & 0 & 1 & 1 & 1 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 \\
|-
0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 0 & 1 & 1 & 1 & 1 & 1 & 1 & 1 & 1
| John || 1
\end{bmatrix}</math>
|-
| likes || 2
|-
| to || 3
|-
| watch || 4
|-
| movies || 5
|-
| Mary || 6
|-
| too || 7
|-
| also || 8
|-
| football || 9
|}

to the term-document matrix

:<math>
\begin{pmatrix}
\textrm{John} & \textrm{likes} & \textrm{to} & \textrm{watch} & \textrm{movies} & \textrm{Mary} & \textrm{too} & \textrm{also} & \textrm{football} \\
1 & 1 & 1 & 1 & 1 & 0 & 0 & 0 & 0 \\
0 & 1 & 0 & 0 & 1 & 1 & 1 & 0 & 0 \\
1 & 1 & 0 & 0 & 0 & 0 & 0 & 1 & 1
\end{pmatrix}
</math>


(Punctuation was removed, as is usual in document classification and clustering.) (Punctuation was removed, as is usual in document classification and clustering.)


The problem with this process is that such dictionaries take up a large amount of storage space, and ].<ref name="mobilenlp">{{cite conference |author1=K. Ganchev |author2=M. Dredze |year=2008 |url=http://www.cs.jhu.edu/~mdredze/publications/mobile_nlp_feature_mixing.pdf |title=Small statistical models by random feature mixing |conference=Proc. ACL08 HLT Workshop on Mobile Language Processing}}</ref> The problem with this process is that such dictionaries take up a large amount of storage space and grow in size as the training set grows.<ref name="mobilenlp">{{cite conference |author1=K. Ganchev |author2=M. Dredze |year=2008 |url=http://www.cs.jhu.edu/~mdredze/publications/mobile_nlp_feature_mixing.pdf |title=Small statistical models by random feature mixing |conference=Proc. ACL08 HLT Workshop on Mobile Language Processing}}</ref> On the contrary, if the vocabulary is kept fixed and not increased with a growing training set, an adversary may try to invent new words or misspellings that are not in the stored vocabulary so as to circumvent a machine learned filter. To address this challenge, ] attempted to use feature hashing for their spam filters.<ref>{{cite journal |author1=Josh Attenberg |author2=Kilian Weinberger |author3=Alex Smola |author4=Anirban Dasgupta |author5=Martin Zinkevich |title=Collaborative spam filtering with the hashing trick |journal=Virus Bulletin |year=2009|url=https://www.virusbulletin.com/virusbulletin/2009/11/collaborative-spam-filtering-hashing-trick}}</ref>

This can be seen in the above example: the third "document" has no terms in common with the other two, and therefore extends the vocabulary with its eight distinct words.
Note that the hashing trick isn't limited to text classification and similar tasks at the document level, but can be applied to any problem that involves large (perhaps unbounded) numbers of features.
Moreover, when the vocabulary is kept fixed, an adversary may try to invent new words or misspellings that are not in the stored vocabulary so as to circumvent a machine learned filter; this is why feature hashing has been tried for ] at ].<ref>{{cite journal |author1=Josh Attenberg |author2=Kilian Weinberger |author3=Alex Smola |author4=Anirban Dasgupta |author5=Martin Zinkevich |title=Collaborative spam filtering with the hashing trick |journal=Virus Bulletin |year=2009}}</ref>

=== Mathematical motivation ===
Mathematically, a token is an element <math>t </math> in a finite (or countably infinite) set <math>T </math>. Suppose we only need to process a finite corpus, then we can put all tokens appearing in the corpus into <math>T </math>, meaning that <math>T </math> is finite. However, suppose we want to process all possible words made of the English letters, then <math>T </math> is countably infinite.

Most neural networks can only operate on real vector inputs, so we must construct a "dictionary" function <math>\phi: T\to \R^n</math>.

When <math>T </math> is finite, of size <math>|T| = m \leq n</math>, then we can use ] to map it into <math>\R^n</math>. First, ''arbitrarily'' enumerate <math>T = \{t_1, t_2, .., t_m\} </math>, then define <math>\phi(t_i) = e_i </math>. In other words, we assign a unique index <math>i </math> to each token, then map the token with index <math>i </math> to the unit basis vector <math>e_i </math>.

One-hot encoding is easy to interpret, but it requires one to maintain the arbitrary enumeration of <math>T </math>. Given a token <math>t\in T </math>, to compute <math>\phi(t) </math>, we must find out the index <math>i </math> of the token <math>t </math>. Thus, to implement <math>\phi </math> efficiently, we need a fast-to-compute bijection <math>h: T\to \{1, ..., m\} </math>, then we have <math>\phi(t) = e_{h(t)} </math>.

In fact, we can relax the requirement slightly: It suffices to have a fast-to-compute ''injection'' <math>h: T\to \{1, ..., n\} </math>, then use <math>\phi(t) = e_{h(t)} </math>.

In practice, there is no simple way to construct an efficient injection <math>h: T\to \{1, ..., n\} </math>. However, we do not need a strict injection, but only an ''approximate'' injection. That is, when <math>t\neq t' </math>, we should ''probably'' have <math>h(t) \neq h(t') </math>, so that ''probably'' <math>\phi(t) \neq \phi(t') </math>.

At this point, we have just specified that <math>h </math> should be a hashing function. Thus we reach the idea of feature hashing.

==Algorithms==

=== Feature hashing (Weinberger et al. 2009) ===
The basic feature hashing algorithm presented in (Weinberger et al. 2009)<ref name="Weinberger" /> is defined as follows.

First, one specifies two hash functions: the '''kernel hash''' <math>h: T\to \{1, 2, ..., n\}</math>, and the '''sign hash''' <math>\zeta: T\to \{-1, +1\} </math>. Next, one defines the feature hashing function:<math display="block">\phi: T\to \R^n, \quad \phi(t) = \zeta(t) e_{h(t)} </math>Finally, extend this feature hashing function to strings of tokens by<math display="block">\phi: T^* \to \R^n, \quad \phi(t_1, ..., t_k) =\sum_{j=1}^k \phi(t_j) </math>where <math>T^* </math> is the set of ] in <math>T </math>.

Equivalently,<math display="block">\phi(t_1, ..., t_k) = \sum_{j=1}^k \zeta(t_j) e_{h(t_j)} = \sum_{i=1}^n \left(\sum_{j: h(t_j)=i} \zeta(t_j)\right) e_i </math>

==== Geometric properties ====
We want to say something about the geometric property of <math>\phi</math>, but <math>T</math>, by itself, is just a set of tokens, we cannot impose a geometric structure on it except the discrete topology, which is generated by the ]. To make it nicer, we lift it to <math>T\to \R^T</math>, and lift <math>\phi</math> from <math>\phi: T \to \R^n</math> to <math>\phi: \R^T \to \R^n</math> by linear extension: <math display="block">\phi((x_t)_{t\in T}) = \sum_{t\in T} x_t \zeta(t) e_{h(t)} = \sum_{i=1}^n \left(\sum_{t: h(t)=i} x_t \zeta(t)\right) e_i </math>There is an infinite sum there, which must be handled at once. There are essentially only two ways to handle infinities. One may impose a metric, then take its ], to allow well-behaved infinite sums, or one may demand that nothing is ]. Here, we go for the potential-infinity way, by restricting <math>\R^T</math> to contain only vectors with ]: <math>\forall (x_t)_{t\in T} \in \R^T</math>, only finitely many entries of <math>(x_t)_{t\in T}</math> are nonzero.

Define an ] on <math>\R^T</math> in the obvious way: <math display="block">\langle e_t, e_{t'} \rangle = \begin{cases}
1, \text{ if }t=t', \\
0, \text{ else.}
\end{cases}
\quad \langle x, x'\rangle = \sum_{t, t'\in T} x_t x_{t'} \langle e_t, e_{t'} \rangle</math>As a side note, if <math>T</math> is infinite, then the inner product space <math>\R^T</math> is not ]. Taking its completion would get us to a ], which allows well-behaved infinite sums.


Now we have an inner product space, with enough structure to describe the geometry of the feature hashing function <math>\phi: \R^T \to \R^n</math>.

First, we can see why <math> h</math> is called a "'''kernel hash'''": it allows us to define a ] <math> K: T\times T \to \R</math> by<math display="block"> K(t, t') = \langle e_{h(t)}, e_{h(t')}\rangle</math>In the language of the "kernel trick", <math> K</math> is the kernel generated by the "feature map" <math display="block"> \varphi : T \to \R^n, \quad \varphi(t) = e_{h(t)}</math>Note that this is not the feature map we were using, which is <math>\phi(t) = \zeta(t) e_{h(t)} </math>. In fact, we have been using ''another kernel'' <math> K_{\zeta}: T\times T \to \R</math>, defined by <math display="block"> K_\zeta (t, t') = \langle \zeta(t)e_{h(t)}, \zeta(t')e_{h(t')}\rangle</math>The benefit of augmenting the kernel hash <math> h</math> with the binary hash <math> \zeta</math> is the following theorem, which states that <math> \phi</math> is an isometry "on average".

{{Math theorem
| name = Theorem (intuitively stated)
| math_statement = If the binary hash <math> \zeta</math> is unbiased (meaning that it takes value <math> -1, +1</math> with equal probability), then <math>\phi: \R^T \to \R^n</math> is an isometry in expectation:<math display="block"> \mathbb{E} = \langle x, x' \rangle.</math>
}}{{Proof| proof=

By linearity of expectation, <math display="block"> \mathbb{E} = \sum_{t, t'\in T} (x_t x'_{t'}) \cdot \mathbb{E} \cdot \langle e_{h(t)}, e_{h(t')}\rangle</math>Now, <math> \mathbb{E} =\begin{cases}
1 \quad \text{ if }t=t'\\
0 \quad \text{ if }t \neq t'\\
\end{cases}</math>, since we assumed <math> \zeta</math> is unbiased. So we continue<math display="block"> \mathbb{E} = \sum_{t\in T} (x_t x'_{t}) \langle e_{h(t)}, e_{h(t)}\rangle = \langle x , x'\rangle</math>

|title=Proof}}



The above statement and proof interprets the binary hash function <math> \zeta</math> not as a deterministic function of type <math> T\to \{-1, +1\}</math>, but as a random binary vector <math> \{-1, +1\}^T</math> with unbiased entries, meaning that <math> Pr(\zeta(t) = +1) = Pr(\zeta(t) = -1) = \frac 1 2</math> for any <math> t\in T</math>.


This is a good intuitive picture, though not rigorous. For a rigorous statement and proof, see <ref name="Weinberger" />
Note that the hashing trick isn't limited to text classification and similar tasks at the document level, but can be applied to any problem that involves large (perhaps unbounded) amounts of features.


==== Pseudocode implementation ====
==Feature vectorization using the hashing trick==
Instead of maintaining a dictionary, a feature vectorizer that uses the hashing trick can build a vector of a pre-defined length by applying a hash function {{mvar|h}} to the features (e.g., words) in the items under consideration, then using the hash values directly as feature indices and updating the resulting vector at those indices: Instead of maintaining a dictionary, a feature vectorizer that uses the hashing trick can build a vector of a pre-defined length by applying a hash function {{mvar|h}} to the features (e.g., words), then using the hash values directly as feature indices and updating the resulting vector at those indices. Here, we assume that feature actually means feature vector.
<source lang="pascal"> <syntaxhighlight lang="pascal">
function hashing_vectorizer(features : array of string, N : integer): function hashing_vectorizer(features : array of string, N : integer):
x := new vector x := new vector
Line 43: Line 125:
x += 1 x += 1
return x return x
</syntaxhighlight>
</source>
Thus, if our feature vector is and hash function is <math> h(x_f)=1</math> if <math>x_f</math> is "cat" and <math>2</math> if <math>x_f</math> is "dog". Let us take the output feature vector dimension ({{mono|N}}) to be 4. Then output {{mono|x}} will be .
It has been suggested that a second, single-bit output hash function {{mvar|ξ}} be used to determine the sign of the update value, to counter the effect of ]s.<ref name="Weinberger">{{cite conference |author1=Kilian Weinberger |author2=Anirban Dasgupta |author3=John Langford |author4=Alex Smola |author5=Josh Attenberg |year=2009 |url=http://alex.smola.org/papers/2009/Weinbergeretal09.pdf |title=Feature Hashing for Large Scale Multitask Learning |conference=Proc. ICML}}</ref> If such a hash function is used, the algorithm becomes
It has been suggested that a second, single-bit output hash function {{mvar|ξ}} be used to determine the sign of the update value, to counter the effect of ]s.<ref name="Weinberger">{{cite conference |author1=Kilian Weinberger |author2=Anirban Dasgupta |author3=John Langford |author4=Alex Smola |author5=Josh Attenberg |year=2009 |url=http://alex.smola.org/papers/2009/Weinbergeretal09.pdf |title=Feature Hashing for Large Scale Multitask Learning |conference=Proc. ICML}}</ref> If such a hash function is used, the algorithm becomes
<source lang="pascal">
<syntaxhighlight lang="pascal">
function hashing_vectorizer(features : array of string, N : integer): function hashing_vectorizer(features : array of string, N : integer):
x := new vector x := new vector
Line 56: Line 139:
x -= 1 x -= 1
return x return x
</syntaxhighlight>
</source>


The above pseudocode actually converts each sample into a vector. An optimized version would instead only generate a stream of ({{mvar|h}},{{mvar|ξ}}) pairs and let the learning and prediction algorithms consume such streams; a ] can then be implemented as a single hash table representing the coefficient vector. The above pseudocode actually converts each sample into a vector. An optimized version would instead only generate a stream of <math>(h, \zeta) </math> pairs and let the learning and prediction algorithms consume such streams; a ] can then be implemented as a single hash table representing the coefficient vector.


===Extensions and variations===
===Properties===
When a second hash function ''ξ'' is used to determine the sign of a feature's value, the ] ] of each column in the output array becomes zero because ''ξ'' causes some collisions to cancel out.<ref name="Weinberger"/> E.g., suppose an input contains two symbolic features ''f''₁ and ''f''₂ that collide with each other, but not with any other features in the same input; then there are four possibilities which, if we make no assumptions about ''ξ'', have equal probability:


==== Learned feature hashing ====
{| class="wikitable"
Feature hashing generally suffers from hash collision, which means that there exist pairs of different tokens with the same hash: <math> t\neq t', \phi(t) = \phi(t') = v</math>. A machine learning model trained on feature-hashed words would then have difficulty distinguishing <math> t</math> and <math> t'</math>, essentially because <math> v</math> is ].
|-
! ''ξ''(''f''₁) !! ''ξ''(''f''₂) !! Final value, ''ξ''(''f''₁) + ''ξ''(''f''₂)
|-
| -1 || -1 || -2
|-
| -1 || 1 || 0
|-
| 1 || -1 || 0
|-
| 1 || 1 || 2
|}


If <math> t'</math> is rare, then performance degradation is small, as the model could always just ignore the rare case, and pretend all <math> v</math> means <math> t</math>. However, if both are common, then the degradation can be serious.
In this example, there is a 50% probability that the hash collision cancels out. Multiple hash functions can be used to further reduce the risk of collisions.<ref name="mahout">{{cite book
|last1 = Owen
|first1 = Sean
|last2 = Anil
|first2 = Robin
|last3 = Dunning
|first3 = Ted
|last4 = Friedman
|first4 = Ellen
|title=Mahout in Action
|pages=261–265
|publisher = Manning
|year = 2012
}}</ref>


To handle this, one can train supervised hashing functions that avoids mapping common tokens to the same feature vectors.<ref>{{cite conference|last=Bai|first=B.|author2=Weston J. |author3=Grangier D. |author4=Collobert R. |author5=Sadamasa K. |author6=Qi Y. |author7=Chapelle O. |author8=Weinberger K. |title=Supervised semantic indexing|conference=CIKM|year=2009|pages=187–196|url=http://www.cs.cornell.edu/~kilian/papers/ssi-cikm.pdf}}</ref>
Furthermore, if ''φ'' is the transformation implemented by a hashing trick with a sign hash ''ξ'' (i.e. ''φ''(''x'') is the feature vector produced for a sample ''x''), then ]s in the hashed space are unbiased:


== Applications and practical performance ==
:<math> \mathbb{E} = \langle x, x' \rangle</math>
Ganchev and Dredze showed that in text classification applications with random hash functions and several tens of thousands of columns in the output vectors, feature hashing need not have an adverse effect on classification performance, even without the signed hash function.<ref name="mobilenlp"/>


Weinberger et al. (2009) applied their version of feature hashing to ], and in particular, ]ing, where the input features are pairs (user, feature) so that a single parameter vector captured per-user spam filters as well as a global filter for several hundred thousand users, and found that the accuracy of the filter went up.<ref name="Weinberger" />
where the expectation is taken over the hashing function ''φ''.<ref name="Weinberger"/> It can be verified that<math>\langle \varphi(x), \varphi(x') \rangle</math> is a ] ].<ref name="Weinberger"/><ref>{{cite conference|last=Shi|first=Q.|coauthors=Petterson J., Dror G., Langford J., Smola A., Strehl A., Vishwanathan V.|title=Hash Kernels|conference=AISTATS|year=2009}}</ref>


Chen et al. (2015) combined the idea of feature hashing and ] to construct "virtual matrices": large matrices with small storage requirements. The idea is to treat a matrix <math>M\in \R^{n\times n}</math> as a dictionary, with keys in <math>n\times n</math>, and values in <math>\R</math>. Then, as usual in hashed dictionaries, one can use a hash function <math>h: \N\times \N\to m</math>, and thus represent a matrix as a vector in <math>\R^m</math>, no matter how big <math>n</math> is. With virtual matrices, they constructed ''HashedNets'', which are large neural networks taking only small amounts of storage.<ref>{{Cite journal |last1=Chen |first1=Wenlin |last2=Wilson |first2=James |last3=Tyree |first3=Stephen |last4=Weinberger |first4=Kilian |last5=Chen |first5=Yixin |date=2015-06-01 |title=Compressing Neural Networks with the Hashing Trick |url=https://proceedings.mlr.press/v37/chenc15.html |journal=International Conference on Machine Learning |language=en |publisher=PMLR |pages=2285–2294|arxiv=1504.04788 }}</ref>
===Extensions and variations===
Recent work extends the hashing trick to supervised mappings from words to indices,<ref>{{cite conference|last=Bai|first=B.|coauthors=Weston J., Grangier D., Collobert R., Sadamasa K., Qi Y., Chapelle O., Weinberger K.|title=Supervised semantic indexing|conference=CIKM|year=2009|pages=187–196|url=http://www.cse.wustl.edu/~kilian/papers/ssi-cikm.pdf}}</ref>
which are explicitly learned to avoid collisions of important terms.

===Applications and practical performance===
Ganchev and Dredze showed that in text classification applications with random hash functions and several tens of thousands of columns in the output vectors, feature hashing need not have an adverse effect on classification performance, even without the signed hash function.<ref name="mobilenlp"/>
Weinberger et al. applied their variant of hashing to the problem of ]ing, formulating this as a ] problem where the input features are pairs (user, feature) so that a single parameter vector captured per-user spam filters as well as a global filter for several hundred thousand users, and found that the accuracy of the filter went up.<ref name="Weinberger"/>


==Implementations== ==Implementations==
Implementations of the hashing trick are present in: Implementations of the hashing trick are present in:


* ]<ref name="mahout">{{cite book |last1=Owen |first1=Sean |title=Mahout in Action |last2=Anil |first2=Robin |last3=Dunning |first3=Ted |last4=Friedman |first4=Ellen |publisher=Manning |year=2012 |pages=261–265}}</ref>
* ]<ref name="mahout"/>
* ]<ref>{{cite web|url=http://radimrehurek.com/gensim/corpora/hashdictionary.html |title=gensim: corpora.hashdictionary – Construct word<->id mappings |publisher=Radimrehurek.com |accessdate=2014-02-13}}</ref>
* ] ()
* ] () * ]<ref>{{cite web|url=http://scikit-learn.org/stable/modules/feature_extraction.html#feature-hashing |title=4.1. Feature extraction — scikit-learn 0.14 documentation |publisher=Scikit-learn.org |accessdate=2014-02-13}}</ref>
* sofia-ml<ref>{{cite web|url=https://code.google.com/p/sofia-ml/ |title=sofia-ml - Suite of Fast Incremental Algorithms for Machine Learning. Includes methods for learning classification and ranking models, using Pegasos SVM, SGD-SVM, ROMMA, Passive-Aggressive Perceptron, Perceptron with Margins, and Logistic Regression |accessdate=2014-02-13}}</ref>
*
* ] * ]
* ]<ref>{{cite web|url=https://spark.apache.org/docs/latest/api/scala/index.html#org.apache.spark.mllib.feature.HashingTF|title=Hashing TF|quote=Maps a sequence of terms to their term frequencies using the hashing trick.|accessdate=4 September 2015}}</ref>
* ]<ref>{{cite web|url=https://cran.r-project.org/web/packages/FeatureHashing/index.html |title=FeatureHashing: Creates a Model Matrix via Feature Hashing With a Formula Interface|date=10 January 2024 }}</ref>
* ]<ref>{{cite web|url=https://www.tensorflow.org/api_docs/python/tf/keras/preprocessing/text/hashing_trick |title=tf.keras.preprocessing.text.hashing_trick — TensorFlow Core v2.0.1 |quote=Converts a text to a sequence of indexes in a fixed-size hashing space. |accessdate=2020-04-29}}</ref>
*<ref>{{Cite web|title=dask_ml.feature_extraction.text.HashingVectorizer — dask-ml 2021.11.17 documentation|url=https://ml.dask.org/modules/generated/dask_ml.feature_extraction.text.HashingVectorizer.html#dask_ml.feature_extraction.text.HashingVectorizer|access-date=2021-11-22|website=ml.dask.org}}</ref>


==See also== ==See also==

* ]
* {{Annotated link |Bloom filter}}
* ]
* {{Annotated link |Count–min sketch}}
* ]
* {{Annotated link |Heaps' law}}
* ]
* {{Annotated link |Locality-sensitive hashing}}
* {{Annotated link |MinHash}}


==References== ==References==
Line 125: Line 185:
==External links== ==External links==
* on John Langford's website * on John Langford's website
* *


] ]
] ]
]

Latest revision as of 18:26, 13 May 2024

Vectorizing features using a hash function

In machine learning, feature hashing, also known as the hashing trick (by analogy to the kernel trick), is a fast and space-efficient way of vectorizing features, i.e. turning arbitrary features into indices in a vector or matrix. It works by applying a hash function to the features and using their hash values as indices directly (after a modulo operation), rather than looking the indices up in an associative array. In addition to its use for encoding non-numeric values, feature hashing can also be used for dimensionality reduction.

This trick is often attributed to Weinberger et al. (2009), but there exists a much earlier description of this method published by John Moody in 1989.

Motivation

Motivating example

In a typical document classification task, the input to the machine learning algorithm (both during learning and classification) is free text. From this, a bag of words (BOW) representation is constructed: the individual tokens are extracted and counted, and each distinct token in the training set defines a feature (independent variable) of each of the documents in both the training and test sets.

Machine learning algorithms, however, are typically defined in terms of numerical vectors. Therefore, the bags of words for a set of documents is regarded as a term-document matrix where each row is a single document, and each column is a single feature/word; the entry i, j in such a matrix captures the frequency (or weight) of the j'th term of the vocabulary in document i. (An alternative convention swaps the rows and columns of the matrix, but this difference is immaterial.) Typically, these vectors are extremely sparse—according to Zipf's law.

The common approach is to construct, at learning time or prior to that, a dictionary representation of the vocabulary of the training set, and use that to map words to indices. Hash tables and tries are common candidates for dictionary implementation. E.g., the three documents

  • John likes to watch movies.
  • Mary likes movies too.
  • John also likes football.

can be converted, using the dictionary

Term Index
John 1
likes 2
to 3
watch 4
movies 5
Mary 6
too 7
also 8
football 9

to the term-document matrix

( John likes to watch movies Mary too also football 1 1 1 1 1 0 0 0 0 0 1 0 0 1 1 1 0 0 1 1 0 0 0 0 0 1 1 ) {\displaystyle {\begin{pmatrix}{\textrm {John}}&{\textrm {likes}}&{\textrm {to}}&{\textrm {watch}}&{\textrm {movies}}&{\textrm {Mary}}&{\textrm {too}}&{\textrm {also}}&{\textrm {football}}\\1&1&1&1&1&0&0&0&0\\0&1&0&0&1&1&1&0&0\\1&1&0&0&0&0&0&1&1\end{pmatrix}}}

(Punctuation was removed, as is usual in document classification and clustering.)

The problem with this process is that such dictionaries take up a large amount of storage space and grow in size as the training set grows. On the contrary, if the vocabulary is kept fixed and not increased with a growing training set, an adversary may try to invent new words or misspellings that are not in the stored vocabulary so as to circumvent a machine learned filter. To address this challenge, Yahoo! Research attempted to use feature hashing for their spam filters.

Note that the hashing trick isn't limited to text classification and similar tasks at the document level, but can be applied to any problem that involves large (perhaps unbounded) numbers of features.

Mathematical motivation

Mathematically, a token is an element t {\displaystyle t} in a finite (or countably infinite) set T {\displaystyle T} . Suppose we only need to process a finite corpus, then we can put all tokens appearing in the corpus into T {\displaystyle T} , meaning that T {\displaystyle T} is finite. However, suppose we want to process all possible words made of the English letters, then T {\displaystyle T} is countably infinite.

Most neural networks can only operate on real vector inputs, so we must construct a "dictionary" function ϕ : T R n {\displaystyle \phi :T\to \mathbb {R} ^{n}} .

When T {\displaystyle T} is finite, of size | T | = m n {\displaystyle |T|=m\leq n} , then we can use one-hot encoding to map it into R n {\displaystyle \mathbb {R} ^{n}} . First, arbitrarily enumerate T = { t 1 , t 2 , . . , t m } {\displaystyle T=\{t_{1},t_{2},..,t_{m}\}} , then define ϕ ( t i ) = e i {\displaystyle \phi (t_{i})=e_{i}} . In other words, we assign a unique index i {\displaystyle i} to each token, then map the token with index i {\displaystyle i} to the unit basis vector e i {\displaystyle e_{i}} .

One-hot encoding is easy to interpret, but it requires one to maintain the arbitrary enumeration of T {\displaystyle T} . Given a token t T {\displaystyle t\in T} , to compute ϕ ( t ) {\displaystyle \phi (t)} , we must find out the index i {\displaystyle i} of the token t {\displaystyle t} . Thus, to implement ϕ {\displaystyle \phi } efficiently, we need a fast-to-compute bijection h : T { 1 , . . . , m } {\displaystyle h:T\to \{1,...,m\}} , then we have ϕ ( t ) = e h ( t ) {\displaystyle \phi (t)=e_{h(t)}} .

In fact, we can relax the requirement slightly: It suffices to have a fast-to-compute injection h : T { 1 , . . . , n } {\displaystyle h:T\to \{1,...,n\}} , then use ϕ ( t ) = e h ( t ) {\displaystyle \phi (t)=e_{h(t)}} .

In practice, there is no simple way to construct an efficient injection h : T { 1 , . . . , n } {\displaystyle h:T\to \{1,...,n\}} . However, we do not need a strict injection, but only an approximate injection. That is, when t t {\displaystyle t\neq t'} , we should probably have h ( t ) h ( t ) {\displaystyle h(t)\neq h(t')} , so that probably ϕ ( t ) ϕ ( t ) {\displaystyle \phi (t)\neq \phi (t')} .

At this point, we have just specified that h {\displaystyle h} should be a hashing function. Thus we reach the idea of feature hashing.

Algorithms

Feature hashing (Weinberger et al. 2009)

The basic feature hashing algorithm presented in (Weinberger et al. 2009) is defined as follows.

First, one specifies two hash functions: the kernel hash h : T { 1 , 2 , . . . , n } {\displaystyle h:T\to \{1,2,...,n\}} , and the sign hash ζ : T { 1 , + 1 } {\displaystyle \zeta :T\to \{-1,+1\}} . Next, one defines the feature hashing function: ϕ : T R n , ϕ ( t ) = ζ ( t ) e h ( t ) {\displaystyle \phi :T\to \mathbb {R} ^{n},\quad \phi (t)=\zeta (t)e_{h(t)}} Finally, extend this feature hashing function to strings of tokens by ϕ : T R n , ϕ ( t 1 , . . . , t k ) = j = 1 k ϕ ( t j ) {\displaystyle \phi :T^{*}\to \mathbb {R} ^{n},\quad \phi (t_{1},...,t_{k})=\sum _{j=1}^{k}\phi (t_{j})} where T {\displaystyle T^{*}} is the set of all finite strings consisting of tokens in T {\displaystyle T} .

Equivalently, ϕ ( t 1 , . . . , t k ) = j = 1 k ζ ( t j ) e h ( t j ) = i = 1 n ( j : h ( t j ) = i ζ ( t j ) ) e i {\displaystyle \phi (t_{1},...,t_{k})=\sum _{j=1}^{k}\zeta (t_{j})e_{h(t_{j})}=\sum _{i=1}^{n}\left(\sum _{j:h(t_{j})=i}\zeta (t_{j})\right)e_{i}}

Geometric properties

We want to say something about the geometric property of ϕ {\displaystyle \phi } , but T {\displaystyle T} , by itself, is just a set of tokens, we cannot impose a geometric structure on it except the discrete topology, which is generated by the discrete metric. To make it nicer, we lift it to T R T {\displaystyle T\to \mathbb {R} ^{T}} , and lift ϕ {\displaystyle \phi } from ϕ : T R n {\displaystyle \phi :T\to \mathbb {R} ^{n}} to ϕ : R T R n {\displaystyle \phi :\mathbb {R} ^{T}\to \mathbb {R} ^{n}} by linear extension: ϕ ( ( x t ) t T ) = t T x t ζ ( t ) e h ( t ) = i = 1 n ( t : h ( t ) = i x t ζ ( t ) ) e i {\displaystyle \phi ((x_{t})_{t\in T})=\sum _{t\in T}x_{t}\zeta (t)e_{h(t)}=\sum _{i=1}^{n}\left(\sum _{t:h(t)=i}x_{t}\zeta (t)\right)e_{i}} There is an infinite sum there, which must be handled at once. There are essentially only two ways to handle infinities. One may impose a metric, then take its completion, to allow well-behaved infinite sums, or one may demand that nothing is actually infinite, only potentially so. Here, we go for the potential-infinity way, by restricting R T {\displaystyle \mathbb {R} ^{T}} to contain only vectors with finite support: ( x t ) t T R T {\displaystyle \forall (x_{t})_{t\in T}\in \mathbb {R} ^{T}} , only finitely many entries of ( x t ) t T {\displaystyle (x_{t})_{t\in T}} are nonzero.

Define an inner product on R T {\displaystyle \mathbb {R} ^{T}} in the obvious way: e t , e t = { 1 ,  if  t = t , 0 ,  else. x , x = t , t T x t x t e t , e t {\displaystyle \langle e_{t},e_{t'}\rangle ={\begin{cases}1,{\text{ if }}t=t',\\0,{\text{ else.}}\end{cases}}\quad \langle x,x'\rangle =\sum _{t,t'\in T}x_{t}x_{t'}\langle e_{t},e_{t'}\rangle } As a side note, if T {\displaystyle T} is infinite, then the inner product space R T {\displaystyle \mathbb {R} ^{T}} is not complete. Taking its completion would get us to a Hilbert space, which allows well-behaved infinite sums.


Now we have an inner product space, with enough structure to describe the geometry of the feature hashing function ϕ : R T R n {\displaystyle \phi :\mathbb {R} ^{T}\to \mathbb {R} ^{n}} .

First, we can see why h {\displaystyle h} is called a "kernel hash": it allows us to define a kernel K : T × T R {\displaystyle K:T\times T\to \mathbb {R} } by K ( t , t ) = e h ( t ) , e h ( t ) {\displaystyle K(t,t')=\langle e_{h(t)},e_{h(t')}\rangle } In the language of the "kernel trick", K {\displaystyle K} is the kernel generated by the "feature map" φ : T R n , φ ( t ) = e h ( t ) {\displaystyle \varphi :T\to \mathbb {R} ^{n},\quad \varphi (t)=e_{h(t)}} Note that this is not the feature map we were using, which is ϕ ( t ) = ζ ( t ) e h ( t ) {\displaystyle \phi (t)=\zeta (t)e_{h(t)}} . In fact, we have been using another kernel K ζ : T × T R {\displaystyle K_{\zeta }:T\times T\to \mathbb {R} } , defined by K ζ ( t , t ) = ζ ( t ) e h ( t ) , ζ ( t ) e h ( t ) {\displaystyle K_{\zeta }(t,t')=\langle \zeta (t)e_{h(t)},\zeta (t')e_{h(t')}\rangle } The benefit of augmenting the kernel hash h {\displaystyle h} with the binary hash ζ {\displaystyle \zeta } is the following theorem, which states that ϕ {\displaystyle \phi } is an isometry "on average".

Theorem (intuitively stated) — If the binary hash ζ {\displaystyle \zeta } is unbiased (meaning that it takes value 1 , + 1 {\displaystyle -1,+1} with equal probability), then ϕ : R T R n {\displaystyle \phi :\mathbb {R} ^{T}\to \mathbb {R} ^{n}} is an isometry in expectation: E [ ϕ ( x ) , ϕ ( x ) ] = x , x . {\displaystyle \mathbb {E} =\langle x,x'\rangle .}

Proof

By linearity of expectation, E [ ϕ ( x ) , ϕ ( x ) ] = t , t T ( x t x t ) E [ ζ ( t ) ζ ( t ) ] e h ( t ) , e h ( t ) {\displaystyle \mathbb {E} =\sum _{t,t'\in T}(x_{t}x'_{t'})\cdot \mathbb {E} \cdot \langle e_{h(t)},e_{h(t')}\rangle } Now, E [ ζ ( t ) ζ ( t ) ] = { 1  if  t = t 0  if  t t {\displaystyle \mathbb {E} ={\begin{cases}1\quad {\text{ if }}t=t'\\0\quad {\text{ if }}t\neq t'\\\end{cases}}} , since we assumed ζ {\displaystyle \zeta } is unbiased. So we continue E [ ϕ ( x ) , ϕ ( x ) ] = t T ( x t x t ) e h ( t ) , e h ( t ) = x , x {\displaystyle \mathbb {E} =\sum _{t\in T}(x_{t}x'_{t})\langle e_{h(t)},e_{h(t)}\rangle =\langle x,x'\rangle }


The above statement and proof interprets the binary hash function ζ {\displaystyle \zeta } not as a deterministic function of type T { 1 , + 1 } {\displaystyle T\to \{-1,+1\}} , but as a random binary vector { 1 , + 1 } T {\displaystyle \{-1,+1\}^{T}} with unbiased entries, meaning that P r ( ζ ( t ) = + 1 ) = P r ( ζ ( t ) = 1 ) = 1 2 {\displaystyle Pr(\zeta (t)=+1)=Pr(\zeta (t)=-1)={\frac {1}{2}}} for any t T {\displaystyle t\in T} .

This is a good intuitive picture, though not rigorous. For a rigorous statement and proof, see

Pseudocode implementation

Instead of maintaining a dictionary, a feature vectorizer that uses the hashing trick can build a vector of a pre-defined length by applying a hash function h to the features (e.g., words), then using the hash values directly as feature indices and updating the resulting vector at those indices. Here, we assume that feature actually means feature vector.

 function hashing_vectorizer(features : array of string, N : integer):
     x := new vector
     for f in features:
         h := hash(f)
         x += 1
     return x

Thus, if our feature vector is and hash function is h ( x f ) = 1 {\displaystyle h(x_{f})=1} if x f {\displaystyle x_{f}} is "cat" and 2 {\displaystyle 2} if x f {\displaystyle x_{f}} is "dog". Let us take the output feature vector dimension (N) to be 4. Then output x will be . It has been suggested that a second, single-bit output hash function ξ be used to determine the sign of the update value, to counter the effect of hash collisions. If such a hash function is used, the algorithm becomes

 function hashing_vectorizer(features : array of string, N : integer):
     x := new vector
     for f in features:
         h := hash(f)
         idx := h mod N
         if ξ(f) == 1:
             x += 1
         else:
             x -= 1
     return x

The above pseudocode actually converts each sample into a vector. An optimized version would instead only generate a stream of ( h , ζ ) {\displaystyle (h,\zeta )} pairs and let the learning and prediction algorithms consume such streams; a linear model can then be implemented as a single hash table representing the coefficient vector.

Extensions and variations

Learned feature hashing

Feature hashing generally suffers from hash collision, which means that there exist pairs of different tokens with the same hash: t t , ϕ ( t ) = ϕ ( t ) = v {\displaystyle t\neq t',\phi (t)=\phi (t')=v} . A machine learning model trained on feature-hashed words would then have difficulty distinguishing t {\displaystyle t} and t {\displaystyle t'} , essentially because v {\displaystyle v} is polysemic.

If t {\displaystyle t'} is rare, then performance degradation is small, as the model could always just ignore the rare case, and pretend all v {\displaystyle v} means t {\displaystyle t} . However, if both are common, then the degradation can be serious.

To handle this, one can train supervised hashing functions that avoids mapping common tokens to the same feature vectors.

Applications and practical performance

Ganchev and Dredze showed that in text classification applications with random hash functions and several tens of thousands of columns in the output vectors, feature hashing need not have an adverse effect on classification performance, even without the signed hash function.

Weinberger et al. (2009) applied their version of feature hashing to multi-task learning, and in particular, spam filtering, where the input features are pairs (user, feature) so that a single parameter vector captured per-user spam filters as well as a global filter for several hundred thousand users, and found that the accuracy of the filter went up.

Chen et al. (2015) combined the idea of feature hashing and sparse matrix to construct "virtual matrices": large matrices with small storage requirements. The idea is to treat a matrix M R n × n {\displaystyle M\in \mathbb {R} ^{n\times n}} as a dictionary, with keys in n × n {\displaystyle n\times n} , and values in R {\displaystyle \mathbb {R} } . Then, as usual in hashed dictionaries, one can use a hash function h : N × N m {\displaystyle h:\mathbb {N} \times \mathbb {N} \to m} , and thus represent a matrix as a vector in R m {\displaystyle \mathbb {R} ^{m}} , no matter how big n {\displaystyle n} is. With virtual matrices, they constructed HashedNets, which are large neural networks taking only small amounts of storage.

Implementations

Implementations of the hashing trick are present in:

See also

References

  1. ^ Moody, John (1989). "Fast learning in multi-resolution hierarchies" (PDF). Advances in Neural Information Processing Systems.
  2. ^ Kilian Weinberger; Anirban Dasgupta; John Langford; Alex Smola; Josh Attenberg (2009). Feature Hashing for Large Scale Multitask Learning (PDF). Proc. ICML.
  3. ^ K. Ganchev; M. Dredze (2008). Small statistical models by random feature mixing (PDF). Proc. ACL08 HLT Workshop on Mobile Language Processing.
  4. Josh Attenberg; Kilian Weinberger; Alex Smola; Anirban Dasgupta; Martin Zinkevich (2009). "Collaborative spam filtering with the hashing trick". Virus Bulletin.
  5. Bai, B.; Weston J.; Grangier D.; Collobert R.; Sadamasa K.; Qi Y.; Chapelle O.; Weinberger K. (2009). Supervised semantic indexing (PDF). CIKM. pp. 187–196.
  6. Chen, Wenlin; Wilson, James; Tyree, Stephen; Weinberger, Kilian; Chen, Yixin (2015-06-01). "Compressing Neural Networks with the Hashing Trick". International Conference on Machine Learning. PMLR: 2285–2294. arXiv:1504.04788.
  7. Owen, Sean; Anil, Robin; Dunning, Ted; Friedman, Ellen (2012). Mahout in Action. Manning. pp. 261–265.
  8. "gensim: corpora.hashdictionary – Construct word<->id mappings". Radimrehurek.com. Retrieved 2014-02-13.
  9. "4.1. Feature extraction — scikit-learn 0.14 documentation". Scikit-learn.org. Retrieved 2014-02-13.
  10. "sofia-ml - Suite of Fast Incremental Algorithms for Machine Learning. Includes methods for learning classification and ranking models, using Pegasos SVM, SGD-SVM, ROMMA, Passive-Aggressive Perceptron, Perceptron with Margins, and Logistic Regression". Retrieved 2014-02-13.
  11. "Hashing TF". Retrieved 4 September 2015. Maps a sequence of terms to their term frequencies using the hashing trick.
  12. "FeatureHashing: Creates a Model Matrix via Feature Hashing With a Formula Interface". 10 January 2024.
  13. "tf.keras.preprocessing.text.hashing_trick — TensorFlow Core v2.0.1". Retrieved 2020-04-29. Converts a text to a sequence of indexes in a fixed-size hashing space.
  14. "dask_ml.feature_extraction.text.HashingVectorizer — dask-ml 2021.11.17 documentation". ml.dask.org. Retrieved 2021-11-22.

External links

Categories: