/* * libpinyin * Library to deal with pinyin. * * Copyright (C) 2010 Peng Wu * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include #include #include #include #include "pinyin_internal.h" /* graph shortest path sentence segment. */ /* Note: * Currently libpinyin only supports ucs2 characters, as this is a * pre-processor tool for raw corpus, it will skip all sentences * which contains non-ucs2 characters. */ struct SegmentStep{ phrase_token_t m_handle; ucs4_t * m_phrase; size_t m_phrase_len; //use formula W = number of words. Zero handle means one word. guint m_nword; //backtrace information, -1 one step backward. gint m_backward_nstep; public: SegmentStep(){ m_handle = 0; m_phrase = NULL; m_phrase_len = 0; m_nword = UINT_MAX; m_backward_nstep = -0; } }; bool backtrace(GArray * steps, glong phrase_len, GArray * strings); //Note: do not free phrase, as it is used by strings (array of segment). bool segment(PhraseLargeTable * phrases, //Lookup Phrase ucs4_t * phrase, glong phrase_len, GArray * strings /* Array of Segment *. */){ /* Prepare for shortest path segment dynamic programming. */ GArray * steps = g_array_new(TRUE, TRUE, sizeof(SegmentStep)); SegmentStep step; for ( glong i = 0; i < phrase_len + 1; ++i ){ g_array_append_val(steps, step); } SegmentStep * first_step = &g_array_index(steps, SegmentStep, 0); first_step->m_nword = 0; for ( glong i = 0; i < phrase_len + 1; ++i ) { SegmentStep * step_begin = &g_array_index(steps, SegmentStep, i); size_t nword = step_begin->m_nword; for ( glong k = i + 1; k < phrase_len + 1; ++k ) { size_t len = k - i; ucs4_t * cur_phrase = phrase + i; phrase_token_t token = 0; int result = phrases->search(len, cur_phrase, token); if ( !(result & SEARCH_OK) ){ token = 0; if ( 1 != len ) continue; } ++nword; SegmentStep * step_end = &g_array_index(steps, SegmentStep, k); if ( nword < step_end->m_nword ) { step_end->m_handle = token; step_end->m_phrase = cur_phrase; step_end->m_phrase_len = len; step_end->m_nword = nword; step_end->m_backward_nstep = i - k; } if ( !(result & SEARCH_CONTINUED) ) break; } } return backtrace(steps, phrase_len, strings); } bool backtrace(GArray * steps, glong phrase_len, GArray * strings){ //backtracing to get the result. size_t cur_step = phrase_len; g_array_set_size(strings, 0); while ( cur_step ){ SegmentStep * step = &g_array_index(steps, SegmentStep, cur_step); g_array_append_val(strings, *step); cur_step = cur_step + step->m_backward_nstep; //intended to avoid leaking internal informations step->m_nword = 0; step->m_backward_nstep = 0; } //reverse the strings for ( size_t i = 0; i < strings->len / 2; ++i ) { SegmentStep * head, * tail; head = &g_array_index(strings, SegmentStep, i); tail = &g_array_index(strings, SegmentStep, strings->len - 1 - i ); SegmentStep tmp; tmp = *head; *head = *tail; *tail = tmp; } g_array_free(steps, TRUE); return true; } void print_help(){ printf("Usage: spseg [--generate-extra-enter]\n"); } int main(int argc, char * argv[]){ int i = 1; bool gen_extra_enter = false; setlocale(LC_ALL, ""); //deal with options. while ( i < argc ){ if ( strcmp ("--help", argv[i]) == 0) { print_help(); exit(0); } else if (strcmp("--generate-extra-enter", argv[i]) == 0) { gen_extra_enter = true; } else { print_help(); exit(EINVAL); } ++i; } //init phrase table PhraseLargeTable phrase_table; MemoryChunk * chunk = new MemoryChunk; chunk->load("phrase_index.bin"); phrase_table.load(chunk); char * linebuf = NULL; size_t size = 0; ssize_t read; while( (read = getline(&linebuf, &size, stdin)) != -1 ){ if ( '\n' == linebuf[strlen(linebuf) - 1] ) { linebuf[strlen(linebuf) - 1] = '\0'; } //check non-ucs2 characters const glong num_of_chars = g_utf8_strlen(linebuf, -1); glong len = 0; ucs4_t * sentence = g_utf8_to_ucs4(linebuf, -1, NULL, &len, NULL); if ( len != num_of_chars ) { fprintf(stderr, "non-ucs2 characters encountered:%s.\n", linebuf); printf("\n"); continue; } //do segment stuff GArray * strings = g_array_new(TRUE, TRUE, sizeof(SegmentStep)); segment(&phrase_table, sentence, len, strings); //print out the split phrase for ( glong i = 0; i < strings->len; ++i ) { SegmentStep * step = &g_array_index(strings, SegmentStep, i); char * string = g_ucs4_to_utf8( step->m_phrase, step->m_phrase_len, NULL, NULL, NULL); printf("%s\n", string); g_free(string); } /* print extra enter */ if ( gen_extra_enter ) printf("\n"); g_array_free(strings, TRUE); g_free(sentence); } /* print enter at file tail */ printf("\n"); return 0; }