Giter VIP home page Giter VIP logo

soynlp's Introduction

Table of contents

soynlp

한국어 분석을 위한 pure python code 입니다. 학습데이터를 이용하지 않으면서 데이터에 존재하는 단어를 찾거나, 문장을 단어열로 분해, 혹은 품사 판별을 할 수 있는 비지도학습 접근법을 지향합니다.

Guide

Usage guide

soynlp 에서 제공하는 WordExtractor 나 NounExtractor 는 여러 개의 문서로부터 학습한 통계 정보를 이용하여 작동합니다. 비지도학습 기반 접근법들은 통계적 패턴을 이용하여 단어를 추출하기 때문에 하나의 문장 혹은 문서에서 보다는 어느 정도 규모가 있는 동일한 집단의 문서 (homogeneous documents) 에서 잘 작동합니다. 영화 댓글들이나 하루의 뉴스 기사처럼 같은 단어를 이용하는 집합의 문서만 모아서 Extractors 를 학습하시면 좋습니다. 이질적인 집단의 문서들은 하나로 모아 학습하면 단어가 잘 추출되지 않습니다.

Parameter naming

soynlp=0.0.46 까지는 min_score, minimum_score, l_len_min 처럼 최소값이나 최대값을 요구하는 parameters 의 이름들에 규칙이 없었습니다. 지금까지 작업하신 코드들 중에서 직접 parameters 를 설정하신 분들에게 혼란을 드릴 수 있으나, 더 늦기전에 이후에 발생할 불편함을 줄이기 위하여 변수 명을 수정하였습니다.

0.0.47 이후 minimum, maximum 의 의미가 들어가는 변수명은 min, max 로 줄여 기입합니다. 그 뒤에 어떤 항목의 threshold parameter 인지 이름을 기입합니다. 다음과 같은 패턴으로 parameter 이름을 통일합니다. {min, max}_{noun, word}_{score, threshold} 등으로 이름을 통일합니다. 항목이 자명한 경우에는 이를 생략할 수 있습니다.

soynlp 에서는 substring counting 을 하는 경우가 많습니다. 빈도수와 관련된 parameter 는 count 가 아닌 frequency 로 통일합니다.

index 와 idx 는 idx 로 통일합니다.

숫자를 의미하는 num 과 n 은 num 으로 통일합니다.

Setup

$ pip install soynlp

Python version

  • Python 3.5+ 를 지원합니다. 3.x 에서 주로 작업을 하기 때문에 3.x 로 이용하시길 권장합니다.
  • Python 2.x 는 모든 기능에 대해서 테스트가 끝나지 않았습니다.

Requires

  • numpy >= 1.12.1
  • psutil >= 5.0.1
  • scipy >= 1.1.0
  • scikit-learn >= 0.20.0

Noun Extractor

명사 추출을 하기 위해 여러 시도를 한 결과, v1, news, v2 세 가지 버전이 만들어졌습니다. 가장 좋은 성능을 보이는 것은 v2 입니다.

WordExtractor 는 통계를 이용하여 단어의 경계 점수를 학습하는 것일 뿐, 각 단어의 품사를 판단하지는 못합니다. 때로는 각 단어의 품사를 알아야 하는 경우가 있습니다. 또한 다른 품사보다도 명사에서 새로운 단어가 가장 많이 만들어집니다. 명사의 오른쪽에는 -은, -는, -라는, -하는 처럼 특정 글자들이 자주 등장합니다. 문서의 어절 (띄어쓰기 기준 유닛)에서 왼쪽에 위치한 substring 의 오른쪽에 어떤 글자들이 등장하는지 분포를 살펴보면 명사인지 아닌지 판단할 수 있습니다. soynlp 에서는 두 가지 종류의 명사 추출기를 제공합니다. 둘 모두 개발 단계이기 때문에 어떤 것이 더 우수하다 말하기는 어렵습니다만, NewsNounExtractor 가 좀 더 많은 기능을 포함하고 있습니다. 추후, 명사 추출기는 하나의 클래스로 정리될 예정입니다.

Noun Extractor ver 1 & News Noun Extractor

from soynlp.noun import LRNounExtractor
noun_extractor = LRNounExtractor()
nouns = noun_extractor.train_extract(sentences) # list of str like

from soynlp.noun import NewsNounExtractor
noun_extractor = NewsNounExtractor()
nouns = noun_extractor.train_extract(sentences) # list of str like

2016-10-20 의 뉴스로부터 학습한 명사의 예시입니다.

덴마크  웃돈  너무너무너무  가락동  매뉴얼  지도교수
전망치  강구  언니들  신산업  기뢰전  노스
할리우드  플라자  불법조업  월스트리트저널  2022년  불허
고씨  어플  1987년  불씨  적기  레스
스퀘어  충당금  건축물  뉴질랜드  사각  하나씩
근대  투자주체별  4위  태권  네트웍스  모바일게임
연동  런칭  만성  손질  제작법  현실화
오해영  심사위원들  단점  부장조리  차관급  게시물
인터폰  원화  단기간  편곡  무산  외국인들
세무조사  석유화학  워킹  원피스  서장  공범

