249. Group Shifted Strings (Medium)
Given a string, we can "shift" each of its letter to its successive letter, for example: "abc" -> "bcd". We can keep "shifting" which forms the sequence:
"abc" -> "bcd" -> ... -> "xyz"
Given a list of strings which contains only lowercase alphabets, group all strings that belong to the same shifting sequence.
For example, given: ["abc", "bcd", "acef", "xyz", "az", "ba", "a", "z"],
A solution is:
[
  ["abc","bcd","xyz"],
  ["az","ba"],
  ["acef"],
  ["a","z"]
]
Solution: Hash Table 3ms
The key to this problem is how to identify strings that are in the same shifting sequence. There are different ways to encode this.
In the following code, this manner is adopted: for a string s of length n, we encode its shifting feature as "s[1] - s[0], s[2] - s[1], ..., s[n - 1] - s[n - 2],".
Then we build an unordered_map, using the above shifting feature string as key and strings that share the shifting feature as value. We store all the strings that share the same shifting feature in a vector.
A final note, since the problem statement has given that "az" and "ba" belong to the same shifting sequence. So if s[i] - s[i - 1] is negative, we need to add 26 to it to make it positive and give the correct result. BTW, taking the suggestion of @StefanPochmann, we change the difference from numbers to lower-case alphabets using 'a' + diff.
The code is as follows.
class Solution {
    string shift(string& s) {
        string key;
        int n = s.size();
        for (int i = 1; i < n; ++i) {
            int diff = s[i]-s[i-1];
            if (diff < 0) diff += 26;
            // change the difference from numbers to lower-case alphabets using 'a' + diff
            key += 'a' + diff + '_';
        }
        return key;
    }
public:
    vector<vector<string>> groupStrings(vector<string>& strings) {
        unordered_map<string, vector<string>> m;
        vector<vector<string>> res;
        for (auto s: strings) {
            m[shift(s)].push_back(s);
        }
        for (auto& a: m) {
            res.push_back(a.second);
        }
        return res;
    }
};