Find the longest word in set D that is a subsequence of string SFind the longest common subsequence algorithm...

Why wasn't TEventArgs made contravariant in the standard event pattern in the .NET ecosystem?

How to tell if a BJT is PNP or NPN by looking at the circuit?

Is it a fallacy if someone claims they need an explanation for every word of your argument to the point where they don't understand common terms?

Why am I able to open Wireshark in macOS without root privileges?

Why was Lupin comfortable with saying Voldemort's name?

Is a new Boolean field better than a null reference when a value can be meaningfully absent?

Absorbing damage with Planeswalker

What's a good word to describe a public place that looks like it wouldn't be rough?

False written accusations not made public - is there law to cover this?

Should I reinstall Linux when changing the laptop's CPU?

Cookies - Should the toggles be on?

Cat is tipping over bed-side lamps during the night

A Missing Symbol for This Logo

Why would space fleets be aligned?

How do I append a character to the end of every line in an Excel cell?

How should I handle players who ignore the session zero agreement?

How to use Mathematica to do a complex integrate with poles in real axis?

What would be the rarity of this magic item(s)?

What is the difference between rolling more dice versus fewer dice?

Why zero tolerance on nudity in space?

Dilemma of explaining to interviewer that he is the reason for declining second interview

Eww, those bytes are gross

Why did the villain in the first Men in Black movie care about Earth's Cockroaches?

In mixed effect models, how account for grouped random effects?



Find the longest word in set D that is a subsequence of string S


Find the longest common subsequence algorithm - low speed.txt word-counterPython Word SubsetsRead file with over 300k words and filter through appropriate filter to return list of matching wordsOutputting all possible words which fit a string of lettersFind smallest subset prefixesFind smallest subset prefixes (part 2)minimum window subsequence leetcode dynamic programming solutionLongest Valid ParenthesesLongest word in dictionary that is a subsequence of a given string













3












$begingroup$



Given a string S and a set of words D, find the longest word in D that is a subsequence of S.



Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.



Note: D can appear in any format (list, hash table, prefix tree, etc.)



For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".




  • The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".

  • The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.

  • The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.




I am currently trying to learn how to become a better programmer. I am taking the Google Tech Dev Guide recommended path and this is the first problem. I came up with my own solution, and read Google's first recommended answer is using brute force, and a slight optimization is to at least ensure the dictionary is represented by a hash table or prefix tree so that lookups are efficient.



This is my first time using a dictionary, and I also don't feel confident about using objects and classes. If you could review and recommend better ways to improve my code, maybe by using more (or better) objects, I would appreciate it.



class Main:
def create_dictionary(string):
dictionary = {}
index = 0

for letter in string:
if letter in dictionary:
dictionary[letter].append(index)
else:
dictionary[letter] = [index]
index += 1
return(dictionary)

def get_word_is_substring(word, dictionary):
index_of_last_letter_found = None
for letter in word:
if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
index = 0
while index < len(dictionary[letter]):
if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
index_of_last_letter_found = dictionary[letter][index]
break
else:
index += 1

else:
return False
return True

def replace_word_if_necessary(word, longest_word):
if (longest_word is None) or (len(word) > len(longest_word)):
longest_word = word
return longest_word

def get_longest_word(s, d):
dictionary = Main.create_dictionary(s)
longest_word = None

for word in d:
word_is_substring = Main.get_word_is_substring(word, dictionary)
if word_is_substring:
longest_word = Main.replace_word_if_necessary(word, longest_word)
print(longest_word)

Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})









share|improve this question









New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$












  • $begingroup$
    For the person who raised the close vote, why? If you flag a question as "unclear what you're asking", consider asking the OP for explanations on what you don't undestand. This post seems very well written, especially for a first post.
    $endgroup$
    – IEatBagels
    2 hours ago










  • $begingroup$
    @IEatBagels - yes, OP was asked to clarify, and did so: look at the edit history to see what changed. I retracted my close vote after the edit, but it looks like the other person didn't revisit. No worry; it will expire, in time.
    $endgroup$
    – Toby Speight
    25 mins ago
















3












$begingroup$



Given a string S and a set of words D, find the longest word in D that is a subsequence of S.



Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.



Note: D can appear in any format (list, hash table, prefix tree, etc.)



For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".




  • The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".

  • The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.

  • The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.




I am currently trying to learn how to become a better programmer. I am taking the Google Tech Dev Guide recommended path and this is the first problem. I came up with my own solution, and read Google's first recommended answer is using brute force, and a slight optimization is to at least ensure the dictionary is represented by a hash table or prefix tree so that lookups are efficient.



This is my first time using a dictionary, and I also don't feel confident about using objects and classes. If you could review and recommend better ways to improve my code, maybe by using more (or better) objects, I would appreciate it.



class Main:
def create_dictionary(string):
dictionary = {}
index = 0

for letter in string:
if letter in dictionary:
dictionary[letter].append(index)
else:
dictionary[letter] = [index]
index += 1
return(dictionary)

def get_word_is_substring(word, dictionary):
index_of_last_letter_found = None
for letter in word:
if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
index = 0
while index < len(dictionary[letter]):
if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
index_of_last_letter_found = dictionary[letter][index]
break
else:
index += 1

else:
return False
return True

def replace_word_if_necessary(word, longest_word):
if (longest_word is None) or (len(word) > len(longest_word)):
longest_word = word
return longest_word

def get_longest_word(s, d):
dictionary = Main.create_dictionary(s)
longest_word = None

for word in d:
word_is_substring = Main.get_word_is_substring(word, dictionary)
if word_is_substring:
longest_word = Main.replace_word_if_necessary(word, longest_word)
print(longest_word)

Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})









share|improve this question









New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$












  • $begingroup$
    For the person who raised the close vote, why? If you flag a question as "unclear what you're asking", consider asking the OP for explanations on what you don't undestand. This post seems very well written, especially for a first post.
    $endgroup$
    – IEatBagels
    2 hours ago










  • $begingroup$
    @IEatBagels - yes, OP was asked to clarify, and did so: look at the edit history to see what changed. I retracted my close vote after the edit, but it looks like the other person didn't revisit. No worry; it will expire, in time.
    $endgroup$
    – Toby Speight
    25 mins ago














3












3








3





$begingroup$



Given a string S and a set of words D, find the longest word in D that is a subsequence of S.



Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.



Note: D can appear in any format (list, hash table, prefix tree, etc.)



For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".




  • The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".

  • The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.

  • The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.




I am currently trying to learn how to become a better programmer. I am taking the Google Tech Dev Guide recommended path and this is the first problem. I came up with my own solution, and read Google's first recommended answer is using brute force, and a slight optimization is to at least ensure the dictionary is represented by a hash table or prefix tree so that lookups are efficient.



This is my first time using a dictionary, and I also don't feel confident about using objects and classes. If you could review and recommend better ways to improve my code, maybe by using more (or better) objects, I would appreciate it.



class Main:
def create_dictionary(string):
dictionary = {}
index = 0

for letter in string:
if letter in dictionary:
dictionary[letter].append(index)
else:
dictionary[letter] = [index]
index += 1
return(dictionary)

def get_word_is_substring(word, dictionary):
index_of_last_letter_found = None
for letter in word:
if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
index = 0
while index < len(dictionary[letter]):
if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
index_of_last_letter_found = dictionary[letter][index]
break
else:
index += 1

else:
return False
return True

def replace_word_if_necessary(word, longest_word):
if (longest_word is None) or (len(word) > len(longest_word)):
longest_word = word
return longest_word

def get_longest_word(s, d):
dictionary = Main.create_dictionary(s)
longest_word = None

for word in d:
word_is_substring = Main.get_word_is_substring(word, dictionary)
if word_is_substring:
longest_word = Main.replace_word_if_necessary(word, longest_word)
print(longest_word)

Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})









share|improve this question









New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.







$endgroup$





Given a string S and a set of words D, find the longest word in D that is a subsequence of S.



Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.