더 자세한 설명은 튜토리얼에 있습니다.

Noun Extractor ver 2

soynlp=0.0.46+ 에서는 명사 추출기 version 2 를 제공합니다. 이전 버전의 명사 추출의 정확성과 합성명사 인식 능력, 출력되는 정보의 오류를 수정한 버전입니다. 사용법은 version 1 과 비슷합니다.

from soynlp.utils import DoublespaceLineCorpus
from soynlp.noun import LRNounExtractor_v2

corpus_path = '2016-10-20-news'
sents = DoublespaceLineCorpus(corpus_path, iter_sent=True)

noun_extractor = LRNounExtractor_v2(verbose=True)
nouns = noun_extractor.train_extract(sents)

추출된 nouns 는 {str:namedtuple} 형식입니다.

print(nouns['뉴스']) # NounScore(frequency=4319, score=1.0)

_compounds_components 에는 복합명사를 구성하는 단일명사들의 정보가 저장되어 있습니다. '대한민국', '녹색성장'과 같이 실제로는 복합형태소이지만, 단일 명사로 이용되는 경우는 단일 명사로 인식합니다.

list(noun_extractor._compounds_components.items())[:5]

# [('잠수함발사탄도미사일', ('잠수함', '발사', '탄도미사일')),
#  ('미사일대응능력위원회', ('미사일', '대응', '능력', '위원회')),
#  ('글로벌녹색성장연구소', ('글로벌', '녹색성장', '연구소')),
#  ('시카고옵션거래소', ('시카고', '옵션', '거래소')),
#  ('대한민국특수임무유공', ('대한민국', '특수', '임무', '유공')),

LRGraph 는 학습된 corpus 에 등장한 어절의 L-R 구조를 저장하고 있습니다. get_r 과 get_l 을 이용하여 이를 확인할 수 있습니다.

noun_extractor.lrgraph.get_r('아이오아이')

# [('', 123),
#  ('의', 47),
#  ('는', 40),
#  ('와', 18),
#  ('가', 18),
#  ('에', 7),
#  ('에게', 6),
#  ('까지', 2),
#  ('랑', 2),
#  ('부터', 1)]

더 자세한 설명은 튜토리얼 2에 있습니다.

Word Extraction

2016 년 10월의 연예기사 뉴스에는 '트와이스', '아이오아이' 와 같은 단어가 존재합니다. 하지만 말뭉치를 기반으로 학습된 품사 판별기 / 형태소 분석기는 이런 단어를 본 적이 없습니다. 늘 새로운 단어가 만들어지기 때문에 학습하지 못한 단어를 제대로 인식하지 못하는 미등록단어 문제 (out of vocabulry, OOV) 가 발생합니다. 하지만 이 시기에 작성된 여러 개의 연예 뉴스 기사를 읽다보면 '트와이스', '아이오아이' 같은 단어가 등장함을 알 수 있고, 사람은 이를 학습할 수 있습니다. 문서집합에서 자주 등장하는 연속된 단어열을 단어라 정의한다면, 우리는 통계를 이용하여 이를 추출할 수 있습니다. 통계 기반으로 단어(의 경계)를 학습하는 방법은 다양합니다. soynlp는 그 중, Cohesion score, Branching Entropy, Accessor Variety 를 제공합니다.

from soynlp.word import WordExtractor

word_extractor = WordExtractor(min_frequency=100,
    min_cohesion_forward=0.05, 
    min_right_branching_entropy=0.0
)
word_extractor.train(sentences) # list of str or like
words = word_extractor.extract()

words 는 Scores 라는 namedtuple 을 value 로 지니는 dict 입니다.

words['아이오아이']

Scores(cohesion_forward=0.30063636035733476,
        cohesion_backward=0,
        left_branching_entropy=0,
        right_branching_entropy=0,
        left_accessor_variety=0,
        right_accessor_variety=0,
        leftside_frequency=270,
        rightside_frequency=0
)

2016-10-26 의 뉴스 기사로부터 학습한 단어 점수 (cohesion * branching entropy) 기준으로 정렬한 예시입니다.

단어   (빈도수, cohesion, branching entropy)

촬영     (2222, 1.000, 1.823)
서울     (25507, 0.657, 2.241)
들어     (3906, 0.534, 2.262)
롯데     (1973, 0.999, 1.542)
한국     (9904, 0.286, 2.729)
북한     (4954, 0.766, 1.729)
투자     (4549, 0.630, 1.889)
떨어     (1453, 0.817, 1.515)
진행     (8123, 0.516, 1.970)
얘기     (1157, 0.970, 1.328)
운영     (4537, 0.592, 1.768)
프로그램  (2738, 0.719, 1.527)
클린턴   (2361, 0.751, 1.420)
뛰어     (927, 0.831, 1.298)
드라마   (2375, 0.609, 1.606)
우리     (7458, 0.470, 1.827)
준비     (1736, 0.639, 1.513)
루이     (1284, 0.743, 1.354)
트럼프   (3565, 0.712, 1.355)
생각     (3963, 0.335, 2.024)
팬들     (999, 0.626, 1.341)
산업     (2203, 0.403, 1.769)
10      (18164, 0.256, 2.210)
확인     (3575, 0.306, 2.016)
필요     (3428, 0.635, 1.279)
문제     (4737, 0.364, 1.808)
혐의     (2357, 0.962, 0.830)
평가     (2749, 0.362, 1.787)
20      (59317, 0.667, 1.171)
스포츠    (3422, 0.428, 1.604)

