Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
PaddlePaddle
PaddleDetection
提交
8818c94c
P
PaddleDetection
项目概览
PaddlePaddle
/
PaddleDetection
1 年多 前同步成功
通知
696
Star
11112
Fork
2696
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
184
列表
看板
标记
里程碑
合并请求
40
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
P
PaddleDetection
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
184
Issue
184
列表
看板
标记
里程碑
合并请求
40
合并请求
40
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
未验证
提交
8818c94c
编写于
5月 24, 2019
作者:
P
pkpk
提交者:
GitHub
5月 24, 2019
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
test=develop (#17207)
上级
ca03f498
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
229 addition
and
92 deletion
+229
-92
python/paddle/fluid/metrics.py
python/paddle/fluid/metrics.py
+229
-92
未找到文件。
python/paddle/fluid/metrics.py
浏览文件 @
8818c94c
...
@@ -153,20 +153,25 @@ class CompositeMetric(MetricBase):
...
@@ -153,20 +153,25 @@ class CompositeMetric(MetricBase):
Examples:
Examples:
.. code-block:: python
.. code-block:: python
labels = fluid.layers.data(name="data", shape=[1], dtype="int32")
import numpy as np
data = fluid.layers.data(name="data", shape=[32, 32], dtype="int32")
preds = [[0.1], [0.7], [0.8], [0.9], [0.2],
pred = fluid.layers.fc(input=data, size=1000, act="tanh")
[0.2], [0.3], [0.5], [0.8], [0.6]]
comp = fluid.metrics.CompositeMetric()
labels = [[0], [1], [1], [1], [1],
acc = fluid.metrics.Precision()
[0], [0], [0], [0], [0]]
recall = fluid.metrics.Recall()
preds = np.array(preds)
comp.add_metric(acc)
labels = np.array(labels)
comp.add_metric(recall)
for pass in range(PASSES):
comp = fluid.metrics.CompositeMetric()
comp.reset()
precision = fluid.metrics.Precision()
for data in train_reader():
recall = fluid.metrics.Recall()
loss, preds, labels = exe.run(fetch_list=[cost, preds, labels])
comp.add_metric(precision)
comp.add_metric(recall)
comp.update(preds=preds, labels=labels)
comp.update(preds=preds, labels=labels)
numpy_acc, numpy_recall = comp.eval()
numpy_precision, numpy_recall = comp.eval()
print("expect precision: %.2f, got %.2f" % ( 3. / 5, numpy_precision ) )
print("expect recall: %.2f, got %.2f" % (3. / 4, numpy_recall ) )
"""
"""
def
__init__
(
self
,
name
=
None
):
def
__init__
(
self
,
name
=
None
):
...
@@ -215,20 +220,30 @@ class Precision(MetricBase):
...
@@ -215,20 +220,30 @@ class Precision(MetricBase):
relevant instances among the retrieved instances.
relevant instances among the retrieved instances.
https://en.wikipedia.org/wiki/Evaluation_of_binary_classifiers
https://en.wikipedia.org/wiki/Evaluation_of_binary_classifiers
Note Precision is different with Accuracy in binary classifiers.
This class mangages the precision score for binary classification task.
accuracy = true positive / total instances
precision = true positive / all positive instance
Examples:
Examples:
.. code-block:: python
.. code-block:: python
import numpy as np
metric = fluid.metrics.Precision()
metric = fluid.metrics.Precision()
for pass in range(PASSES):
metric.reset()
# generate the preds and labels
for data in train_reader():
loss, preds, labels = exe.run(fetch_list=[cost, preds, labels])
preds = [[0.1], [0.7], [0.8], [0.9], [0.2],
metric.update(preds=preds, labels=labels)
[0.2], [0.3], [0.5], [0.8], [0.6]]
numpy_precision = metric.eval()
labels = [[0], [1], [1], [1], [1],
[0], [0], [0], [0], [0]]
preds = np.array(preds)
labels = np.array(labels)
metric.update(preds=preds, labels=labels)
numpy_precision = metric.eval()
print("expct precision: %.2f and got %.2f" % ( 3.0 / 5.0, numpy_precision))
"""
"""
def
__init__
(
self
,
name
=
None
):
def
__init__
(
self
,
name
=
None
):
...
@@ -247,7 +262,7 @@ class Precision(MetricBase):
...
@@ -247,7 +262,7 @@ class Precision(MetricBase):
for
i
in
range
(
sample_num
):
for
i
in
range
(
sample_num
):
pred
=
preds
[
i
]
pred
=
preds
[
i
]
label
=
labels
[
i
]
label
=
labels
[
i
]
if
label
==
1
:
if
pred
==
1
:
if
pred
==
label
:
if
pred
==
label
:
self
.
tp
+=
1
self
.
tp
+=
1
else
:
else
:
...
@@ -266,16 +281,30 @@ class Recall(MetricBase):
...
@@ -266,16 +281,30 @@ class Recall(MetricBase):
https://en.wikipedia.org/wiki/Precision_and_recall
https://en.wikipedia.org/wiki/Precision_and_recall
This class mangages the recall score for binary classification task.
Examples:
Examples:
.. code-block:: python
.. code-block:: python
import numpy as np
metric = fluid.metrics.Recall()
metric = fluid.metrics.Recall()
for pass in range(PASSES):
metric.reset()
# generate the preds and labels
for data in train_reader():
loss, preds, labels = exe.run(fetch_list=[cost, preds, labels])
preds = [[0.1], [0.7], [0.8], [0.9], [0.2],
metric.update(preds=preds, labels=labels)
[0.2], [0.3], [0.5], [0.8], [0.6]]
numpy_recall = metric.eval()
labels = [[0], [1], [1], [1], [1],
[0], [0], [0], [0], [0]]
preds = np.array(preds)
labels = np.array(labels)
metric.update(preds=preds, labels=labels)
numpy_precision = metric.eval()
print("expct precision: %.2f and got %.2f" % ( 3.0 / 4.0, numpy_precision))
"""
"""
def
__init__
(
self
,
name
=
None
):
def
__init__
(
self
,
name
=
None
):
...
@@ -288,15 +317,16 @@ class Recall(MetricBase):
...
@@ -288,15 +317,16 @@ class Recall(MetricBase):
raise
ValueError
(
"The 'preds' must be a numpy ndarray."
)
raise
ValueError
(
"The 'preds' must be a numpy ndarray."
)
if
not
_is_numpy_
(
labels
):
if
not
_is_numpy_
(
labels
):
raise
ValueError
(
"The 'labels' must be a numpy ndarray."
)
raise
ValueError
(
"The 'labels' must be a numpy ndarray."
)
sample_num
=
labels
[
0
]
sample_num
=
labels
.
shape
[
0
]
preds
=
np
.
rint
(
preds
).
astype
(
"int32"
)
for
i
in
range
(
sample_num
):
for
i
in
range
(
sample_num
):
pred
=
preds
[
i
]
.
astype
(
"int32"
)
pred
=
preds
[
i
]
label
=
labels
[
i
]
label
=
labels
[
i
]
if
label
==
1
:
if
label
==
1
:
if
pred
==
label
:
if
pred
==
label
:
self
.
tp
+=
1
self
.
tp
+=
1
else
:
else
:
if
pred
!=
label
:
self
.
fn
+=
1
self
.
fn
+=
1
def
eval
(
self
):
def
eval
(
self
):
...
@@ -306,8 +336,7 @@ class Recall(MetricBase):
...
@@ -306,8 +336,7 @@ class Recall(MetricBase):
class
Accuracy
(
MetricBase
):
class
Accuracy
(
MetricBase
):
"""
"""
Accumulate the accuracy from minibatches and compute the average accuracy
Calculate the mean accuracy over multiple batches.
for every pass.
https://en.wikipedia.org/wiki/Accuracy_and_precision
https://en.wikipedia.org/wiki/Accuracy_and_precision
Args:
Args:
...
@@ -316,18 +345,28 @@ class Accuracy(MetricBase):
...
@@ -316,18 +345,28 @@ class Accuracy(MetricBase):
Examples:
Examples:
.. code-block:: python
.. code-block:: python
labels = fluid.layers.data(name="data", shape=[1], dtype="int32")
#suppose we have batch_size = 128
data = fluid.layers.data(name="data", shape=[32, 32], dtype="int32")
batch_size=128
pred = fluid.layers.fc(input=data, size=1000, act="tanh")
accuracy_manager = fluid.metrics.Accuracy()
minibatch_accuracy = fluid.layers.accuracy(pred, label)
accuracy_evaluator = fluid.metrics.Accuracy()
#suppose the accuracy is 0.9 for the 1st batch
for pass in range(PASSES):
batch1_acc = 0.9
accuracy_evaluator.reset()
accuracy_manager.update(value = batch1_acc, weight = batch_size)
for data in train_reader():
print("expect accuracy: %.2f, get accuracy: %.2f" % (batch1_acc, accuracy_manager.eval()))
batch_size = data[0]
loss = exe.run(fetch_list=[cost, minibatch_accuracy])
#suppose the accuracy is 0.8 for the 2nd batch
accuracy_evaluator.update(value=minibatch_accuracy, weight=batch_size)
batch2_acc = 0.8
numpy_acc = accuracy_evaluator.eval()
accuracy_manager.update(value = batch2_acc, weight = batch_size)
#the joint acc for batch1 and batch2 is (batch1_acc * batch_size + batch2_acc * batch_size) / batch_size / 2
print("expect accuracy: %.2f, get accuracy: %.2f" % ((batch1_acc * batch_size + batch2_acc * batch_size) / batch_size / 2, accuracy_manager.eval()))
#reset the accuracy_manager
accuracy_manager.reset()
#suppose the accuracy is 0.8 for the 3rd batch
batch3_acc = 0.8
accuracy_manager.update(value = batch3_acc, weight = batch_size)
print("expect accuracy: %.2f, get accuracy: %.2f" % (batch3_acc, accuracy_manager.eval()))
"""
"""
def
__init__
(
self
,
name
=
None
):
def
__init__
(
self
,
name
=
None
):
...
@@ -348,10 +387,15 @@ class Accuracy(MetricBase):
...
@@ -348,10 +387,15 @@ class Accuracy(MetricBase):
"The 'value' must be a number(int, float) or a numpy ndarray."
)
"The 'value' must be a number(int, float) or a numpy ndarray."
)
if
not
_is_number_
(
weight
):
if
not
_is_number_
(
weight
):
raise
ValueError
(
"The 'weight' must be a number(int, float)."
)
raise
ValueError
(
"The 'weight' must be a number(int, float)."
)
if
_is_number_
(
weight
)
and
weight
<
0
:
raise
ValueError
(
"The 'weight' can not be negative"
)
self
.
value
+=
value
*
weight
self
.
value
+=
value
*
weight
self
.
weight
+=
weight
self
.
weight
+=
weight
def
eval
(
self
):
def
eval
(
self
):
"""
Return the mean accuracy (float or numpy.array) for all accumulated batches.
"""
if
self
.
weight
==
0
:
if
self
.
weight
==
0
:
raise
ValueError
(
"There is no data in Accuracy Metrics.
\
raise
ValueError
(
"There is no data in Accuracy Metrics.
\
Please check layers.accuracy output has added to Accuracy."
)
Please check layers.accuracy output has added to Accuracy."
)
...
@@ -371,17 +415,29 @@ class ChunkEvaluator(MetricBase):
...
@@ -371,17 +415,29 @@ class ChunkEvaluator(MetricBase):
Examples:
Examples:
.. code-block:: python
.. code-block:: python
labels = fluid.layers.data(name="data", shape=[1], dtype="int32")
# init the chunck-level evaluation manager
data = fluid.layers.data(name="data", shape=[32, 32], dtype="int32")
pred = fluid.layers.fc(input=data, size=1000, act="tanh")
precision, recall, f1_score, num_infer_chunks, num_label_chunks, num_correct_chunks = layers.chunk_eval(
input=pred,
label=label)
metric = fluid.metrics.ChunkEvaluator()
metric = fluid.metrics.ChunkEvaluator()
for data in train_reader():
loss, preds, labels = exe.run(fetch_list=[cost, preds, labels])
# suppose the model predict 10 chuncks, while 8 ones are correct and the ground truth has 9 chuncks.
metric.update(num_infer_chunks, num_label_chunks, num_correct_chunks)
num_infer_chunks = 10
numpy_precision, numpy_recall, numpy_f1 = metric.eval()
num_label_chunks = 9
num_correct_chunks = 8
metric.update(num_infer_chunks, num_label_chunks, num_correct_chunks)
numpy_precision, numpy_recall, numpy_f1 = metric.eval()
print("precision: %.2f, recall: %.2f, f1: %.2f" % (numpy_precision, numpy_recall, numpy_f1))
# the next batch, predicting 3 prefectly correct chuncks.
num_infer_chunks = 3
num_label_chunks = 3
num_correct_chunks = 3
metric.update(num_infer_chunks, num_label_chunks, num_correct_chunks)
numpy_precision, numpy_recall, numpy_f1 = metric.eval()
print("precision: %.2f, recall: %.2f, f1: %.2f" % (numpy_precision, numpy_recall, numpy_f1))
"""
"""
def
__init__
(
self
,
name
=
None
):
def
__init__
(
self
,
name
=
None
):
...
@@ -430,12 +486,17 @@ class ChunkEvaluator(MetricBase):
...
@@ -430,12 +486,17 @@ class ChunkEvaluator(MetricBase):
class
EditDistance
(
MetricBase
):
class
EditDistance
(
MetricBase
):
"""
"""
Edit distance is a way of quantifying how dissimilar two strings
Edit distance is a way of quantifying how dissimilar two strings
(e.g., words) are to one another by counting the minimum number
(e.g., words) are to each another by counting the minimum number
of operations required to transform one string into the other.
of edit operations (add, remove or replace) required to transform
one string into the other.
Refer to https://en.wikipedia.org/wiki/Edit_distance
Refer to https://en.wikipedia.org/wiki/Edit_distance
Accumulate edit distance sum and sequence number from mini-batches and
This EditDistance class takes two inputs by using update function:
compute the average edit_distance and instance error of all batches.
1. distances: a (batch_size, 1) numpy.array, each element represents the
edit distance between two sequences.
2. seq_num: a int|float value, standing for the number of sequence pairs.
and returns the overall edit distance of multiple sequence-pairs.
Args:
Args:
name: the metrics name
name: the metrics name
...
@@ -443,19 +504,37 @@ class EditDistance(MetricBase):
...
@@ -443,19 +504,37 @@ class EditDistance(MetricBase):
Examples:
Examples:
.. code-block:: python
.. code-block:: python
distances, seq_num = fluid.layers.edit_distance(input, label)
import numpy as np
distance_evaluator = fluid.metrics.EditDistance()
for epoch in PASS_NUM:
# suppose that batch_size is 128
distance_evaluator.reset()
batch_size = 128
for data in batches:
loss = exe.run(fetch_list=[cost] + list(edit_distance_metrics))
# init the edit distance manager
distance_evaluator.update(distances, seq_num)
distance_evaluator = fluid.metrics.EditDistance("EditDistance")
distance, instance_error = distance_evaluator.eval()
# generate the edit distance across 128 sequence pairs, the max distance is 10 here
edit_distances_batch0 = np.random.randint(low = 0, high = 10, size = (batch_size, 1))
seq_num_batch0 = batch_size
distance_evaluator.update(edit_distances_batch0, seq_num_batch0)
avg_distance, wrong_instance_ratio = distance_evaluator.eval()
print("the average edit distance for batch0 is %.2f and the wrong instance ratio is %.2f " % (avg_distance, wrong_instance_ratio))
In the above example:
edit_distances_batch1 = np.random.randint(low = 0, high = 10, size = (batch_size, 1))
seq_num_batch1 = batch_size
- 'distance' is the average of the edit distance in a pass.
distance_evaluator.update(edit_distances_batch1, seq_num_batch1)
- 'instance_error' is the instance error rate in a pass.
avg_distance, wrong_instance_ratio = distance_evaluator.eval()
print("the average edit distance for batch0 and batch1 is %.2f and the wrong instance ratio is %.2f " % (avg_distance, wrong_instance_ratio))
distance_evaluator.reset()
edit_distances_batch2 = np.random.randint(low = 0, high = 10, size = (batch_size, 1))
seq_num_batch2 = batch_size
distance_evaluator.update(edit_distances_batch2, seq_num_batch2)
avg_distance, wrong_instance_ratio = distance_evaluator.eval()
print("the average edit distance for batch2 is %.2f and the wrong instance ratio is %.2f " % (avg_distance, wrong_instance_ratio))
"""
"""
...
@@ -466,6 +545,15 @@ class EditDistance(MetricBase):
...
@@ -466,6 +545,15 @@ class EditDistance(MetricBase):
self
.
instance_error
=
0
self
.
instance_error
=
0
def
update
(
self
,
distances
,
seq_num
):
def
update
(
self
,
distances
,
seq_num
):
"""
Update the overall edit distance
Args:
distances: a (batch_size, 1) numpy.array, each element represents the
edit distance between two sequences.
seq_num: a int|float value, standing for the number of sequence pairs.
"""
if
not
_is_numpy_
(
distances
):
if
not
_is_numpy_
(
distances
):
raise
ValueError
(
"The 'distances' must be a numpy ndarray."
)
raise
ValueError
(
"The 'distances' must be a numpy ndarray."
)
if
not
_is_number_
(
seq_num
):
if
not
_is_number_
(
seq_num
):
...
@@ -477,6 +565,11 @@ class EditDistance(MetricBase):
...
@@ -477,6 +565,11 @@ class EditDistance(MetricBase):
self
.
total_distance
+=
total_distance
self
.
total_distance
+=
total_distance
def
eval
(
self
):
def
eval
(
self
):
"""
Return two floats:
avg_distance: the average distance for all sequence pairs updated using the update function.
avg_instance_error: the ratio of sequence pairs whose edit distance is not zero.
"""
if
self
.
seq_num
==
0
:
if
self
.
seq_num
==
0
:
raise
ValueError
(
raise
ValueError
(
"There is no data in EditDistance Metric. Please check layers.edit_distance output has been added to EditDistance."
"There is no data in EditDistance Metric. Please check layers.edit_distance output has been added to EditDistance."
...
@@ -488,9 +581,9 @@ class EditDistance(MetricBase):
...
@@ -488,9 +581,9 @@ class EditDistance(MetricBase):
class
Auc
(
MetricBase
):
class
Auc
(
MetricBase
):
"""
"""
Auc metric adapts to the
binary classification.
The auc metric is for
binary classification.
Refer to https://en.wikipedia.org/wiki/Receiver_operating_characteristic#Area_under_the_curve
Refer to https://en.wikipedia.org/wiki/Receiver_operating_characteristic#Area_under_the_curve
Need to note that auc metric compute the value via Python natively
.
Please notice that the auc metric is implemented with python, which may be a little bit slow
.
If you concern the speed, please use the fluid.layers.auc instead.
If you concern the speed, please use the fluid.layers.auc instead.
The `auc` function creates four local variables, `true_positives`,
The `auc` function creates four local variables, `true_positives`,
...
@@ -511,12 +604,26 @@ class Auc(MetricBase):
...
@@ -511,12 +604,26 @@ class Auc(MetricBase):
Examples:
Examples:
.. code-block:: python
.. code-block:: python
pred = fluid.layers.fc(input=data, size=1000, act="tanh")
import numpy as np
metric = fluid.metrics.Auc()
# init the auc metric
for data in train_reader():
auc_metric = fluid.metrics.Auc("ROC")
loss, preds, labels = exe.run(fetch_list=[cost, preds, labels])
metric.update(preds, labels)
# suppose that batch_size is 128
numpy_auc = metric.eval()
batch_num = 100
batch_size = 128
for batch_id in range(batch_num):
class0_preds = np.random.random(size = (batch_size, 1))
class1_preds = 1 - class0_preds
preds = np.concatenate((class0_preds, class1_preds), axis=1)
labels = np.random.randint(2, size = (batch_size, 1))
auc_metric.update(preds = preds, labels = labels)
# shall be some score closing to 0.5 as the preds are randomly assigned
print("auc for iteration %d is %.2f" % (batch_id, auc_metric.eval()))
"""
"""
def
__init__
(
self
,
name
,
curve
=
'ROC'
,
num_thresholds
=
4095
):
def
__init__
(
self
,
name
,
curve
=
'ROC'
,
num_thresholds
=
4095
):
...
@@ -529,6 +636,15 @@ class Auc(MetricBase):
...
@@ -529,6 +636,15 @@ class Auc(MetricBase):
self
.
_stat_neg
=
[
0
]
*
_num_pred_buckets
self
.
_stat_neg
=
[
0
]
*
_num_pred_buckets
def
update
(
self
,
preds
,
labels
):
def
update
(
self
,
preds
,
labels
):
"""
Update the auc curve with the given predictions and labels
Args:
preds: an numpy array in the shape of (batch_size, 2), preds[i][j] denotes the probability
of classifying the instance i into the class j.
labels: an numpy array in the shape of (batch_size, 1), labels[i] is either o or 1, representing
the label of the instance i.
"""
if
not
_is_numpy_
(
labels
):
if
not
_is_numpy_
(
labels
):
raise
ValueError
(
"The 'labels' must be a numpy ndarray."
)
raise
ValueError
(
"The 'labels' must be a numpy ndarray."
)
if
not
_is_numpy_
(
preds
):
if
not
_is_numpy_
(
preds
):
...
@@ -548,6 +664,9 @@ class Auc(MetricBase):
...
@@ -548,6 +664,9 @@ class Auc(MetricBase):
return
abs
(
x1
-
x2
)
*
(
y1
+
y2
)
/
2.0
return
abs
(
x1
-
x2
)
*
(
y1
+
y2
)
/
2.0
def
eval
(
self
):
def
eval
(
self
):
"""
Return the area (a float score) under auc curve
"""
tot_pos
=
0.0
tot_pos
=
0.0
tot_neg
=
0.0
tot_neg
=
0.0
auc
=
0.0
auc
=
0.0
...
@@ -609,20 +728,38 @@ class DetectionMAP(object):
...
@@ -609,20 +728,38 @@ class DetectionMAP(object):
Examples:
Examples:
.. code-block:: python
.. code-block:: python
exe = fluid.Executor(place)
import paddle.fluid.layers as layers
map_evaluator = fluid.Evaluator.DetectionMAP(input,
gt_label, gt_box, gt_difficult)
cur_map, accum_map = map_evaluator.get_map_var()
fetch = [cost, cur_map, accum_map]
for epoch in PASS_NUM:
map_evaluator.reset(exe)
for data in batches:
loss, cur_map_v, accum_map_v = exe.run(fetch_list=fetch)
In the above example:
batch_size = -1 # can be any size
image_boxs_num = 10
bounding_bboxes_num = 21
pb = layers.data(name='prior_box', shape=[image_boxs_num, 4],
append_batch_size=False, dtype='float32')
pbv = layers.data(name='prior_box_var', shape=[image_boxs_num, 4],
append_batch_size=False, dtype='float32')
loc = layers.data(name='target_box', shape=[batch_size, bounding_bboxes_num, 4],
append_batch_size=False, dtype='float32')
scores = layers.data(name='scores', shape=[batch_size, bounding_bboxes_num, image_boxs_num],
append_batch_size=False, dtype='float32')
nmsed_outs = fluid.layers.detection_output(scores=scores,
loc=loc, prior_box=pb, prior_box_var=pbv)
gt_box = fluid.layers.data(name="gt_box", shape=[batch_size, 4], dtype="float32")
gt_label = fluid.layers.data(name="gt_label", shape=[batch_size, 1], dtype="float32")
difficult = fluid.layers.data(name="difficult", shape=[batch_size, 1], dtype="float32")
exe = fluid.Executor(fluid.CUDAPlace(0))
map_evaluator = fluid.metrics.DetectionMAP(nmsed_outs, gt_label, gt_box, difficult, class_num = 3)
cur_map, accum_map = map_evaluator.get_map_var()
- 'cur_map_v' is the mAP of current mini-batch.
# see detailed examples at
- 'accum_map_v' is the accumulative mAP of one pass.
https://github.com/PaddlePaddle/models/blob/43cdafbb97e52e6d93cc5bbdc6e7486f27665fc8/PaddleCV/object_detection
"""
"""
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录