Note: D can appear in any format (list, hash table, prefix tree, etc.)



For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".




  • The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".

  • The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.

  • The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.




I am currently trying to learn how to become a better programmer. I am taking the Google Tech Dev Guide recommended path and this is the first problem. I came up with my own solution, and read Google's first recommended answer is using brute force, and a slight optimization is to at least ensure the dictionary is represented by a hash table or prefix tree so that lookups are efficient.



This is my first time using a dictionary, and I also don't feel confident about using objects and classes. If you could review and recommend better ways to improve my code, maybe by using more (or better) objects, I would appreciate it.



class Main:
def create_dictionary(string):
dictionary = {}
index = 0

for letter in string:
if letter in dictionary:
dictionary[letter].append(index)
else:
dictionary[letter] = [index]
index += 1
return(dictionary)

def get_word_is_substring(word, dictionary):
index_of_last_letter_found = None
for letter in word:
if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
index = 0
while index < len(dictionary[letter]):
if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
index_of_last_letter_found = dictionary[letter][index]
break
else:
index += 1

else:
return False
return True

def replace_word_if_necessary(word, longest_word):
if (longest_word is None) or (len(word) > len(longest_word)):
longest_word = word
return longest_word

def get_longest_word(s, d):
dictionary = Main.create_dictionary(s)
longest_word = None

for word in d:
word_is_substring = Main.get_word_is_substring(word, dictionary)
if word_is_substring:
longest_word = Main.replace_word_if_necessary(word, longest_word)
print(longest_word)

Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})






python python-3.x






share|improve this question









New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.











share|improve this question









New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.









share|improve this question




share|improve this question








edited 4 hours ago







K. Kretz













New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.









asked 4 hours ago









K. KretzK. Kretz

162




162




New contributor




K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.





New contributor





K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.






K. Kretz is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.












  • $begingroup$
    For the person who raised the close vote, why? If you flag a question as "unclear what you're asking", consider asking the OP for explanations on what you don't undestand. This post seems very well written, especially for a first post.
    $endgroup$
    – IEatBagels
    2 hours ago










  • $begingroup$
    @IEatBagels - yes, OP was asked to clarify, and did so: look at the edit history to see what changed. I retracted my close vote after the edit, but it looks like the other person didn't revisit. No worry; it will expire, in time.
    $endgroup$
    – Toby Speight
    25 mins ago


















  • $begingroup$
    For the person who raised the close vote, why? If you flag a question as "unclear what you're asking", consider asking the OP for explanations on what you don't undestand. This post seems very well written, especially for a first post.
    $endgroup$
    – IEatBagels
    2 hours ago










  • $begingroup$
    @IEatBagels - yes, OP was asked to clarify, and did so: look at the edit history to see what changed. I retracted my close vote after the edit, but it looks like the other person didn't revisit. No worry; it will expire, in time.
    $endgroup$
    – Toby Speight
    25 mins ago
















$begingroup$
For the person who raised the close vote, why? If you flag a question as "unclear what you're asking", consider asking the OP for explanations on what you don't undestand. This post seems very well written, especially for a first post.
$endgroup$
– IEatBagels
2 hours ago




$begingroup$
For the person who raised the close vote, why? If you flag a question as "unclear what you're asking", consider asking the OP for explanations on what you don't undestand. This post seems very well written, especially for a first post.
$endgroup$
– IEatBagels
2 hours ago












$begingroup$
@IEatBagels - yes, OP was asked to clarify, and did so: look at the edit history to see what changed. I retracted my close vote after the edit, but it looks like the other person didn't revisit. No worry; it will expire, in time.
$endgroup$
– Toby Speight
25 mins ago




$begingroup$
@IEatBagels - yes, OP was asked to clarify, and did so: look at the edit history to see what changed. I retracted my close vote after the edit, but it looks like the other person didn't revisit. No worry; it will expire, in time.
$endgroup$
– Toby Speight
25 mins ago










2 Answers
2






active

oldest

votes


















2












$begingroup$

First, the good news: your code looks pretty good. It's indented properly, spelled properly, spaced properly, capitalized properly, and uses the right Python style. Good work!



Here are a couple of nits:





  1. This is a "program." (As opposed to just a module, or package, or library.) So use the standard python idiom for programs at the bottom:



    if __name__ == '__main__':
    Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})


    This mechanism is to allow your code to be loaded (via import myprogram) without having the main entry point automatically get invoked. That lets you load it up in the interpreter and browse around before you call it.




  2. Use docstrings! Write down the problem you are trying to solve, and any notes about inputs or outputs for functions. Especially data formats.



    For coding puzzle type problems, the docstring is a great place to copy the puzzle specification. This lets you refer back to it, lets you remember what you were doing when you wrote this code, and lets you paste it up at CodeReview with ease!



    #!/usr/bin/env python3
    """ Given a string S and a set of words D, find the longest word in D that is a subsequence of S.

    Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.

    Note: D can appear in any format (list, hash table, prefix tree, etc.)

    For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".

    - The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".
    - The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.
    - The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.
    """



Now here are some possible improvements:



In create_dictionary



def create_dictionary(string):
dictionary = {}
index = 0

for letter in string:
if letter in dictionary:
dictionary[letter].append(index)
else:
dictionary[letter] = [index]
index += 1
return(dictionary)


First, congratulations! You have written the hand-coded version of how to manage a dictionary whose values are lists. You got it right, your code works, and you should never do that again because it's boring. Instead, use collections.defaultdict(list). A defaultdict remembers a factory function that it will call whenever it is asked about a key but doesn't have a corresponding value.



The word list is not just the name of a Python data type, it's also the function to call when you want to construct one! (Just like you use the name of every class as it's constructor: my_obj = MyClass(1, 2, "hello")) So when you want a dictionary that looks up lists, it's much easier to assume that there will always be a list in the dictionary, but it might be empty. That's what defaultdict(list) will get you:



import collections  # somewhere at top of file

def create_dictionary(s):
dictionary = collections.defaultdict(list)
index = 0

for letter in string:
dictionary[letter].append(index) # defaultdict FTW!
index += 1
return(dictionary)


Next, give up your love affair with integers! Ned Batchelder gave a nice talk on the subject, so here's a link to that: https://www.youtube.com/watch?v=EnSu9hHGq5o



The idea is that many (most?) python loops don't need to use integers to index over things. And when you do need an integer index (and in this case you do!) there are better ways than maintaining your own integer index variable. Here's one such way: the enumerate() built-in function.



With enumerate, you can write a loop that iterates over the values from some iterable along with an automatically-associated integer:



    # No index=0 here
for index, letter in enumerate(string):
dictionary[letter].append(index)
# No index+=1 here!