자세한 내용은 word extraction tutorial 에 있습니다. 현재 버전에서 제공하는 기능은 다음과 같습니다.

Tokenizer

WordExtractor 로부터 단어 점수를 학습하였다면, 이를 이용하여 단어의 경계를 따라 문장을 단어열로 분해할 수 있습니다. soynlp 는 세 가지 토크나이저를 제공합니다. 띄어쓰기가 잘 되어 있다면 LTokenizer 를 이용할 수 있습니다. 한국어 어절의 구조를 "명사 + 조사" 처럼 "L + [R]" 로 생각합니다.

LTokenizer

L parts 에는 명사/동사/형용사/부사가 위치할 수 있습니다. 어절에서 L 만 잘 인식한다면 나머지 부분이 R parts 가 됩니다. LTokenizer 에는 L parts 의 단어 점수를 입력합니다.

from soynlp.tokenizer import LTokenizer

scores = {'데이':0.5, '데이터':0.5, '데이터마이닝':0.5, '공부':0.5, '공부중':0.45}
tokenizer = LTokenizer(scores=scores)

sent = '데이터마이닝을 공부한다'

print(tokenizer.tokenize(sent, flatten=False))
#[['데이터마이닝', '을'], ['공부', '중이다']]

print(tokenizer.tokenize(sent))
# ['데이터마이닝', '을', '공부', '중이다']

만약 WordExtractor 를 이용하여 단어 점수를 계산하였다면, 단어 점수 중 하나를 택하여 scores 를 만들 수 있습니다. 아래는 Forward cohesion 의 점수만을 이용하는 경우입니다. 그 외에도 다양하게 단어 점수를 정의하여 이용할 수 있습니다.

from soynlp.word import WordExtractor
from soynlp.utils import DoublespaceLineCorpus

file_path = 'your file path'
corpus = DoublespaceLineCorpus(file_path, iter_sent=True)

word_extractor = WordExtractor(
    min_frequency=100, # example
    min_cohesion_forward=0.05,
    min_right_branching_entropy=0.0
)

word_extractor.train(sentences)
words = word_extractor.extract()

cohesion_score = {word:score.cohesion_forward for word, score in words.items()}
tokenizer = LTokenizer(scores=cohesion_score)

명사 추출기의 명사 점수와 Cohesion 을 함께 이용할 수도 있습니다. 한 예로, "Cohesion 점수 + 명사 점수"를 단어 점수로 이용하려면 아래처럼 작업할 수 있습니다.

from soynlp.noun import LRNounExtractor_2
noun_extractor = LRNounExtractor_v2()
nouns = noun_extractor.train_extract(corpus) # list of str like

noun_scores = {noun:score.score for noun, score in nouns.items()}
combined_scores = {noun:score + cohesion_score.get(noun, 0)
    for noun, score in noun_scores.items()}
combined_scores.update(
    {subword:cohesion for subword, cohesion in cohesion_score.items()
    if not (subword in combined_scores)}
)

tokenizer = LTokenizer(scores=combined_scores)

MaxScoreTokenizer

띄어쓰기가 제대로 지켜지지 않은 데이터라면, 문장의 띄어쓰기 기준으로 나뉘어진 단위가 L + [R] 구조라 가정할 수 없습니다. 하지만 사람은 띄어쓰기가 지켜지지 않은 문장에서 익숙한 단어부터 눈에 들어옵니다. 이 과정을 모델로 옮긴 MaxScoreTokenizer 역시 단어 점수를 이용합니다.

from soynlp.tokenizer import MaxScoreTokenizer

scores = {'파스': 0.3, '파스타': 0.7, '좋아요': 0.2, '좋아':0.5}
tokenizer = MaxScoreTokenizer(scores=scores)

print(tokenizer.tokenize('난파스타가좋아요'))
# ['난', '파스타', '가', '좋아', '요']

print(tokenizer.tokenize('난파스타가 좋아요', flatten=False))
# [[('난', 0, 1, 0.0, 1), ('파스타', 1, 4, 0.7, 3),  ('가', 4, 5, 0.0, 1)],
#  [('좋아', 0, 2, 0.5, 2), ('요', 2, 3, 0.0, 1)]]

