* Left-Arc from Root, to allow non-monotonic reduce to compete with left-arc when the stack is not empty.

This commit is contained in:
Matthew Honnibal 2015-03-27 17:39:16 +01:00
parent db5a43318c
commit a3af6b7c3d
3 changed files with 14 additions and 8 deletions

View File

@ -97,7 +97,7 @@ cdef inline bint at_eol(const State *s) nogil:
cdef inline bint is_final(const State *s) nogil: cdef inline bint is_final(const State *s) nogil:
return at_eol(s) # The stack will be attached to root anyway return at_eol(s) and s.stack_len < 2
cdef int children_in_buffer(const State *s, const int head, const int* gold) except -1 cdef int children_in_buffer(const State *s, const int head, const int* gold) except -1

View File

@ -35,11 +35,6 @@ cdef int push_stack(State *s) except -1:
s.stack[0] = s.i s.stack[0] = s.i
s.stack_len += 1 s.stack_len += 1
s.i += 1 s.i += 1
if at_eol(s):
while s.stack_len != 0:
if not has_head(get_s0(s)):
get_s0(s).dep = 0
pop_stack(s)
cdef int children_in_buffer(const State *s, int head, const int* gold) except -1: cdef int children_in_buffer(const State *s, int head, const int* gold) except -1:

View File

@ -43,7 +43,7 @@ cdef class ArcEager(TransitionSystem):
@classmethod @classmethod
def get_labels(cls, gold_parses): def get_labels(cls, gold_parses):
move_labels = {SHIFT: {'': True}, REDUCE: {'': True}, RIGHT: {}, move_labels = {SHIFT: {'': True}, REDUCE: {'': True}, RIGHT: {},
LEFT: {}, BREAK: {'ROOT': True}} LEFT: {'ROOT': True}, BREAK: {'ROOT': True}}
for raw_text, segmented, (ids, words, tags, heads, labels, iob) in gold_parses: for raw_text, segmented, (ids, words, tags, heads, labels, iob) in gold_parses:
for child, head, label in zip(ids, heads, labels): for child, head, label in zip(ids, heads, labels):
if label != 'ROOT': if label != 'ROOT':
@ -126,6 +126,10 @@ cdef int _do_shift(const Transition* self, State* state) except -1:
cdef int _do_left(const Transition* self, State* state) except -1: cdef int _do_left(const Transition* self, State* state) except -1:
# Interpret left-arcs from EOL as attachment to root
if at_eol(state):
add_dep(state, state.stack[0], state.stack[0], self.label)
else:
add_dep(state, state.i, state.stack[0], self.label) add_dep(state, state.i, state.stack[0], self.label)
pop_stack(state) pop_stack(state)
@ -195,6 +199,13 @@ cdef int _left_cost(const Transition* self, const State* s, GoldParse gold) exce
if gold.c_heads[s.stack[0]] == s.i: if gold.c_heads[s.stack[0]] == s.i:
cost += self.label != gold.c_labels[s.stack[0]] cost += self.label != gold.c_labels[s.stack[0]]
return cost return cost
# If we're at EOL, then the left arc will add an arc to ROOT.
elif at_eol(s):
# Are we root?
cost += gold.c_heads[s.stack[0]] != s.stack[0]
# Are we labelling correctly?
cost += self.label != gold.c_labels[s.stack[0]]
return cost
cost += head_in_buffer(s, s.stack[0], gold.c_heads) cost += head_in_buffer(s, s.stack[0], gold.c_heads)
cost += children_in_buffer(s, s.stack[0], gold.c_heads) cost += children_in_buffer(s, s.stack[0], gold.c_heads)