The index, string pair is called a tuple, and it is a built-in type just like list. The act of assigning or iterating using multiple target variables that take their values from a single tuple is called tuple unpacking. (Remember that phrase, you'll need it when you want to ask for help on the subject.)



In get_longest_word



The issue I have with this function is not one of Python, but rather of design. You are given some words, d. You want to find the longest word that is a subsequence of the string s. How do you do it?



In your case, the answer is "look at every single word in d, ignore the ones that are not subsequences of s, pick the longest one that remains."



There are a couple of better (read: faster, more efficient) ways of doing that job. Let me make one simple suggestion: sort the words!



In Python, there are Iterables and Sequences. An iterable is something you can iterate. A sequence is something you can access using s[i]. It's possible to have an infinite iterable, by writing an infinite generator function. It's not possible to have an infinite sequence, since you'll run out of memory trying to store it.



In this particular case it seems okay to assume that d is going to be a sequence: a finite list or tuple. So the fastest way to find the "longest word" is to start by looking at long words first. Because once you find a long word that is a subsequence, you can stop - there are no shorter ones!



The way to sort things in Python is the sorted built-in function. (Yes, it takes an iterable. No, it won't sort an infinite one. Yeesh!) By default, it sorts things by comparing the items using the "native" comparison. You can specify a key function, however, to sort using some different mechanism. Let's use the length of the words, which is the len function
(function len(x) -> int returns the length of the list/string/whatever). And let's reverse the order, so that the big ones come first:



d = sorted(d, key=len, reverse=True)


Now, instead of needing to check the length and update the longest-word-so-far variable, you can just return immediately once you find a subsequence.



In get_word_is_substring



Let's talk about default values. You say:



index_of_last_letter_found = None


But :! grep letter_found % in Vim tells me:



    index_of_last_letter_found = None
if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
index_of_last_letter_found = dictionary[letter][index]


You're spending a lot of keystrokes checking for None. And all you do is compare using <, and assign new values. Why not just set the default value to some value that you know is going to be "too low"? Since string index values start at 0, maybe -1 would make sense:



index_of_last_letter_found = -1


While were at it, shorten that variable name. Names should be as long as they need to be, and no longer!



def get_word_is_substring(word, dictionary):
last_index = -1 # Index of last letter found

for letter in word:
if letter in dictionary and dictionary[letter][-1] > last_index:
index = 0
while index < len(dictionary[letter]):
if last_index < dictionary[letter][index]:
last_index = dictionary[letter][index]
break
else:
index += 1
else:
return False
return True


That's a lot more readable, since there are fewer tests and fewer characters.



Next, let's go back and work on your fetish for simple integer arithmetic. Now that you've watched Ned Batchelder's talk, you can see that index += 1 is not the way!



for index in dictionary[letter]:
if last_index < index:
last_index = index
break


(There are some other ways to find the first element in an iterable that matches a condition. See here for many of them. But this is nice and clear, and works.)



Some words on "style"



The above code may be somewhat more difficult to understand because I changed the meaning of index. In your original code, index was an index into the list of indices that pointed to where letters occur in the master sequence. In my updated version index means the actual indices from the list. I have removed one level of indirection.



This is an example of why "too short" variable names are actually a good thing. You'll find a lot of people use very small names, like ch or i, to represent loop variables. This is because most of the time, loop variables are not a "concept" or a "noun". Instead, they are a subscript. We write a[i] because the original Teletype devices hand-carved out of wood by our pioneer forbears would not allow writing 𝓐ᵢ.



But we don't really care about i, or about a[i]. We care about the thing stored there, and its properties. So, I encourage you to use short little 1-letter and 2-letter names when you are indexing a sequence, or when you're stuck in some other language that doesn't provide the nice looping options that Python does. Then you could say:



for i ...
index = s_dict[letter][i]


And there would be no confusion between index and that-loop-variable.



Next, the structuring of your escape clauses: I see a lot of people being taught to code in a way that puts exit conditions last. You're doing that in your loop:



    for letter in word:
if letter in dictionary and dictionary[letter][-1] > last_index:
# much code here
else:
return False


But moving the exit condition to the top (by reversing the sense of the if statement) makes it clear what the alternatives are (if this is true, leave, otherwise ...) and reduces the indentation level of your code (by removing the else indent). Some people would have you use two-space tabs to address the indentation problem. Those people are all young with good eyesight. They're also quite wrong. :-)



    for letter in word:
if letter not in dictionary or dictionary[letter][-1] <= last_index:
return False

# much code here


Lastly, there's the name. You called it get_word_is_substring, but we want subsequences not substring, and there's no reason to say get_ at the beginning. In fact, the word can be assumed, since you're passing it in, so:



def is_subsequence(word, s_dict):
last_index = -1 # Index (in s) of last letter found

for letter in word:
if letter not in s_dict or s_dict[letter][-1] <= last_index:
return False

for index in s_dict[letter]:
if last_index < index:
last_index = index
break

return True


(Note: Because I suggested s_dict be a defaultdict, you could eliminate the if test entirely and rely on the for ... else construct, which is rather obscure. I don't recommend it, simply because it's non-intuitive, hard to understand, and because spelling out the failure conditions is better than leaving them to the language. ("explicit is better than implicit"))



The class



Finally, let's talk about the class:



class Main:


But first! Here's another video for you to watch, Jack Diederich's talk at Pycon 2012 entitled: "Stop Writing Classes!": https://www.youtube.com/watch?v=o9pEzgHorH0



As Jack says, "The signature of 'this shouldn't be a class' is that it has two methods, one of which is __init__."



In this case, you've created a class, but you don't store any data and you only call one method from the outside. It should be a function.






share|improve this answer











$endgroup$





















    1












    $begingroup$

    Use defaultdict and enumerate



    Python offers the defaultdict class, which works like a regular dictonary but returns a default value (here: empty list) if the requested element is not present. Also use enumerate if you want an index in a loop.



    from collections import defaultdict

    def create_dictionary(string):
    dictionary = defaultdict(list)
    for index, letter in enumerate(string):
    dictionary[letter].append(index)
    return dictionary


    In the next method, work with lists of indices to leverage the defaultdict. Also, use a list comprehension to filter for valid indices because that also defaults to an empty list, so you don't need any special case handling.



    def get_word_is_substring(word, dictionary):
    indices = [-1]
    for letter in word:
    indices = [index for index in dictionary[letter] if index > indices[0]]
    if not indices:
    return False
    return True





    share|improve this answer









    $endgroup$













      Your Answer





      StackExchange.ifUsing("editor", function () {
      return StackExchange.using("mathjaxEditing", function () {
      StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
      StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
      });
      });
      }, "mathjax-editing");

      StackExchange.ifUsing("editor", function () {
      StackExchange.using("externalEditor", function () {
      StackExchange.using("snippets", function () {
      StackExchange.snippets.init();
      });
      });
      }, "code-snippets");

      StackExchange.ready(function() {
      var channelOptions = {
      tags: "".split(" "),
      id: "196"
      };
      initTagRenderer("".split(" "), "".split(" "), channelOptions);

      StackExchange.using("externalEditor", function() {
      // Have to fire editor after snippets, if snippets enabled
      if (StackExchange.settings.snippets.snippetsEnabled) {
      StackExchange.using("snippets", function() {
      createEditor();
      });
      }
      else {
      createEditor();
      }
      });

      function createEditor() {
      StackExchange.prepareEditor({
      heartbeatType: 'answer',
      autoActivateHeartbeat: false,
      convertImagesToLinks: false,
      noModals: true,
      showLowRepImageUploadWarning: true,
      reputationToPostImages: null,
      bindNavPrevention: true,
      postfix: "",
      imageUploader: {
      brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
      contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
      allowUrls: true
      },
      onDemand: true,
      discardSelector: ".discard-answer"
      ,immediatelyShowMarkdownHelp:true
      });


      }
      });






      K. Kretz is a new contributor. Be nice, and check out our Code of Conduct.










      draft saved

      draft discarded


















      StackExchange.ready(
      function () {
      StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f214408%2ffind-the-longest-word-in-set-d-that-is-a-subsequence-of-string-s%23new-answer', 'question_page');
      }
      );

      Post as a guest















      Required, but never shown

























      2 Answers
      2






      active

      oldest

      votes








      2 Answers
      2






      active

      oldest

      votes









      active

      oldest

      votes






      active

      oldest

      votes









      2












      $begingroup$

      First, the good news: your code looks pretty good. It's indented properly, spelled properly, spaced properly, capitalized properly, and uses the right Python style. Good work!



      Here are a couple of nits:





      1. This is a "program." (As opposed to just a module, or package, or library.) So use the standard python idiom for programs at the bottom:



        if __name__ == '__main__':
        Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})


        This mechanism is to allow your code to be loaded (via import myprogram) without having the main entry point automatically get invoked. That lets you load it up in the interpreter and browse around before you call it.




      2. Use docstrings! Write down the problem you are trying to solve, and any notes about inputs or outputs for functions. Especially data formats.



        For coding puzzle type problems, the docstring is a great place to copy the puzzle specification. This lets you refer back to it, lets you remember what you were doing when you wrote this code, and lets you paste it up at CodeReview with ease!



        #!/usr/bin/env python3
        """ Given a string S and a set of words D, find the longest word in D that is a subsequence of S.

        Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.

        Note: D can appear in any format (list, hash table, prefix tree, etc.)

        For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".

        - The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".
        - The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.
        - The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.
        """



      Now here are some possible improvements:



      In create_dictionary



      def create_dictionary(string):
      dictionary = {}
      index = 0

      for letter in string:
      if letter in dictionary:
      dictionary[letter].append(index)
      else:
      dictionary[letter] = [index]
      index += 1
      return(dictionary)


      First, congratulations! You have written the hand-coded version of how to manage a dictionary whose values are lists. You got it right, your code works, and you should never do that again because it's boring. Instead, use collections.defaultdict(list). A defaultdict remembers a factory function that it will call whenever it is asked about a key but doesn't have a corresponding value.



      The word list is not just the name of a Python data type, it's also the function to call when you want to construct one! (Just like you use the name of every class as it's constructor: my_obj = MyClass(1, 2, "hello")) So when you want a dictionary that looks up lists, it's much easier to assume that there will always be a list in the dictionary, but it might be empty. That's what defaultdict(list) will get you:



      import collections  # somewhere at top of file

      def create_dictionary(s):
      dictionary = collections.defaultdict(list)
      index = 0

      for letter in string:
      dictionary[letter].append(index) # defaultdict FTW!
      index += 1
      return(dictionary)


      Next, give up your love affair with integers! Ned Batchelder gave a nice talk on the subject, so here's a link to that: https://www.youtube.com/watch?v=EnSu9hHGq5o



      The idea is that many (most?) python loops don't need to use integers to index over things. And when you do need an integer index (and in this case you do!) there are better ways than maintaining your own integer index variable. Here's one such way: the enumerate() built-in function.



      With enumerate, you can write a loop that iterates over the values from some iterable along with an automatically-associated integer:



          # No index=0 here
      for index, letter in enumerate(string):
      dictionary[letter].append(index)
      # No index+=1 here!


      The index, string pair is called a tuple, and it is a built-in type just like list. The act of assigning or iterating using multiple target variables that take their values from a single tuple is called tuple unpacking. (Remember that phrase, you'll need it when you want to ask for help on the subject.)



      In get_longest_word



      The issue I have with this function is not one of Python, but rather of design. You are given some words, d. You want to find the longest word that is a subsequence of the string s. How do you do it?



      In your case, the answer is "look at every single word in d, ignore the ones that are not subsequences of s, pick the longest one that remains."



      There are a couple of better (read: faster, more efficient) ways of doing that job. Let me make one simple suggestion: sort the words!



      In Python, there are Iterables and Sequences. An iterable is something you can iterate. A sequence is something you can access using s[i]. It's possible to have an infinite iterable, by writing an infinite generator function. It's not possible to have an infinite sequence, since you'll run out of memory trying to store it.



      In this particular case it seems okay to assume that d is going to be a sequence: a finite list or tuple. So the fastest way to find the "longest word" is to start by looking at long words first. Because once you find a long word that is a subsequence, you can stop - there are no shorter ones!



      The way to sort things in Python is the sorted built-in function. (Yes, it takes an iterable. No, it won't sort an infinite one. Yeesh!) By default, it sorts things by comparing the items using the "native" comparison. You can specify a key function, however, to sort using some different mechanism. Let's use the length of the words, which is the len function
      (function len(x) -> int returns the length of the list/string/whatever). And let's reverse the order, so that the big ones come first:



      d = sorted(d, key=len, reverse=True)


      Now, instead of needing to check the length and update the longest-word-so-far variable, you can just return immediately once you find a subsequence.



      In get_word_is_substring



      Let's talk about default values. You say:



      index_of_last_letter_found = None


      But :! grep letter_found % in Vim tells me:



          index_of_last_letter_found = None
      if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
      if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
      index_of_last_letter_found = dictionary[letter][index]


      You're spending a lot of keystrokes checking for None. And all you do is compare using <, and assign new values. Why not just set the default value to some value that you know is going to be "too low"? Since string index values start at 0, maybe -1 would make sense:



      index_of_last_letter_found = -1


      While were at it, shorten that variable name. Names should be as long as they need to be, and no longer!



      def get_word_is_substring(word, dictionary):
      last_index = -1 # Index of last letter found

      for letter in word:
      if letter in dictionary and dictionary[letter][-1] > last_index:
      index = 0
      while index < len(dictionary[letter]):
      if last_index < dictionary[letter][index]:
      last_index = dictionary[letter][index]
      break
      else:
      index += 1
      else:
      return False
      return True


      That's a lot more readable, since there are fewer tests and fewer characters.



      Next, let's go back and work on your fetish for simple integer arithmetic. Now that you've watched Ned Batchelder's talk, you can see that index += 1 is not the way!



      for index in dictionary[letter]:
      if last_index < index:
      last_index = index
      break


      (There are some other ways to find the first element in an iterable that matches a condition. See here for many of them. But this is nice and clear, and works.)



      Some words on "style"



      The above code may be somewhat more difficult to understand because I changed the meaning of index. In your original code, index was an index into the list of indices that pointed to where letters occur in the master sequence. In my updated version index means the actual indices from the list. I have removed one level of indirection.



      This is an example of why "too short" variable names are actually a good thing. You'll find a lot of people use very small names, like ch or i, to represent loop variables. This is because most of the time, loop variables are not a "concept" or a "noun". Instead, they are a subscript. We write a[i] because the original Teletype devices hand-carved out of wood by our pioneer forbears would not allow writing 𝓐ᵢ.



      But we don't really care about i, or about a[i]. We care about the thing stored there, and its properties. So, I encourage you to use short little 1-letter and 2-letter names when you are indexing a sequence, or when you're stuck in some other language that doesn't provide the nice looping options that Python does. Then you could say:



      for i ...
      index = s_dict[letter][i]


      And there would be no confusion between index and that-loop-variable.



      Next, the structuring of your escape clauses: I see a lot of people being taught to code in a way that puts exit conditions last. You're doing that in your loop:



          for letter in word:
      if letter in dictionary and dictionary[letter][-1] > last_index:
      # much code here
      else:
      return False


      But moving the exit condition to the top (by reversing the sense of the if statement) makes it clear what the alternatives are (if this is true, leave, otherwise ...) and reduces the indentation level of your code (by removing the else indent). Some people would have you use two-space tabs to address the indentation problem. Those people are all young with good eyesight. They're also quite wrong. :-)



          for letter in word:
      if letter not in dictionary or dictionary[letter][-1] <= last_index:
      return False

      # much code here


      Lastly, there's the name. You called it get_word_is_substring, but we want subsequences not substring, and there's no reason to say get_ at the beginning. In fact, the word can be assumed, since you're passing it in, so:



      def is_subsequence(word, s_dict):
      last_index = -1 # Index (in s) of last letter found

      for letter in word:
      if letter not in s_dict or s_dict[letter][-1] <= last_index:
      return False

      for index in s_dict[letter]:
      if last_index < index:
      last_index = index
      break

      return True


      (Note: Because I suggested s_dict be a defaultdict, you could eliminate the if test entirely and rely on the for ... else construct, which is rather obscure. I don't recommend it, simply because it's non-intuitive, hard to understand, and because spelling out the failure conditions is better than leaving them to the language. ("explicit is better than implicit"))



      The class



      Finally, let's talk about the class:



      class Main:


      But first! Here's another video for you to watch, Jack Diederich's talk at Pycon 2012 entitled: "Stop Writing Classes!": https://www.youtube.com/watch?v=o9pEzgHorH0



      As Jack says, "The signature of 'this shouldn't be a class' is that it has two methods, one of which is __init__."



      In this case, you've created a class, but you don't store any data and you only call one method from the outside. It should be a function.






      share|improve this answer











      $endgroup$


















        2












        $begingroup$

        First, the good news: your code looks pretty good. It's indented properly, spelled properly, spaced properly, capitalized properly, and uses the right Python style. Good work!



        Here are a couple of nits:





        1. This is a "program." (As opposed to just a module, or package, or library.) So use the standard python idiom for programs at the bottom:



          if __name__ == '__main__':
          Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})


          This mechanism is to allow your code to be loaded (via import myprogram) without having the main entry point automatically get invoked. That lets you load it up in the interpreter and browse around before you call it.




        2. Use docstrings! Write down the problem you are trying to solve, and any notes about inputs or outputs for functions. Especially data formats.



          For coding puzzle type problems, the docstring is a great place to copy the puzzle specification. This lets you refer back to it, lets you remember what you were doing when you wrote this code, and lets you paste it up at CodeReview with ease!



          #!/usr/bin/env python3
          """ Given a string S and a set of words D, find the longest word in D that is a subsequence of S.

          Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.

          Note: D can appear in any format (list, hash table, prefix tree, etc.)

          For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".

          - The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".
          - The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.
          - The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.
          """



        Now here are some possible improvements:



        In create_dictionary



        def create_dictionary(string):
        dictionary = {}
        index = 0

        for letter in string:
        if letter in dictionary:
        dictionary[letter].append(index)
        else:
        dictionary[letter] = [index]
        index += 1
        return(dictionary)


        First, congratulations! You have written the hand-coded version of how to manage a dictionary whose values are lists. You got it right, your code works, and you should never do that again because it's boring. Instead, use collections.defaultdict(list). A defaultdict remembers a factory function that it will call whenever it is asked about a key but doesn't have a corresponding value.



        The word list is not just the name of a Python data type, it's also the function to call when you want to construct one! (Just like you use the name of every class as it's constructor: my_obj = MyClass(1, 2, "hello")) So when you want a dictionary that looks up lists, it's much easier to assume that there will always be a list in the dictionary, but it might be empty. That's what defaultdict(list) will get you:



        import collections  # somewhere at top of file

        def create_dictionary(s):
        dictionary = collections.defaultdict(list)
        index = 0

        for letter in string:
        dictionary[letter].append(index) # defaultdict FTW!
        index += 1
        return(dictionary)


        Next, give up your love affair with integers! Ned Batchelder gave a nice talk on the subject, so here's a link to that: https://www.youtube.com/watch?v=EnSu9hHGq5o



        The idea is that many (most?) python loops don't need to use integers to index over things. And when you do need an integer index (and in this case you do!) there are better ways than maintaining your own integer index variable. Here's one such way: the enumerate() built-in function.



        With enumerate, you can write a loop that iterates over the values from some iterable along with an automatically-associated integer:



            # No index=0 here
        for index, letter in enumerate(string):
        dictionary[letter].append(index)
        # No index+=1 here!


        The index, string pair is called a tuple, and it is a built-in type just like list. The act of assigning or iterating using multiple target variables that take their values from a single tuple is called tuple unpacking. (Remember that phrase, you'll need it when you want to ask for help on the subject.)



        In get_longest_word



        The issue I have with this function is not one of Python, but rather of design. You are given some words, d. You want to find the longest word that is a subsequence of the string s. How do you do it?



        In your case, the answer is "look at every single word in d, ignore the ones that are not subsequences of s, pick the longest one that remains."



        There are a couple of better (read: faster, more efficient) ways of doing that job. Let me make one simple suggestion: sort the words!



        In Python, there are Iterables and Sequences. An iterable is something you can iterate. A sequence is something you can access using s[i]. It's possible to have an infinite iterable, by writing an infinite generator function. It's not possible to have an infinite sequence, since you'll run out of memory trying to store it.



        In this particular case it seems okay to assume that d is going to be a sequence: a finite list or tuple. So the fastest way to find the "longest word" is to start by looking at long words first. Because once you find a long word that is a subsequence, you can stop - there are no shorter ones!



        The way to sort things in Python is the sorted built-in function. (Yes, it takes an iterable. No, it won't sort an infinite one. Yeesh!) By default, it sorts things by comparing the items using the "native" comparison. You can specify a key function, however, to sort using some different mechanism. Let's use the length of the words, which is the len function
        (function len(x) -> int returns the length of the list/string/whatever). And let's reverse the order, so that the big ones come first:



        d = sorted(d, key=len, reverse=True)


        Now, instead of needing to check the length and update the longest-word-so-far variable, you can just return immediately once you find a subsequence.



        In get_word_is_substring



        Let's talk about default values. You say:



        index_of_last_letter_found = None


        But :! grep letter_found % in Vim tells me:



            index_of_last_letter_found = None
        if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
        if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
        index_of_last_letter_found = dictionary[letter][index]


        You're spending a lot of keystrokes checking for None. And all you do is compare using <, and assign new values. Why not just set the default value to some value that you know is going to be "too low"? Since string index values start at 0, maybe -1 would make sense:



        index_of_last_letter_found = -1


        While were at it, shorten that variable name. Names should be as long as they need to be, and no longer!



        def get_word_is_substring(word, dictionary):
        last_index = -1 # Index of last letter found

        for letter in word:
        if letter in dictionary and dictionary[letter][-1] > last_index:
        index = 0
        while index < len(dictionary[letter]):
        if last_index < dictionary[letter][index]:
        last_index = dictionary[letter][index]
        break
        else:
        index += 1
        else:
        return False
        return True


        That's a lot more readable, since there are fewer tests and fewer characters.



        Next, let's go back and work on your fetish for simple integer arithmetic. Now that you've watched Ned Batchelder's talk, you can see that index += 1 is not the way!



        for index in dictionary[letter]:
        if last_index < index:
        last_index = index
        break


        (There are some other ways to find the first element in an iterable that matches a condition. See here for many of them. But this is nice and clear, and works.)



        Some words on "style"



        The above code may be somewhat more difficult to understand because I changed the meaning of index. In your original code, index was an index into the list of indices that pointed to where letters occur in the master sequence. In my updated version index means the actual indices from the list. I have removed one level of indirection.



        This is an example of why "too short" variable names are actually a good thing. You'll find a lot of people use very small names, like ch or i, to represent loop variables. This is because most of the time, loop variables are not a "concept" or a "noun". Instead, they are a subscript. We write a[i] because the original Teletype devices hand-carved out of wood by our pioneer forbears would not allow writing 𝓐ᵢ.



        But we don't really care about i, or about a[i]. We care about the thing stored there, and its properties. So, I encourage you to use short little 1-letter and 2-letter names when you are indexing a sequence, or when you're stuck in some other language that doesn't provide the nice looping options that Python does. Then you could say:



        for i ...
        index = s_dict[letter][i]


        And there would be no confusion between index and that-loop-variable.



        Next, the structuring of your escape clauses: I see a lot of people being taught to code in a way that puts exit conditions last. You're doing that in your loop:



            for letter in word:
        if letter in dictionary and dictionary[letter][-1] > last_index:
        # much code here
        else:
        return False


        But moving the exit condition to the top (by reversing the sense of the if statement) makes it clear what the alternatives are (if this is true, leave, otherwise ...) and reduces the indentation level of your code (by removing the else indent). Some people would have you use two-space tabs to address the indentation problem. Those people are all young with good eyesight. They're also quite wrong. :-)



            for letter in word:
        if letter not in dictionary or dictionary[letter][-1] <= last_index:
        return False

        # much code here


        Lastly, there's the name. You called it get_word_is_substring, but we want subsequences not substring, and there's no reason to say get_ at the beginning. In fact, the word can be assumed, since you're passing it in, so:



        def is_subsequence(word, s_dict):
        last_index = -1 # Index (in s) of last letter found

        for letter in word:
        if letter not in s_dict or s_dict[letter][-1] <= last_index:
        return False

        for index in s_dict[letter]:
        if last_index < index:
        last_index = index
        break

        return True


        (Note: Because I suggested s_dict be a defaultdict, you could eliminate the if test entirely and rely on the for ... else construct, which is rather obscure. I don't recommend it, simply because it's non-intuitive, hard to understand, and because spelling out the failure conditions is better than leaving them to the language. ("explicit is better than implicit"))



        The class



        Finally, let's talk about the class:



        class Main:


        But first! Here's another video for you to watch, Jack Diederich's talk at Pycon 2012 entitled: "Stop Writing Classes!": https://www.youtube.com/watch?v=o9pEzgHorH0



        As Jack says, "The signature of 'this shouldn't be a class' is that it has two methods, one of which is __init__."



        In this case, you've created a class, but you don't store any data and you only call one method from the outside. It should be a function.






        share|improve this answer











        $endgroup$
















          2












          2








          2





          $begingroup$

          First, the good news: your code looks pretty good. It's indented properly, spelled properly, spaced properly, capitalized properly, and uses the right Python style. Good work!



          Here are a couple of nits:





          1. This is a "program." (As opposed to just a module, or package, or library.) So use the standard python idiom for programs at the bottom:



            if __name__ == '__main__':
            Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})


            This mechanism is to allow your code to be loaded (via import myprogram) without having the main entry point automatically get invoked. That lets you load it up in the interpreter and browse around before you call it.




          2. Use docstrings! Write down the problem you are trying to solve, and any notes about inputs or outputs for functions. Especially data formats.



            For coding puzzle type problems, the docstring is a great place to copy the puzzle specification. This lets you refer back to it, lets you remember what you were doing when you wrote this code, and lets you paste it up at CodeReview with ease!



            #!/usr/bin/env python3
            """ Given a string S and a set of words D, find the longest word in D that is a subsequence of S.

            Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.

            Note: D can appear in any format (list, hash table, prefix tree, etc.)

            For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".

            - The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".
            - The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.
            - The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.
            """



          Now here are some possible improvements:



          In create_dictionary



          def create_dictionary(string):
          dictionary = {}
          index = 0

          for letter in string:
          if letter in dictionary:
          dictionary[letter].append(index)
          else:
          dictionary[letter] = [index]
          index += 1
          return(dictionary)


          First, congratulations! You have written the hand-coded version of how to manage a dictionary whose values are lists. You got it right, your code works, and you should never do that again because it's boring. Instead, use collections.defaultdict(list). A defaultdict remembers a factory function that it will call whenever it is asked about a key but doesn't have a corresponding value.



          The word list is not just the name of a Python data type, it's also the function to call when you want to construct one! (Just like you use the name of every class as it's constructor: my_obj = MyClass(1, 2, "hello")) So when you want a dictionary that looks up lists, it's much easier to assume that there will always be a list in the dictionary, but it might be empty. That's what defaultdict(list) will get you:



          import collections  # somewhere at top of file

          def create_dictionary(s):
          dictionary = collections.defaultdict(list)
          index = 0

          for letter in string:
          dictionary[letter].append(index) # defaultdict FTW!
          index += 1
          return(dictionary)


          Next, give up your love affair with integers! Ned Batchelder gave a nice talk on the subject, so here's a link to that: https://www.youtube.com/watch?v=EnSu9hHGq5o



          The idea is that many (most?) python loops don't need to use integers to index over things. And when you do need an integer index (and in this case you do!) there are better ways than maintaining your own integer index variable. Here's one such way: the enumerate() built-in function.



          With enumerate, you can write a loop that iterates over the values from some iterable along with an automatically-associated integer:



              # No index=0 here
          for index, letter in enumerate(string):
          dictionary[letter].append(index)
          # No index+=1 here!


          The index, string pair is called a tuple, and it is a built-in type just like list. The act of assigning or iterating using multiple target variables that take their values from a single tuple is called tuple unpacking. (Remember that phrase, you'll need it when you want to ask for help on the subject.)



          In get_longest_word



          The issue I have with this function is not one of Python, but rather of design. You are given some words, d. You want to find the longest word that is a subsequence of the string s. How do you do it?



          In your case, the answer is "look at every single word in d, ignore the ones that are not subsequences of s, pick the longest one that remains."



          There are a couple of better (read: faster, more efficient) ways of doing that job. Let me make one simple suggestion: sort the words!



          In Python, there are Iterables and Sequences. An iterable is something you can iterate. A sequence is something you can access using s[i]. It's possible to have an infinite iterable, by writing an infinite generator function. It's not possible to have an infinite sequence, since you'll run out of memory trying to store it.



          In this particular case it seems okay to assume that d is going to be a sequence: a finite list or tuple. So the fastest way to find the "longest word" is to start by looking at long words first. Because once you find a long word that is a subsequence, you can stop - there are no shorter ones!



          The way to sort things in Python is the sorted built-in function. (Yes, it takes an iterable. No, it won't sort an infinite one. Yeesh!) By default, it sorts things by comparing the items using the "native" comparison. You can specify a key function, however, to sort using some different mechanism. Let's use the length of the words, which is the len function
          (function len(x) -> int returns the length of the list/string/whatever). And let's reverse the order, so that the big ones come first:



          d = sorted(d, key=len, reverse=True)


          Now, instead of needing to check the length and update the longest-word-so-far variable, you can just return immediately once you find a subsequence.



          In get_word_is_substring



          Let's talk about default values. You say:



          index_of_last_letter_found = None


          But :! grep letter_found % in Vim tells me:



              index_of_last_letter_found = None
          if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
          if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
          index_of_last_letter_found = dictionary[letter][index]


          You're spending a lot of keystrokes checking for None. And all you do is compare using <, and assign new values. Why not just set the default value to some value that you know is going to be "too low"? Since string index values start at 0, maybe -1 would make sense:



          index_of_last_letter_found = -1


          While were at it, shorten that variable name. Names should be as long as they need to be, and no longer!



          def get_word_is_substring(word, dictionary):
          last_index = -1 # Index of last letter found

          for letter in word:
          if letter in dictionary and dictionary[letter][-1] > last_index:
          index = 0
          while index < len(dictionary[letter]):
          if last_index < dictionary[letter][index]:
          last_index = dictionary[letter][index]
          break
          else:
          index += 1
          else:
          return False
          return True


          That's a lot more readable, since there are fewer tests and fewer characters.



          Next, let's go back and work on your fetish for simple integer arithmetic. Now that you've watched Ned Batchelder's talk, you can see that index += 1 is not the way!



          for index in dictionary[letter]:
          if last_index < index:
          last_index = index
          break


          (There are some other ways to find the first element in an iterable that matches a condition. See here for many of them. But this is nice and clear, and works.)



          Some words on "style"



          The above code may be somewhat more difficult to understand because I changed the meaning of index. In your original code, index was an index into the list of indices that pointed to where letters occur in the master sequence. In my updated version index means the actual indices from the list. I have removed one level of indirection.



          This is an example of why "too short" variable names are actually a good thing. You'll find a lot of people use very small names, like ch or i, to represent loop variables. This is because most of the time, loop variables are not a "concept" or a "noun". Instead, they are a subscript. We write a[i] because the original Teletype devices hand-carved out of wood by our pioneer forbears would not allow writing 𝓐ᵢ.



          But we don't really care about i, or about a[i]. We care about the thing stored there, and its properties. So, I encourage you to use short little 1-letter and 2-letter names when you are indexing a sequence, or when you're stuck in some other language that doesn't provide the nice looping options that Python does. Then you could say:



          for i ...
          index = s_dict[letter][i]


          And there would be no confusion between index and that-loop-variable.



          Next, the structuring of your escape clauses: I see a lot of people being taught to code in a way that puts exit conditions last. You're doing that in your loop:



              for letter in word:
          if letter in dictionary and dictionary[letter][-1] > last_index:
          # much code here
          else:
          return False


          But moving the exit condition to the top (by reversing the sense of the if statement) makes it clear what the alternatives are (if this is true, leave, otherwise ...) and reduces the indentation level of your code (by removing the else indent). Some people would have you use two-space tabs to address the indentation problem. Those people are all young with good eyesight. They're also quite wrong. :-)



              for letter in word:
          if letter not in dictionary or dictionary[letter][-1] <= last_index:
          return False

          # much code here


          Lastly, there's the name. You called it get_word_is_substring, but we want subsequences not substring, and there's no reason to say get_ at the beginning. In fact, the word can be assumed, since you're passing it in, so:



          def is_subsequence(word, s_dict):
          last_index = -1 # Index (in s) of last letter found

          for letter in word:
          if letter not in s_dict or s_dict[letter][-1] <= last_index:
          return False

          for index in s_dict[letter]:
          if last_index < index:
          last_index = index
          break

          return True


          (Note: Because I suggested s_dict be a defaultdict, you could eliminate the if test entirely and rely on the for ... else construct, which is rather obscure. I don't recommend it, simply because it's non-intuitive, hard to understand, and because spelling out the failure conditions is better than leaving them to the language. ("explicit is better than implicit"))



          The class



          Finally, let's talk about the class:



          class Main:


          But first! Here's another video for you to watch, Jack Diederich's talk at Pycon 2012 entitled: "Stop Writing Classes!": https://www.youtube.com/watch?v=o9pEzgHorH0



          As Jack says, "The signature of 'this shouldn't be a class' is that it has two methods, one of which is __init__."



          In this case, you've created a class, but you don't store any data and you only call one method from the outside. It should be a function.






          share|improve this answer











          $endgroup$



          First, the good news: your code looks pretty good. It's indented properly, spelled properly, spaced properly, capitalized properly, and uses the right Python style. Good work!



          Here are a couple of nits:





          1. This is a "program." (As opposed to just a module, or package, or library.) So use the standard python idiom for programs at the bottom:



            if __name__ == '__main__':
            Main.get_longest_word("abppplee", {"ale", "bale", "able", "apple", "kangaroo"})


            This mechanism is to allow your code to be loaded (via import myprogram) without having the main entry point automatically get invoked. That lets you load it up in the interpreter and browse around before you call it.




          2. Use docstrings! Write down the problem you are trying to solve, and any notes about inputs or outputs for functions. Especially data formats.



            For coding puzzle type problems, the docstring is a great place to copy the puzzle specification. This lets you refer back to it, lets you remember what you were doing when you wrote this code, and lets you paste it up at CodeReview with ease!



            #!/usr/bin/env python3
            """ Given a string S and a set of words D, find the longest word in D that is a subsequence of S.

            Word W is a subsequence of S if some number of characters, possibly zero, can be deleted from S to form W, without reordering the remaining characters.

            Note: D can appear in any format (list, hash table, prefix tree, etc.)

            For example, given the input of S = "abppplee" and D = {"able", "ale", "apple", "bale", "kangaroo"} the correct output would be "apple".

            - The words "able" and "ale" are both subsequences of S, but they are shorter than "apple".
            - The word "bale" is not a subsequence of S because even though S has all the right letters, they are not in the right order.
            - The word "kangaroo" is the longest word in D, but it isn't a subsequence of S.
            """



          Now here are some possible improvements:



          In create_dictionary



          def create_dictionary(string):
          dictionary = {}
          index = 0

          for letter in string:
          if letter in dictionary:
          dictionary[letter].append(index)
          else:
          dictionary[letter] = [index]
          index += 1
          return(dictionary)


          First, congratulations! You have written the hand-coded version of how to manage a dictionary whose values are lists. You got it right, your code works, and you should never do that again because it's boring. Instead, use collections.defaultdict(list). A defaultdict remembers a factory function that it will call whenever it is asked about a key but doesn't have a corresponding value.



          The word list is not just the name of a Python data type, it's also the function to call when you want to construct one! (Just like you use the name of every class as it's constructor: my_obj = MyClass(1, 2, "hello")) So when you want a dictionary that looks up lists, it's much easier to assume that there will always be a list in the dictionary, but it might be empty. That's what defaultdict(list) will get you:



          import collections  # somewhere at top of file

          def create_dictionary(s):
          dictionary = collections.defaultdict(list)
          index = 0

          for letter in string:
          dictionary[letter].append(index) # defaultdict FTW!
          index += 1
          return(dictionary)


          Next, give up your love affair with integers! Ned Batchelder gave a nice talk on the subject, so here's a link to that: https://www.youtube.com/watch?v=EnSu9hHGq5o



          The idea is that many (most?) python loops don't need to use integers to index over things. And when you do need an integer index (and in this case you do!) there are better ways than maintaining your own integer index variable. Here's one such way: the enumerate() built-in function.



          With enumerate, you can write a loop that iterates over the values from some iterable along with an automatically-associated integer:



              # No index=0 here
          for index, letter in enumerate(string):
          dictionary[letter].append(index)
          # No index+=1 here!


          The index, string pair is called a tuple, and it is a built-in type just like list. The act of assigning or iterating using multiple target variables that take their values from a single tuple is called tuple unpacking. (Remember that phrase, you'll need it when you want to ask for help on the subject.)



          In get_longest_word



          The issue I have with this function is not one of Python, but rather of design. You are given some words, d. You want to find the longest word that is a subsequence of the string s. How do you do it?



          In your case, the answer is "look at every single word in d, ignore the ones that are not subsequences of s, pick the longest one that remains."



          There are a couple of better (read: faster, more efficient) ways of doing that job. Let me make one simple suggestion: sort the words!



          In Python, there are Iterables and Sequences. An iterable is something you can iterate. A sequence is something you can access using s[i]. It's possible to have an infinite iterable, by writing an infinite generator function. It's not possible to have an infinite sequence, since you'll run out of memory trying to store it.



          In this particular case it seems okay to assume that d is going to be a sequence: a finite list or tuple. So the fastest way to find the "longest word" is to start by looking at long words first. Because once you find a long word that is a subsequence, you can stop - there are no shorter ones!



          The way to sort things in Python is the sorted built-in function. (Yes, it takes an iterable. No, it won't sort an infinite one. Yeesh!) By default, it sorts things by comparing the items using the "native" comparison. You can specify a key function, however, to sort using some different mechanism. Let's use the length of the words, which is the len function
          (function len(x) -> int returns the length of the list/string/whatever). And let's reverse the order, so that the big ones come first:



          d = sorted(d, key=len, reverse=True)


          Now, instead of needing to check the length and update the longest-word-so-far variable, you can just return immediately once you find a subsequence.



          In get_word_is_substring



          Let's talk about default values. You say:



          index_of_last_letter_found = None


          But :! grep letter_found % in Vim tells me:



              index_of_last_letter_found = None
          if letter in dictionary and (index_of_last_letter_found is None or dictionary[letter][-1] > index_of_last_letter_found):
          if index_of_last_letter_found is None or index_of_last_letter_found < dictionary[letter][index]:
          index_of_last_letter_found = dictionary[letter][index]


          You're spending a lot of keystrokes checking for None. And all you do is compare using <, and assign new values. Why not just set the default value to some value that you know is going to be "too low"? Since string index values start at 0, maybe -1 would make sense:



          index_of_last_letter_found = -1


          While were at it, shorten that variable name. Names should be as long as they need to be, and no longer!



          def get_word_is_substring(word, dictionary):
          last_index = -1 # Index of last letter found

          for letter in word:
          if letter in dictionary and dictionary[letter][-1] > last_index:
          index = 0
          while index < len(dictionary[letter]):
          if last_index < dictionary[letter][index]:
          last_index = dictionary[letter][index]
          break
          else:
          index += 1
          else:
          return False
          return True


          That's a lot more readable, since there are fewer tests and fewer characters.



          Next, let's go back and work on your fetish for simple integer arithmetic. Now that you've watched Ned Batchelder's talk, you can see that index += 1 is not the way!



          for index in dictionary[letter]:
          if last_index < index:
          last_index = index
          break


          (There are some other ways to find the first element in an iterable that matches a condition. See here for many of them. But this is nice and clear, and works.)



          Some words on "style"



          The above code may be somewhat more difficult to understand because I changed the meaning of index. In your original code, index was an index into the list of indices that pointed to where letters occur in the master sequence. In my updated version index means the actual indices from the list. I have removed one level of indirection.



          This is an example of why "too short" variable names are actually a good thing. You'll find a lot of people use very small names, like ch or i, to represent loop variables. This is because most of the time, loop variables are not a "concept" or a "noun". Instead, they are a subscript. We write a[i] because the original Teletype devices hand-carved out of wood by our pioneer forbears would not allow writing 𝓐ᵢ.



          But we don't really care about i, or about a[i]. We care about the thing stored there, and its properties. So, I encourage you to use short little 1-letter and 2-letter names when you are indexing a sequence, or when you're stuck in some other language that doesn't provide the nice looping options that Python does. Then you could say:



          for i ...
          index = s_dict[letter][i]


          And there would be no confusion between index and that-loop-variable.



          Next, the structuring of your escape clauses: I see a lot of people being taught to code in a way that puts exit conditions last. You're doing that in your loop:



              for letter in word:
          if letter in dictionary and dictionary[letter][-1] > last_index:
          # much code here
          else:
          return False


          But moving the exit condition to the top (by reversing the sense of the if statement) makes it clear what the alternatives are (if this is true, leave, otherwise ...) and reduces the indentation level of your code (by removing the else indent). Some people would have you use two-space tabs to address the indentation problem. Those people are all young with good eyesight. They're also quite wrong. :-)



              for letter in word:
          if letter not in dictionary or dictionary[letter][-1] <= last_index:
          return False

          # much code here


          Lastly, there's the name. You called it get_word_is_substring, but we want subsequences not substring, and there's no reason to say get_ at the beginning. In fact, the word can be assumed, since you're passing it in, so:



          def is_subsequence(word, s_dict):
          last_index = -1 # Index (in s) of last letter found

          for letter in word:
          if letter not in s_dict or s_dict[letter][-1] <= last_index:
          return False

          for index in s_dict[letter]:
          if last_index < index:
          last_index = index
          break

          return True


          (Note: Because I suggested s_dict be a defaultdict, you could eliminate the if test entirely and rely on the for ... else construct, which is rather obscure. I don't recommend it, simply because it's non-intuitive, hard to understand, and because spelling out the failure conditions is better than leaving them to the language. ("explicit is better than implicit"))



          The class



          Finally, let's talk about the class:



          class Main:


          But first! Here's another video for you to watch, Jack Diederich's talk at Pycon 2012 entitled: "Stop Writing Classes!": https://www.youtube.com/watch?v=o9pEzgHorH0



          As Jack says, "The signature of 'this shouldn't be a class' is that it has two methods, one of which is __init__."



          In this case, you've created a class, but you don't store any data and you only call one method from the outside. It should be a function.







          share|improve this answer














          share|improve this answer



          share|improve this answer








          edited 2 hours ago

























          answered 2 hours ago









          Austin HastingsAustin Hastings

          6,8371232




          6,8371232

























              1












              $begingroup$

              Use defaultdict and enumerate



              Python offers the defaultdict class, which works like a regular dictonary but returns a default value (here: empty list) if the requested element is not present. Also use enumerate if you want an index in a loop.



              from collections import defaultdict

              def create_dictionary(string):
              dictionary = defaultdict(list)
              for index, letter in enumerate(string):
              dictionary[letter].append(index)
              return dictionary


              In the next method, work with lists of indices to leverage the defaultdict. Also, use a list comprehension to filter for valid indices because that also defaults to an empty list, so you don't need any special case handling.



              def get_word_is_substring(word, dictionary):
              indices = [-1]
              for letter in word:
              indices = [index for index in dictionary[letter] if index > indices[0]]
              if not indices:
              return False
              return True





              share|improve this answer









              $endgroup$


















                1












                $begingroup$

                Use defaultdict and enumerate



                Python offers the defaultdict class, which works like a regular dictonary but returns a default value (here: empty list) if the requested element is not present. Also use enumerate if you want an index in a loop.



                from collections import defaultdict

                def create_dictionary(string):
                dictionary = defaultdict(list)
                for index, letter in enumerate(string):
                dictionary[letter].append(index)
                return dictionary


                In the next method, work with lists of indices to leverage the defaultdict. Also, use a list comprehension to filter for valid indices because that also defaults to an empty list, so you don't need any special case handling.



                def get_word_is_substring(word, dictionary):
                indices = [-1]
                for letter in word:
                indices = [index for index in dictionary[letter] if index > indices[0]]
                if not indices:
                return False
                return True





                share|improve this answer









                $endgroup$
















                  1












                  1








                  1





                  $begingroup$

                  Use defaultdict and enumerate



                  Python offers the defaultdict class, which works like a regular dictonary but returns a default value (here: empty list) if the requested element is not present. Also use enumerate if you want an index in a loop.



                  from collections import defaultdict

                  def create_dictionary(string):
                  dictionary = defaultdict(list)
                  for index, letter in enumerate(string):
                  dictionary[letter].append(index)
                  return dictionary


                  In the next method, work with lists of indices to leverage the defaultdict. Also, use a list comprehension to filter for valid indices because that also defaults to an empty list, so you don't need any special case handling.



                  def get_word_is_substring(word, dictionary):
                  indices = [-1]
                  for letter in word:
                  indices = [index for index in dictionary[letter] if index > indices[0]]
                  if not indices:
                  return False
                  return True





                  share|improve this answer









                  $endgroup$



                  Use defaultdict and enumerate



                  Python offers the defaultdict class, which works like a regular dictonary but returns a default value (here: empty list) if the requested element is not present. Also use enumerate if you want an index in a loop.



                  from collections import defaultdict

                  def create_dictionary(string):
                  dictionary = defaultdict(list)
                  for index, letter in enumerate(string):
                  dictionary[letter].append(index)
                  return dictionary


                  In the next method, work with lists of indices to leverage the defaultdict. Also, use a list comprehension to filter for valid indices because that also defaults to an empty list, so you don't need any special case handling.



                  def get_word_is_substring(word, dictionary):
                  indices = [-1]
                  for letter in word:
                  indices = [index for index in dictionary[letter] if index > indices[0]]
                  if not indices:
                  return False
                  return True






                  share|improve this answer












                  share|improve this answer



                  share|improve this answer










                  answered 2 hours ago









                  Rainer P.Rainer P.

                  1,221414




                  1,221414






















                      K. Kretz is a new contributor. Be nice, and check out our Code of Conduct.










                      draft saved

                      draft discarded


















                      K. Kretz is a new contributor. Be nice, and check out our Code of Conduct.













                      K. Kretz is a new contributor. Be nice, and check out our Code of Conduct.












                      K. Kretz is a new contributor. Be nice, and check out our Code of Conduct.
















                      Thanks for contributing an answer to Code Review Stack Exchange!


                      • Please be sure to answer the question. Provide details and share your research!

                      But avoid



                      • Asking for help, clarification, or responding to other answers.

                      • Making statements based on opinion; back them up with references or personal experience.


                      Use MathJax to format equations. MathJax reference.


                      To learn more, see our tips on writing great answers.




                      draft saved


                      draft discarded














                      StackExchange.ready(
                      function () {
                      StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f214408%2ffind-the-longest-word-in-set-d-that-is-a-subsequence-of-string-s%23new-answer', 'question_page');
                      }
                      );

                      Post as a guest















                      Required, but never shown





















































                      Required, but never shown














                      Required, but never shown












                      Required, but never shown







                      Required, but never shown

































                      Required, but never shown














                      Required, but never shown












                      Required, but never shown







                      Required, but never shown







                      Popular posts from this blog

                      Benedict Cumberbatch Contingut Inicis Debut professional Premis Filmografia bàsica Premis i...

                      Monticle de plataforma Contingut Est de Nord Amèrica Interpretacions Altres cultures Vegeu...

                      Escacs Janus Enllaços externs Menú de navegacióEscacs JanusJanusschachBrainKing.comChessV