MaxScoreTokenizer 역시 WordExtractor 의 결과를 이용하실 때에는 위의 예시처럼 적절히 scores 를 만들어 사용합니다. 이미 알려진 단어 사전이 있다면 이 단어들은 다른 어떤 단어보다도 더 큰 점수를 부여하면 그 단어는 토크나이저가 하나의 단어로 잘라냅니다.

RegexTokenizer

규칙 기반으로도 단어열을 만들 수 있습니다. 언어가 바뀌는 부분에서 우리는 단어의 경계를 인식합니다. 예를 들어 "아이고ㅋㅋㅜㅜ진짜?" 는 [아이고, ㅋㅋ, ㅜㅜ, 진짜, ?]로 쉽게 단어열을 나눕니다.

from soynlp.tokenizer import RegexTokenizer

tokenizer = RegexTokenizer()

print(tokenizer.tokenize('이렇게연속된문장은잘리지않습니다만'))
# ['이렇게연속된문장은잘리지않습니다만']

print(tokenizer.tokenize('숫자123이영어abc에섞여있으면ㅋㅋ잘리겠죠'))
# ['숫자', '123', '이영어', 'abc', '에섞여있으면', 'ㅋㅋ', '잘리겠죠']

Part of Speech Tagger

단어 사전이 잘 구축되어 있다면, 이를 이용하여 사전 기반 품사 판별기를 만들 수 있습니다. 단, 형태소분석을 하는 것이 아니기 때문에 '하는', '하다', '하고'는 모두 동사에 해당합니다. Lemmatizer 는 현재 개발/정리 중입니다.

pos_dict = {
    'Adverb': {'너무', '매우'}, 
    'Noun': {'너무너무너무', '아이오아이', '아이', '노래', '오', '이', '고양'},
    'Josa': {'는', '의', '이다', '입니다', '이', '이는', '를', '라', '라는'},
    'Verb': {'하는', '하다', '하고'},
    'Adjective': {'예쁜', '예쁘다'},
    'Exclamation': {'우와'}    
}

from soynlp.postagger import Dictionary
from soynlp.postagger import LRTemplateMatcher
from soynlp.postagger import LREvaluator
from soynlp.postagger import SimpleTagger
from soynlp.postagger import UnknowLRPostprocessor

dictionary = Dictionary(pos_dict)
generator = LRTemplateMatcher(dictionary)    
evaluator = LREvaluator()
postprocessor = UnknowLRPostprocessor()
tagger = SimpleTagger(generator, evaluator, postprocessor)

sent = '너무너무너무는아이오아이의노래입니다!!'
print(tagger.tag(sent))
# [('너무너무너무', 'Noun'),
#  ('는', 'Josa'),
#  ('아이오아이', 'Noun'),
#  ('의', 'Josa'),
#  ('노래', 'Noun'),
#  ('입니다', 'Josa'),
#  ('!!', None)]

더 자세한 사용법은 사용법 튜토리얼 에 기술되어 있으며, 개발과정 노트는 여기에 기술되어 있습니다.

Vectorizer

토크나이저를 학습하거나, 혹은 학습된 토크나이저를 이용하여 문서를 sparse matrix 로 만듭니다. minimum / maximum of term frequency / document frequency 를 조절할 수 있습니다. Verbose mode 에서는 현재의 벡터라이징 상황을 print 합니다.

vectorizer = BaseVectorizer(
    tokenizer=tokenizer,
    min_tf=0,
    max_tf=10000,
    min_df=0,
    max_df=1.0,
    stopwords=None,
    lowercase=True,
    verbose=True
)

corpus.iter_sent = False
x = vectorizer.fit_transform(corpus)

문서의 크기가 크거나, 곧바로 sparse matrix 를 이용할 것이 아니라면 이를 메모리에 올리지 않고 그대로 파일로 저장할 수 있습니다. fit_to_file() 혹은 to_file() 함수는 하나의 문서에 대한 term frequency vector 를 얻는대로 파일에 기록합니다. BaseVectorizer 에서 이용할 수 있는 parameters 는 동일합니다.

vectorizer = BaseVectorizer(min_tf=1, tokenizer=tokenizer)
corpus.iter_sent = False

matrix_path = 'YOURS'
vectorizer.fit_to_file(corpus, matrix_path)

하나의 문서를 sparse matrix 가 아닌 list of int 로 출력이 가능합니다. 이 때 vectorizer.vocabulary_ 에 학습되지 않은 단어는 encoding 이 되지 않습니다.

vectorizer.encode_a_doc_to_bow('오늘 뉴스는 이것이 전부다')
# {3: 1, 258: 1, 428: 1, 1814: 1}

list of int 는 list of str 로 decoding 이 가능합니다.

vectorizer.decode_from_bow({3: 1, 258: 1, 428: 1, 1814: 1})
# {'뉴스': 1, '는': 1, '오늘': 1, '이것이': 1}

dict 형식의 bag of words 로도 encoding 이 가능합니다.

vectorizer.encode_a_doc_to_list('오늘의 뉴스는 매우 심각합니다')
# [258, 4, 428, 3, 333]

dict 형식의 bag of words 는 decoding 이 가능합니다.

