java - 如何使用马尔可夫链生成句子?

标签 java string dictionary bots markov-chains

我正在尝试使用马尔可夫链制作一个简单的聊天机器人。我已经能够使用输入文本中的模式成功创建字典,但我无法弄清楚如何使用它来生成句子。

import java.text.BreakIterator;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;

final class MarkovChain {

    private static final BreakIterator sentenceIterator = BreakIterator.getSentenceInstance();
    private static final BreakIterator wordIterator = BreakIterator.getWordInstance();

    private static final Map<String, List<String>> dictionary = new TreeMap<>();

    public static void addDictionary(String string) {
        string = string.toLowerCase().trim();
        for (final String sentence : splitSentences(string)) {
            String lastWord = null, lastLastWord = null;
            for (final String word : splitWords(sentence)) {
                if (lastLastWord != null) {
                    final String key = lastLastWord + ' ' + lastWord;
                    List<String> value = dictionary.get(key);
                    if (value == null)
                        value = new ArrayList<>();
                    value.add(word);
                    dictionary.put(key, value);
                }
                lastLastWord = lastWord;
                lastWord = word;
            }
        }
    }

    private static List<String> splitSentences(final String string) {
        sentenceIterator.setText(string);
        final List<String> sentences = new ArrayList<>();
        for (int start = sentenceIterator.first(), end = sentenceIterator.next(); end != BreakIterator.DONE; start = end, end = sentenceIterator.next()) {
            sentences.add(string.substring(start, end).trim());
        }
        return sentences;
    }

    private static List<String> splitWords(final String string) {
        wordIterator.setText(string);
        final List<String> words = new ArrayList<>();
        for (int start = wordIterator.first(), end = wordIterator.next(); end != BreakIterator.DONE; start = end, end = wordIterator.next()) {
            String word = string.substring(start, end).trim();
            if (word.length() > 0 && Character.isLetterOrDigit(word.charAt(0)))
                words.add(word);
        }
        return words;
    }
}

我如何从字典中生成句子?

最佳答案

以下是我将如何更改您的代码以使其能够生成句子。我添加了Map<String, List<String>> singleWords将前一个单词指向可能的下一个单词的列表,并在循环中迭代句子中的单词来填充此映射的代码。此外,我在单词列表的两侧添加了点,以便注册称为“第一个单词之前”和“最后一个单词之后”的特殊状态(请参阅 addDots(...) )。

import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.text.BreakIterator;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.TreeMap;

final class MarkovChain {

    private static final BreakIterator sentenceIterator = BreakIterator.getSentenceInstance();
    private static final BreakIterator wordIterator = BreakIterator.getWordInstance();

    private static final Map<String, List<String>> singleWords = new TreeMap<>();
    private static final Map<String, List<String>> dictionary = new TreeMap<>();

    public static void main(String[] args) throws Exception {
        String text = new String(Files.readAllBytes(Paths.get("text.txt")), Charset.defaultCharset());
        addDictionary(text);
        StringBuilder output = new StringBuilder();
        generateSentence(singleWords, dictionary, output, 5);
        System.out.println(output.toString());
    }

    public static void addDictionary(String string) {
        string = string.toLowerCase().trim();
        for (final String sentence : splitSentences(string)) {
            String lastWord = null, lastLastWord = null;
            for (final String word : addDots(splitWords(sentence))) {
                if (lastLastWord != null) {
                    final String key = lastLastWord + ' ' + lastWord;
                    List<String> value = dictionary.get(key);
                    if (value == null)
                        value = new ArrayList<>();
                    value.add(word);
                    dictionary.put(key, value);
                }
                if (lastWord != null) {
                    final String key = lastWord;
                    List<String> value = singleWords.get(key);
                    if (value == null)
                        value = new ArrayList<>();
                    value.add(word);
                    singleWords.put(key, value);
                }
                lastLastWord = lastWord;
                lastWord = word;
            }
        }
    }

    private static List<String> splitSentences(final String string) {
        sentenceIterator.setText(string);
        final List<String> sentences = new ArrayList<>();
        for (int start = sentenceIterator.first(), end = sentenceIterator.next(); end != BreakIterator.DONE; start = end, end = sentenceIterator.next()) {
            sentences.add(string.substring(start, end).trim());
        }
        return sentences;
    }

    private static List<String> splitWords(final String string) {
        wordIterator.setText(string);
        final List<String> words = new ArrayList<>();
        for (int start = wordIterator.first(), end = wordIterator.next(); end != BreakIterator.DONE; start = end, end = wordIterator.next()) {
            String word = string.substring(start, end).trim();
            if (word.length() > 0 && Character.isLetterOrDigit(word.charAt(0)))
                words.add(word);
        }
        return words;
    }

    private static List<String> addDots(List<String> words) {
        words.add(0, ".");
        words.add(".");
        return words;
    }

    public static void generateSentence(Map<String, List<String>> singleWords,
            Map<String, List<String>> dictionary, StringBuilder target, int count) {
        Random r = new Random();
        for (int i = 0; i < 5; i++) {
            String w1 = ".";
            String w2 = pickRandom(singleWords.get(w1), r);
            while (w2 != null) {
                target.append(w2).append(" ");              
                if (w2.equals("."))
                    break;
                String w3 = pickRandom(dictionary.get(w1 + " " + w2), r);
                w1 = w2;
                w2 = w3;
            }
            target.append("\n");
        }
    }

    private static String pickRandom(List<String> alternatives, Random r) {
        return alternatives.get(r.nextInt(alternatives.size()));
    }
}

我应该指出的是,这种方法并未经过优化。如果我需要提高效率,我会计算字典图中的单词数量,最后将它们标准化以产生频率。类似:Map<String, Map<String, Double>> dictionary ,其中内部映射将单词指向频率。不过,它需要选择与我的示例中不同的单词。

关于java - 如何使用马尔可夫链生成句子?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33159881/

相关文章:

java - GWT CellTable 选择并单击 CheckBoxCell

java - 如何做 “sequential” 作业调度( quartz ?)

java - 为什么我的 arrayList 只添加来自 StringTokenizer 的最后一个标记?

php - 显示 Woocommerce 存档页面中的可用产品尺寸或尺寸计数

java - 替换字符串 Java

java构造函数访问(如果我们在子类中有私有(private)构造函数)

c# - 覆盖 List<MyClass> 的 ToString()

python - 如何将字符串 append 到字典中的列表

c++ - 2个unique_ptr的 map

c# - 在 C# 中合并具有相似键但不同值的字典