vectorizer.decode_from_list([258, 4, 428, 3, 333])
['오늘', '의', '뉴스', '는', '매우']

Normalizer

대화 데이터, 댓글 데이터에 등장하는 반복되는 이모티콘의 정리 및 한글, 혹은 텍스트만 남기기 위한 함수를 제공합니다.

from soynlp.normalizer import *

emoticon_normalize('ㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋ쿠ㅜㅜㅜㅜㅜㅜ', num_repeats=3)
# 'ㅋㅋㅋㅜㅜㅜ'

repeat_normalize('와하하하하하하하하하핫', num_repeats=2)
# '와하하핫'

only_hangle('가나다ㅏㅑㅓㅋㅋ쿠ㅜㅜㅜabcd123!!아핫')
# '가나다ㅏㅑㅓㅋㅋ쿠ㅜㅜㅜ 아핫'

only_hangle_number('가나다ㅏㅑㅓㅋㅋ쿠ㅜㅜㅜabcd123!!아핫')
# '가나다ㅏㅑㅓㅋㅋ쿠ㅜㅜㅜ 123 아핫'

only_text('가나다ㅏㅑㅓㅋㅋ쿠ㅜㅜㅜabcd123!!아핫')
# '가나다ㅏㅑㅓㅋㅋ쿠ㅜㅜㅜabcd123!!아핫'

더 자세한 설명은 튜토리얼에 있습니다.

Point-wise Mutual Information (PMI)

연관어 분석을 위한 co-occurrence matrix 계산과 이를 이용한 Point-wise Mutual Information (PMI) 계산을 위한 함수를 제공합니다.

아래 sent_to_word_contexts_matrix 함수를 이용하여 (word, context words) matrix 를 만들 수 있습니다. x 는 scipy.sparse.csr_matrix 이며, (n_vocabs, n_vocabs) 크기입니다. idx2vocab 은 x 의 각 row, column 에 해당하는 단어가 포함된 list of str 입니다. 문장의 앞/뒤 windows 단어를 context 로 인식하며, min_tf 이상의 빈도수로 등장한 단어에 대해서만 계산을 합니다. dynamic_weight 는 context 길이에 반비례하여 weighting 을 합니다. windows 가 3 일 경우, 1, 2, 3 칸 떨어진 단어의 co-occurrence 는 1, 2/3, 1/3 으로 계산됩니다.

from soynlp.vectorizer import sent_to_word_contexts_matrix

x, idx2vocab = sent_to_word_contexts_matrix(
    corpus,
    windows=3,
    min_tf=10,
    tokenizer=tokenizer, # (default) lambda x:x.split(),
    dynamic_weight=False,
    verbose=True
)

Co-occurrence matrix 인 x 를 pmi 에 입력하면 row 와 column 을 각 축으로 PMI 가 계산됩니다. pmi_dok 은 scipy.sparse.dok_matrix 형식입니다. min_pmi 이상의 값만 저장되며, default 는 min_pmi = 0 이기 때문에 Positive PMI (PPMI) 입니다. alpha 는 PMI(x,y) = p(x,y) / ( p(x) * ( p(y) + alpha ) ) 에 입력되는 smoothing parameter 입니다. 계산 과정이 오래 걸리기 때문에 verbose = True 로 설정하면 현재의 진행 상황을 출력합니다.

from soynlp.word import pmi

pmi_dok = pmi(
    x,
    min_pmi=0,
    alpha=0.0001,
    verbose=True
)

더 자세한 설명은 튜토리얼에 있습니다.

notes

Slides

  • slide files에 알고리즘들의 원리 및 설명을 적어뒀습니다. 데이터야놀자에서 발표했던 자료입니다.
  • textmining tutorial 을 만들고 있습니다. soynlp project 에서 구현 중인 알고리즘들의 설명 및 텍스트 마이닝에 이용되는 머신 러닝 방법들을 설명하는 slides 입니다.

Blogs

  • github io blog 에서 slides 에 있는 내용들의 텍스트 설명 글들을 올리고 있습니다. Slides 의 내용에 대해 더 자세하게 보고 싶으실 때 읽으시길 권합니다.

함께 이용하면 좋은 라이브러리들

세종 말뭉치 정제를 위한 utils

자연어처리 모델 학습을 위하여 세종 말뭉치 데이터를 정제하기 위한 함수들을 제공합니다. 형태소/품사 형태로 정제된 학습용 데이터를 만드는 함수, 용언의 활용 형태를 정리하여 테이블로 만드는 함수, 세종 말뭉치의 품사 체계를 단순화 시키는 함수를 제공합니다.

soyspacing

띄어쓰기 오류가 있을 경우 이를 제거하면 텍스트 분석이 쉬워질 수 있습니다. 분석하려는 데이터를 기반으로 띄어쓰기 엔진을 학습하고, 이를 이용하여 띄어쓰기 오류를 교정합니다.

KR-WordRank

토크나이저나 단어 추출기를 학습할 필요없이, HITS algorithm 을 이용하여 substring graph 에서 키워드를 추출합니다.

soykeyword

키워드 추출기입니다. Logistic Regression 을 이용하는 모델과 통계 기반 모델, 두 종류의 키워드 추출기를 제공합니다. scipy.sparse 의 sparse matrix 형식과 텍스트 파일 형식을 지원합니다.

Analytics

soynlp's People

Contributors

chulphan avatar epsilon-deltta avatar jeongukjae avatar lovit avatar ming99999 avatar owlur avatar parksjin01 avatar sagnjun-nam avatar skyer9 avatar yonghyunryu avatar ywkim avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

soynlp's Issues

그랫 -> 그래, 알았엇 -> 알았어

  • 대화체에서 어절의 마지막에 불필요한 받침을 넣는 경우가 있음
  • 특정 패턴이므로 통계적으로 해결할 수 있을 것으로 예상

카메라는 청년은 카메라를 좋아한다

  • tagger 에서 최장일치법을 이용하여 remove_subset_l 을 수행하면 [카메, 카메라] 에서 '카메'가 후보에서 제외됨
  • 후보에서 제외할 수 있는 조건은 '카메' 옆에 등장하는 글자가 R set 이 아닌 경우로 제한해야 함

LRNounExtractor_v2 오류

아래와 같은 에러가 발생하는 군요..

INFO] 2018-11-03 14:17:24.884 noun_data = noun_extractor.train_extract(text)
[INFO] 2018-11-03 14:17:24.884 File "/Users/ikarroo/anaconda3/envs/batch/lib/python3.6/site-packages/soynlp/noun/noun_ver2.py", line 142, in train_extract
[INFO] 2018-11-03 14:17:24.884 self.train(sentences)
[INFO] 2018-11-03 14:17:24.884 File "/Users/ikarroo/anaconda3/envs/batch/lib/python3.6/site-packages/soynlp/noun/noun_ver2.py", line 154, in train
[INFO] 2018-11-03 14:17:24.885 verbose=self.verbose)
[INFO] 2018-11-03 14:17:24.885 File "/Users/ikarroo/anaconda3/envs/batch/lib/python3.6/site-packages/soynlp/utils/utils.py", line 143, in__init

[INFO] 2018-11-03 14:17:24.885 if sents:
[INFO] 2018-11-03 14:17:24.885 File "/Users/ikarroo/anaconda3/envs/batch/lib/python3.6/site-packages/pandas/core/generic.py", line 1576, in nonzero
[INFO] 2018-11-03 14:17:24.886 .format(self.class.name))
[INFO] 2018-11-03 14:17:24.886 ValueError: The truth value of a Series is ambiguous. Use a.empty, a.bool(), a.item(), a.any() or a.all().

RegexTokenizer 숫자 (3.1.1)

  • 3.1 처럼 소수점 형식은 제대로 나뉘어지지만 3.1.1 은 제대로 나뉘어지지 않음

from soynlp.tokenizer import RegexTokenizer

regex_tokenizer = RegexTokenizer()
regex_tokenizer.tokenize('다음에는3.1.1장입니다')

['다음에는', '3.1', '.1', '장입니다']

명사/용언 추출기 용 LR graph 생성 시 기호 제거

  • "잘들어갔어요?" 라는 어절은 명사/용언 추출용 LR graph 에 적절하지 않음
    • 이 어절에서 기호를 제거하여 "잘들어갔어요"로 라는 어절을 만든 뒤 LR graph 를 만들어야 함
    • 어절의 마지막 글자는 한글이어야 하며, 어절의 첫글자는 영어/숫자/한글/자음/모음이 되도록 normalization 을 하는 기능을 optional 하게 제공

PredicatorExtractor 에서 '김병기'가 용언으로 추출됨

$ noun_extractor.predict('김병기', debug=True)

pos=0, common=0, neg=0, unk=0, end=152, n_features_=0

$ noun_extractor.lrgraph.get_r('김병')

[('기', 152),
('준', 49),
('욱', 48),
('우', 13),
('호', 12),
('관', 12),
('상', 11),
('진', 11),
('수', 8),
('희', 8)]

'-기' 가 대표적인 어미여서 PredicatorExtractor 가 '김병'을 stem 으로 인식함

'미스터' 가 명사로 뽑히지 않는 Case

뉴스 데이터를 전처리하여 명사 추출에 사용하고 있습니다.
최근 드라마 '미스터 션샤인' 이 자주 등장해서 '미스터'와 '션샤인'이 당연히 명사로 잡힐 줄 알았는데
NounExtractor_v2를 사용하여 명사 스코어링을 해 보니 '미스터'가 명사로 잡히질 않았습니다.
LTokenizer를 사용했을 때 '미스터 션샤인'이 [미스, 터, 션샤인]으로 나뉘더라고요

'미스'는 잘 잡히는 것 처럼 '미스터'도 명사로 잘 잡고 싶습니다.
방법이 없을까요 ㅠ?

lemmatize('쓰러진') = '쓰러지 + ㄴ' 만 남도록 필터링

'쓰러진'의 어간에 '쓰리다'가 포함되어 있음

  • _lemma_candidate('쓰러진', '') -> {('쓰러지', 'ㄴ'), ('쓰러진', '')}
  • _lemma_candidate('쓰러', '진') -> {('쓰러', '진'), ('쓰렇', '진'), ('쓰르', '어진'), ('쓰리', '어진')}
  • _lemma_candidate('쓰려', '진') -> {('쓰려', '진'), ('쓰리', '어진')}

LRNounExtractor_v2 에서 '관계자' 가 명사로 추출되지 않음

$ noun_extractor.predict('관계자', debug=True)

pos=339, common=3, neg=2942, unk=3, end=258, n_features_=13

$ '는' in noun_extractor._pos_features

True

$ '자는' in noun_extractor._neg_features

True

longer negative features 가 존재할 경우 negative 로 counting 하기 때문에 '-자'로 끝나는 명사 중 '-는'과 결합되는 경우가 많은 명사는 명사가 아닌 것으로 추출됨.

class LRNounExtractor_v2:

    def _predict(self, word, features):
        ...
        for r, freq in features:
            ...
            if self._exist_longer_neg(word, r): # negative -다고
                neg += freq

동사 '조아 (좋아)' 인식

'좋다'의 활용형태는 어미 추출을 통하여 인식할 수 있지만, '조아'는 인식할 수가 없다. '조 + 다'라는 동사를 추가하는 것도 옳지 않은데, 받침이 생략되는 빈번한 규칙을 이용하여, '조아'의 원형태가 '좋아'임을 추정할 수 있어야 한다.

Noun Extractor 관련 질문입니다.

안녕하세요. soynlp를 만들어주셔서 감사합니다.
초보 프로그래머라 질문 드립니다ㅜ

corpus_fname = 'YOUR_DoubleSpaceLine_Corpus'
sentences = DoublespaceLineCorpus(corpus_fname, iter_sent=True)
len(sentences)

이부분에서 'YOUR_DoubleSpaceLine_Corpus' 이부분이 이해가 가지 않습니다.
디렉토리 내에 있는 txt파일명을 넣어야 하는건지
아니면 그 파일을 불러와서 list 형태로 만든 변수명을 넣어야하는건지...

감사합니다ㅜ

README 의 tutorials 링크 수정

README 의 tutorials 들의 링크가 상대위치로 작성되다보니 PyPI 에서 해당 링크들이 잘못 잡힘. github url 의 절대 주소로 이를 대체해야 함

(믿음, 먹음, 마음) + 으로

  • '-음, -ㅁ' 은 용언을 명사로 전성하는 명사형전성어미.
  • '음' 으로 시작하는 negative feature 를 제거하여 '마음'처럼 -음으로 끝나는 명사가 추출될 수 있도록 해야 함

LRNounExtractor_v2 후처리 함수

후처리 과정에서 걸러지는 단어 예시

  • 확실한 경우 : '이상으로'
  • 문맥에 따라 다른 경우 : '최고가', '만족도'

'최고가', '만족도'는 문맥에 따라서 true 일수도 false 일수도 있으나, base postprocessor 로는 이를 구분할 수 없음.

금액, 기간, 날짜, 시간, 숫자, 개수 인식기

  • 금액: 28조원, 숫자+단위
  • 기간: 15년간, 매년간, 5개월동안
  • 날짜: 2018-08-12, 15년 3월 4일, 지난주, 이번주, 다음주, [월 ~ 일]요일, [첫,둘째,셋째,,,]날
  • 시간: 3시5분, 17시25분2초
  • 숫자: 383,1215

lemmatizer 예외

  • 르 불규칙 예외
    • (치르다', '따르다', '다다르다', '우러르다', '들르다)
    • 들르 + 어 -> 들러, 다다르 + 어 -> 다다러

나+N

  • 대화데이터처럼 띄어쓰기가 잘 지켜지지 않는 데이터에서는, 일종의 접두사처럼 이용되는 글자들이 있다.
  • 나사당, 나학교 처럼 일종의 관형사 (determiner) 처럼 이용되는 '나-'를 제거해야 한다.

lemmatize('만져', '라구')

from soynlp.lemmatize import _lemma_candidate

_lemma_candidate('만졌', '던') # {('만져', 'ㅆ던'), ('만졌', '던'), ('만졓', 'ㅆ던'), ('만지', '었던')}
_lemma_candidate('만져', '라고') # {('만져', '라고'), ('만졓', '라고')}

어간의 표현형의 마지막 글자의 모음이 ㅕ 일 때, 종성 ㅆ 이 있으면 '이 + 었' 으로 어미가 분리되지만, 마지막 글자의 종성이 없으면 '이 + 었'으로 분리되지 않음

conjugate('파랗', '자') -> '파래' 를 '파랗자'로 수정

형용사와 동사를 구분하는 방법 중 하나로, 어간이 청유형으로 활용되는지 확인할 수 있다. 그런데 '파랗다'의 청유형 활용 형태가 '파래' 처럼 ㅎ 불규칙이 잘못 적용된다.

ㅎ 불규칙은 어미의 초성이 'ㅇ' 으로 시작할 때 적용되어야 한다.

먹 + 어라 (o) vs 먹 + 아라 (x)

  • 어간의 마지막 종성이 ㅓ 일 때에만 -어라 와 결합 가능하며,
  • 파랗/먹/그/기 + 아라 만 가능
    • 파랗 + 아라 = 파랗거라, 파래라 로 활용되지만, 형용사이기 때문에 문법적으로 옳지는 않음

normalizer exception

  • "앜ㅋㅋㅋㅋ"는 "아ㅋㅋㅋㅋㅋ" 가 되지 않음
  • 'ㅋㅋㅋ쿠'는 'ㅋㅋㅋㅋㅜ'가 되지 않음
  • (자음, 완전글자, 모음) 외에도 (완전글자, 자음), (자음, 완전글자) 패턴에 대해서도 정규화를 수행해야 함

'돕다' 활용

현재 상태

  • _lemma_candidate('도', '울까') = {('도', '울까'), ('돗', '울까')}
  • conjugate('돕', '울까') = '돕울까'

Python 2.7 설치

Soynlp를 pip 9.0.1 from /Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/site-packages (python 2.7) 가지고 설치를 시도했더니 description 모듈을 인식하지 못하는 문제가 발생해서 설치가 제대로 진행되지 않습니다. 혹시 해결해주실수 있나요?

conjugate('먹', '자') -> '먹저' 를 '먹자'로 수정

모음조화 현상을 반영하기 위하여 어간의 마지막 중성과 어미의 첫 중성의 모음조화를 확인한다. 이 때 어미의 첫 초성이 'ㅇ' 이 아니어도 모음조화를 맞추기 때문에 '먹 + 자'가 '먹저'로 활용된다.

모음조화는 어미의 첫 초성이 'ㅇ' 일 때로 한정한다

어간의 마지막 종성이 없을 경우, '-는, -을' 이 아닌 -ㄴ, -ㄹ'만 올 수 있어야 한다

  • '-는다'는 어간의 마지막 종성이 있을 경우, -ㄴ다'는 어간의 마지막 종성이 없을 경우 활용 가능하다.
    • 먹 + 는다 -> 먹는다 // 가 + ㄴ다 -> 간다
  • 현재 '먹 + ㄴ다'는 conjugate 가 되지 않지만, '가 + 는다 = 가는다' 가 되고 있음.
  • 어간의 마지막 종성이 없고 어미의 첫 글자의 초성이 ㅇ 이거나 초/종성이 같고 모음이 ㅏ, ㅡ 이면 conjugate 가 되지 않도록 해야 함

이대 + 는, 이대 + 로

  • substring '이대'의 문맥이 다르다.
  • '이대로'가 '이대 + 는' 보다 자주 등장하기 때문에 '이대'는 명사가 아닌 것으로 측정된다.
  • substring 의 문맥적 모호성을 해결해야 한다.
    • noun score < threshold 인 subword 중에서 최소 몇 개 이상의 조사, pos features 가 있다면 tokenizer 입장에서 (불안하지만) 명사로 잘라내야 한다.

LRNounExtractor_v2가 어미를 잘 제거하지 못하는 문제가 있습니다.

LRNounExtractor와 LRNounExtractor_v2에 의한 명사 추출 결과를 비교해 봤는데, 원래 버전(1.0)에서는 '하다', '하게', '하기', '하고', '하려' 등의 어미를 제거한 명사 part가 잘 추출되는 반면, v2에서는 어미가 붙어서 명사로 추출되는 경우가 다수 발견됩니다. 원래 의도대로 잘 구현된 것인지 검토가 필요한 듯 합니다.

soynlp Version 0.0.47 pip으로 설치시, import 에러

PyPI에 올라와있는 Soynlp 버전 0.0.47을 pip으로 설치시, import soynlp에서 FileNotFoundError가 발생합니다.

FileNotFoundError: [Errno 2] No such file or directory: '/home/shuuki4/venv_tensorflow/lib/python3.5/site-packages/soynlp/noun/frequent_enrolled_josa.txt'

PyPI에 있는 0.0.47 버전 tar.gz 파일을 확인해보니, 해당 파일 자체가 누락되어 있는 것 같습니다.

0.0.49버전에서 scipy import 에러

soynlp/word/_pmi.py, soynlp/vectorizer/_word_context.py, soynlp/vectorizer/_vectorizer.py의 코드에 scipy를 import하는 것을 확인하였으나, README.md, setup.py에 추가되지 않아 있는 것으로 보아 누락되어 있는 것으로 보입니다.

lemmatize(간지러)

  • '-럽다'로 끝나는 용언 중에서 '-워'가 생략되는 경우가 있음.
  • 간지럽다 + 워 -> 간지러워 -> (워 생략) -> 간지러

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.