关于DPM(Deformable Part Model)算法中模型结构的解释
关于可变部件模型的描写叙述在作者[2010 PAMI]Object Detection with Discriminatively Trained Part Based Models的论文中已经有说明:
含有n个部件的目标模型能够形式上定义为一个(n+2)元组:(F0,P1,...,
Pn, b),F0是根滤波器,Pi是第i个部件的模型,b是表示偏差的实数值。每一个部件模型用一个三元组定义:(Fi,vi,
di),Fi是第i个部件的滤波器;vi是一个二维向量,指定第i个滤波器的锚点位置(anchor
position,即未发生形变时的标准位置) 相对于根的坐标;di是一个四维向量,指定了一个二次函数的參数,此二次函数表示部件的每一个可能位置相对于锚点位置的变形花费(deformation
cost)。
可是有了这个说明我们在看源代码时还是会有非常多不明确的地方,刚開始困扰我非常长时间,经过一段时间的分析,有了一定的理解,以下是我对这个模型的分析,如有不妥之处,请大家留言指正。
我分析的源代码版本号是voc-release3.1,从第4版開始增加了语法模型,更加复杂了,这里不讨论。
分析模型结构体主要看initmodel.m这个文件,通过看他怎样初始化模型,我们能够明确当中大多数字段的含义。
在这之前,我们能够在matlab中打开一两个源代码中自带的训练好的模型m文件,看看是什么样的。
inria_final.mat,Inria数据集上训练的单组件模型
person_final.mat,VOC2007数据集上训练的2组件人体模型
以下我会依次对各个字段做详解
(1) sbin,整型标量
sbin是HOG特征中cell的尺寸,即cell的尺寸为sbin * sbin
定义cell是为了将像素级特征聚合成为基于cell的特征C(i,j),0<=i<=|(w-1)/k|,0<=j<=|(h-1)/k|,參见[2010 PAMI]论文6.1.2 空间聚合,这里的k就是cell尺寸sbin,这样能够更明白的理解sbin的用处。
(2) interval,整型标量
HOG金字塔每组的层数
(3) numblocks,整型标量
模型中总的数据块的个数
(4) numcomponents,整型标量
组件的个数(含多少个组件模型)
(5) blocksizes,double数组,长度等于numblock
blocksizes[]数组的元素个数等于numblock,指明每一个数据块的大小,
所以,想知道哪个数据的大小,就先找到他的数据块标识,然后去blocksizes[]数组查找相应的数据块大小。
(6) regmult,double数组,长度等于numblock
还不清楚干什么的
(7) learnmult,double数组,长度等于numblock
还不清楚干什么的
(8) lowerbounds,cell类型数组,长度等于numblock
每一个数据块的数据值的下界,lowerbounds[]数组的元素个数等于numblock。
假如某个数据块的数据是向量或矩阵的话,则相应的lowerbounds元素值也是一个下界向量或矩阵
(9) maxsize,二元组
全部组件的根滤波器的[高度 宽度]的最大值,maxsize是统计全部组件的根滤波器的尺寸获得的。
(10) minsize,二元组
全部组件的根滤波器的[高度 宽度]的最小值,minsize是统计全部组件的根滤波器的尺寸获得的。
(11) thresh,double类型标量
保证高召回率须要的得分阈值,依据训练结果计算得到的。
也就是说,假设用此模型进行目标检測时,将得分阈值设为thresh可以保证非常高的召回率(recall rate),可是同一时候精度肯定就低了。
(12) rootfilters,cell类型数组,长度为组件个数
根滤波器数组,其每一个元素表示一个组件模型的根滤波器的信息,每一个元素包含3个字段:
size:根滤波器的尺寸,以cell为单位,w*h
w:根滤波器的參数向量,维数为(w*h)*31
blocklabel:此根滤波器所在的数据块标识
(13) partfilters,cell类型数组,长度为模型中全部组件的部件的个数之和
部件滤波器数组,其每一个元素表示一个部件滤波器的信息,每一个元素包含4个字段,注意:全部组件的部件是连续编号的
w:部件滤波器的參数向量,维数为(w*h)*31
partner:此部件相应的伙伴部件(对称部件)的索引,假设partner的值为0,表示此部件没有伙伴部件。
fake:是否假部件,值为1表示此部件是假部件,假部件不须要blocklabel
blocklabel:此部件滤波器所在的数据块标识
(14) defs,cell类型数组,长度为模型中全部组件的部件的个数之和
部件的锚点数组 (变形信息数组),每一个元素表示相应部件的可变形信息。注意:全部组件的部件在defs中也是连续编号的。每一个元素包括3个字段:
anchor:部件的锚点坐标,即部件未变形时的左上角点坐标,參见论文[2010 PAMI]中的公式(3)中的vi变量
w:w是一个四维向量,指明部件的变形花费函数的參数,參见论文[2010 PAMI]中的公式(2)中的di变量
blocklabel:所在的数据块标识
(15) offsets,cell类型数组,长度为组件个数
偏移量数组,每一个元素表示相应组件模型的偏移量,每一个元素又含有2个字段
w:组件的偏移量,是个实数值
blocklabel:所在的数据块标识
(16) components,cell类型数组,长度为组件个数
组件信息数组,每一个元素存储一个组件模型的信息,每一个元素又包括7个字段:
rootindex:根滤波器的索引,指出组件的根滤波器在rootfilters[]数组的位置
offsetindex:组件的偏移量索引,指出组件的偏移量在offsets[]数组的位置
part:组件的部件信息数组,长度为该组件的部件个数,每一个元素表示一个部件,每一个元素又有2个字段:
partindex:部件的索引,指出此部件在partfilters[]数组的位置,注意:全部组件的部件是连续编号的
defindex:部件的锚点数组索引,指出此部件的锚点在defs[]数组的位置,注意:全部组件的部件是连续编号的
dim:组件的维数(至于这个值详细是怎么计算来的,我如今还不清楚,弄清楚了来更新)
numblocks:组件所占的数据块个数(须要多少个数据块来存储此组件模型)
x1,y1,x2,y2:包围盒预測參数
一些须要解释的地方
关于模型中的blocklabel的理讲解明:
模型中有好多数组,比如rootfilters[],partfilters[],defs[],offsets[],这些数组的数据是以块block为单位进行存储的,通过块标识blocklabel来识别哪个数据位于哪个块上,所以上述数组的每一个成员都有一个blocklabel域,来指明此数据所在的块。numblock是总的数据块个数。blocksizes[]数组的元素个数等于numblock,指明每一个数据块的大小,
所以,想知道哪个数据的大小,就先找到他的数据块标识,然后去blocksizes[]数组查找相应的数据块大小。
关于部件的伙伴partner的解释:
在voc-release3.1中,模型是关于中轴对称的,所以就有了伙伴部件(或称为对称部件)的概念。注意:从第4版開始模型不再要求是对称的。一个部件存在伙伴,就说明有和此部件对称的部件。伙伴部件的參数向量和此部件的參数向量w全然同样,一个部件和其伙伴部件是关于模型的中轴对称的,所以不须要反复存储參数向量w,这样,就产生了一些假(fake)部件,它们仅仅有一个空壳。假部件(fake=1)没有blocklabel,由于它们不须要数据块来存储信息,它们的信息存储在相应的伙伴部件中。假设partner的值为0,表示此部件没有伙伴部件。
关于偏移量offset的解释:
偏移量offset就是加在每一个组件模型的得分公式最后的一个实数值,參见论文[2010 PAMI]中的公式(2)中的b值。
关于锚点数组defs的解释:
一開始我以为def是定义definition的缩写,但解释不通,如今知道了,def是可变形deformable的缩写,所以defs数组中存储的是全部部件的变形參数。anchor,锚点坐标,是部件未发生移动时的坐标,即理想状况下的位置,更具体的说,anchor是部件未变形时的左上角点坐标。w,是个4维向量,指明部件i的变形花费函数的參数,參见论文[2010
PAMI]中的公式(2)中的di变量。通过分析源代码发现w中4个參数分别相应这4个偏移量[Δx^2, Δx, Δy^2, Δy](分析可视化变形花费的代码时确定的,在文件visualizemodel.m中)。
关于每一个cell的特征向量的维数为什么是31维的解释:
设C是聚合有9个对照度不敏感方向的像素级特征映射而获得的基于cell的特征映射,D是聚合有18个对照度敏感方向的像素级特征而获得的基于cell的特征映射。用4种不同的归一化方法对C(i,j)和D(i,j)进行归一化和截断(限幅),能够获得一个4*(9+18)=108维的特征向量F(i,j)。实际中我们使用此108维向量的一个解析投影,此投影由以下几个统计量定义:27个在不同归一化因子上的累加和(即列的和),F中的每一个方向通道相应一个;以及4个在不同方向(9维对照度不敏感方向)上的累加和(即行的和),每一个归一化因子相应一个。cell尺寸k=8,截断(限幅)阈值α=0.2。终于的特征映射是31维向量G(i,j),当中27维相应不同的方向通道(9个对照度不敏感方向和18个对照度敏感方向),剩下4维表示(i,j)周围4个cell组成的block的梯度能量。详见[2010
PAMI]论文中6.2节 PCA和解析降维。
明确了这些,初始化的过程就非常easy看懂了,初始化的过程在以下的源代码上有凝视:
function model = initmodel(pos, sbin, size)
% 初始化模型结构
% 參数
% pos:用来训练此模型的正样本数组
% sbin:HOG特征的最小单位,cell的长度
% size:根滤波器尺寸
% 假设不提供根滤波器的尺寸,则依据正样本的长宽比统计信息计算得出
%
% model = initmodel(pos, sbin, size)
% Initialize model structure.
% If not supplied,the dimensions of the model template are computed from statistics in the postive examples.
%
% 关于cell尺寸sbin的解释:
% 将像素级特征聚合成为基于cell的特征C(i,j),0<=i<=|(w-1)/k|,0<=j<=|(h-1)/k|,參见[2010 PAMI]论文6.1.2 空间聚合,
% 这里的k就是cell尺寸sbin,这样能够更明白的理解sbin的用处
%
% 关于部件的伙伴partner的解释:
% 在voc-release3.1中,模型是关于中轴对称的,所以就有了伙伴部件(或称为对称部件)的概念。注意:从第4版開始模型不再要求是对称的。
% 一个部件存在伙伴,就说明有和此部件对称的部件。伙伴部件的參数向量和此部件的參数向量w全然同样,一个部件和其伙伴部件是关于模型的中轴对称的,
% 所以不须要反复存储參数向量w,这样,就产生了一些假(fake)部件,它们仅仅有一个空壳。假部件(fake=1)没有blocklabel,由于它们不须要数据块来存储信息,
% 它们的信息存储在相应的伙伴部件中。假设partner的值为0,表示此部件没有伙伴部件。
%
% 关于每一个cell的特征向量的维数为什么是31维的解释:
% 设C是聚合有9个对照度不敏感方向的像素级特征映射而获得的基于cell的特征映射,D是聚合有18个对照度敏感方向的像素级特征而获得的基于cell的特征映射。
% 用4种不同的归一化方法对C(i,j)和D(i,j)进行归一化和截断(限幅),能够获得一个4*(9+18)=108维的特征向量F(i,j)。
% 实际中我们使用此108维向量的一个解析投影,此投影由以下几个统计量定义:
% 27个在不同归一化因子上的累加和(即列的和),F中的每一个方向通道相应一个;以及4个在不同方向(9维对照度不敏感方向)上的累加和(即行的和),每一个归一化因子相应一个。
% cell尺寸k=8,截断(限幅)阈值α=0.2。终于的特征映射是31维向量G(i,j),当中27维相应不同的方向通道(9个对照度不敏感方向和18个对照度敏感方向),
% 剩下4维表示(i,j)周围4个cell组成的block的梯度能量。详见[2010 PAMI]论文中6.2节 PCA和解析降维
%
% 关于模型中的blocklabel的理讲解明:
% 模型中有好多数组,比如rootfilters[],partfilters[],defs[],offsets[],这些数组的数据是以块block为单位进行存储的,
% 通过块标识blocklabel来识别哪个数据位于哪个块上,所以上述数组的每一个成员都有一个blocklabel域,来指明此数据所在的块。
% numblock是总的数据块个数(注意不是总的部件个数,刚開始理解错了),blocksizes[]数组的元素个数等于numblock,指明每一个数据块的大小,
% 所以,想知道哪个数据的大小,就先找到他的数据块标识,然后去blocksizes[]数组查找相应的数据块大小。
%
% 关于偏移量offset的解释:
% 偏移量offset就是加在每一个组件模型的得分公式最后的一个实数值,參见论文[2010 PAMI]中的公式(2)中的b值
%
% 关于锚点坐标数组defs[]的解释:
% 一開始我以为def是定义definition的缩写,但解释不通,如今知道了,def是可变形deformable的缩写,所以defs数组中存储的是全部部件的变形參数
% anchor,锚点坐标,是部件未发生移动时的坐标,即理想状况下的位置,更具体的说,anchor是部件未变形时的左上角点坐标
% w,是个4维向量,指明部件i的变形花费函数的參数,參见论文[2010 PAMI]中的公式(2)中的di变量
% 通过分析源代码发现w中4个參数分别相应这4个偏移量[Δx^2, Δx, Δy^2, Δy](分析可视化变形花费的代码时确定的,在文件visualizemodel.m中)
%
% model.sbin HOG cell的尺寸
% model.interval 金字塔每组的层数
% model.numblocks 总的数据块个数
% model.numcomponents 组件的个数(含多少个组件模型)
% model.blocksizes 每一个数据块的大小,blocksizes[]数组的元素个数等于numblock
% model.regmult
% model.learnmult
% model.lowerbounds 每一个数据块的数据值的下界,lowerbounds[]数组的元素个数等于numblock
%假如某个数据块的数据是向量或矩阵的话,则相应的lowerbounds元素值也是一个下界向量或矩阵
% model.maxsize 全部组件的根滤波器的[高度 宽度]的最大值
% model.minsize 全部组件的根滤波器的[高度 宽度]的最小值
% model.rootfilters{i} 组件i的根滤波器,根滤波器数组的长度等于组件个数
% .size 组件i的根滤波器的尺寸(以cell为单位),w*h
% .w 组件i的根滤波器向量,维数为(w*h)*31
% .blocklabel 组件i的根滤波器所在的数据块标识
% model.partfilters{i} 第i个部件滤波器,部件滤波器数组的长度等于全部组件的部件个数的总和,注意:全部组件模型的part是连续编号的
% .w 第i个部件滤波器向量,维数为(w*h)*31
% .blocklabel 第i个部件滤波器所在的数据块标识
% model.defs{i} 第i个部件的锚点,部件锚点数组的长度等于全部组件的部件个数的总和,全部组件模型的part是连续编号的
% .anchor 第i个部件的锚点坐标,即部件i未变形时的左上角点坐标,參见论文[2010 PAMI]中的公式(3)中的vi变量
% .w w是一个四维向量,指明部件i的变形花费函数的參数,參见论文[2010 PAMI]中的公式(2)中的di变量
% .blocklabel 第i个部件的锚点所在的数据块标识
% model.offsets{i} 第i个组件的偏移量,偏移量数组的长度等于组件个数
% .w 第i个组件的偏移量,是个实数值
% .blocklabel 第i个组件的偏移量所在的数据块标识
% model.components{i} 第i个组件的信息
% .rootindex 第i个组件的根滤波器的索引,指出组件i的根滤波器在rootfilters[]数组的位置
% .parts{j} 第i个组件的第j个部件的信息
% .partindex 第i个组件的第j个部件的部件滤波器数组索引,指出在partfilters[]数组的位置,注意:全部组件模型的part是连续编号的
% .defindex 第i个组件的第j个部件的锚点坐标数组索引,指出在defs[]数组的位置,注意:全部组件模型的part是连续编号的
% .offsetindex 第i个组件的偏移量的索引,指出在offsets[]数组的位置
% .dim 第i个组件的维数
% .numblocks 第i个组件模型的数据块个数(须要多少个数据块来存储组件模型i),全部组件的numblocks之和等于整个模型的numblocks % 计算此组正样本典型的高宽比aspect
% pick mode of aspect ratios
h = [pos(:).y2]' - [pos(:).y1]' + 1; % 全部正样本的高度数组
w = [pos(:).x2]' - [pos(:).x1]' + 1; % 全部正样本的宽度数组
xx = -2:.02:2;
filter = exp(-[-100:100].^2/400);
aspects = hist(log(h./w), xx);
aspects = convn(aspects, filter, 'same');
[peak, I] = max(aspects);
aspect = exp(xx(I)); % 选择根滤波器的面积(以像素为单位)
% pick 20 percentile area
areas = sort(h.*w); % 对面积从小到大排序
area = areas(floor(length(areas) * 0.2)); % 选择从小到大20%处的面积
area = max(min(area, 5000), 3000); % 面积最大值5000,最小值3000 % 依据以上计算出的高宽比和面积,得到根滤波器的高度和宽度(以像素为单位)
% pick dimensions
w = sqrt(area/aspect); % 根的宽度(以像素为单位),area=w*h,aspect=h/w,所以area/aspect = (w*h)*(w/h) = w^2
h = w*aspect; % 根的高度(以像素为单位) % HOG cell的尺寸,默觉得8
% size of HOG features
if nargin < 4
model.sbin = 8;
else
model.sbin = sbin;
end % 设定根滤波器尺寸(以cell为单位)
% size of root filter
if nargin < 5
model.rootfilters{1}.size = [round(h/model.sbin) round(w/model.sbin)]; % 以像素为单位的根的尺寸除以cell的尺寸,等于以cell为单位的根的尺寸
else
model.rootfilters{1}.size = size;
end % 初始化偏移量数组
% 偏移量offset就是加在每一个组件模型的得分公式最后的一个实数值,參见论文[2010 PAMI]中的公式(2)中的b值
model.offsets{1}.w = 0; % 组件1的偏移量的初始值设为0
model.offsets{1}.blocklabel = 1; % 组件1的偏移量的块标识,值为1表明组件1的偏移量是模型中的第一个数据块
model.blocksizes(1) = 1; % 第一个数据块的大小,值为1,由于仅仅用来存储组件1的偏移量,即一个实数值
model.regmult(1) = 0;
model.learnmult(1) = 20;
model.lowerbounds{1} = -100; % 第一个数据块的数据值的下界,-100 % 初始化根滤波器数组
model.rootfilters{1}.w = zeros([model.rootfilters{1}.size 31]); % 组件1的初始根滤波器的向量值为全0
height = model.rootfilters{1}.size(1); % 组件1的根滤波器的高度
% root filter is symmetric 注意:根滤波器是左右对称的
width = ceil(model.rootfilters{1}.size(2)/2); % 组件1的根滤波器的宽度的一半
model.rootfilters{1}.blocklabel = 2; % 组件1的根滤波器的块标识
model.blocksizes(2) = width * height * 31; % 存储组件1的根滤波器的数据块(第二个数据块)的大小,由于根滤波器是左右对称的,所以是h*(w/2)*31
model.regmult(2) = 1;
model.learnmult(2) = 1;
model.lowerbounds{2} = -100*ones(model.blocksizes(2),1); % 第二个数据块的值的下界,-100 % 初始化组件模型
% set up one component model
model.components{1}.rootindex = 1; % 组件1的根滤波器数组索引
model.components{1}.offsetindex = 1; % 组件1的偏移量数组索引
model.components{1}.parts = {}; % 组件1的部件信息,当前为空
model.components{1}.dim = 2 + model.blocksizes(1) + model.blocksizes(2); % 组件1的模型參数向量维数
model.components{1}.numblocks = 2; % 组件1的数据块个数 % 初始化剩下的模型參数
% initialize the rest of the model structure
model.interval = 10; % 金字塔每组的层数
model.numcomponents = 1; % 模型的组件个数
model.numblocks = 2; % 模型的总的数据块个数
model.partfilters = {}; % 部件滤波器数组
model.defs = {}; % 部件锚点数组
model.maxsize = model.rootfilters{1}.size; % 全部组件的根滤波器的最大值
model.minsize = model.rootfilters{1}.size; % 全部组件的根滤波器的最小值
相关链接
Deformable Part Model 相关网页:http://www.cs.berkeley.edu/~rbg/latent/index.html
Pedro Felzenszwalb的个人主页:http://cs.brown.edu/~pff/
PASCAL VOC 目标检測挑战:http://pascallin.ecs.soton.ac.uk/challenges/VOC/
A Discriminatively Trained, Multiscale, Deformable Part Model [CVPR 2008] 中文翻译
Object Detection with Discriminatively Trained Part Based Models [PAMI 2010]中文翻译
有关可变形部件模型(Deformable Part Model)的一些说明
在Windows下执行Felzenszwalb的Deformable Part Models(voc-release4.01)目标检測matlab源代码
在Windows下执行Felzenszwalb的star-cascade DPM(Deformable Part Models)目标检測Matlab源代码
在windows下执行Felzenszwalb的Deformable Part Model(DPM)源代码voc-release3.1来训练自己的模型
用DPM(Deformable Part Model,voc-release3.1)算法在INRIA数据集上训练自己的人体检測模型
关于DPM(Deformable Part Model)算法中模型结构的解释的更多相关文章
- 关于DPM(Deformable Part Model)算法中模型可视化的解释
搭建了自己的博客平台,本文地址:http://masikkk.com/blog/DPM-model-visualization/ DPM源代码(voc-release)中的模型可视化做的还算相当炫酷的 ...
- DPM(Deformable Parts Model)--原理(一)(转载)
DPM(Deformable Parts Model) Reference: Object detection with discriminatively trained partbased mode ...
- DPM(Deformable Parts Model)
DPM(Deformable Parts Model) Reference: Object detection with discriminatively trained partbased mode ...
- DPM(Deformable Parts Model)--原理(一)
http://blog.csdn.net/ttransposition/article/details/12966521 DPM(Deformable Parts Model) Reference: ...
- (2) 用DPM(Deformable Part Model,voc-release4.01)算法在INRIA数据集上训练自己的人体检測模型
步骤一,首先要使voc-release4.01目标检測部分的代码在windows系统下跑起来: 參考在window下执行DPM(deformable part models) -(检測demo部分) ...
- 使用DPM(Deformable Part Model,voc-release3.1)算法INRIA通过训练你的身体检测模型数据集
我的环境 DPM源代码版本号:voc-release3.1 VOC开发包版本号:VOC2007_devkit_08-Jun Matlab版本号:MatlabR2012b c++编译器:VS2010 系 ...
- kmp算法中的nextval实例解释
求nextval数组值有两种方法,一种是不依赖next数组值直接用观察法求得,一种方法是根据next数组值进行推理,两种方法均可使用,视更喜欢哪种方法而定. 本文主要分析nextval数组值的第二种方 ...
- DPM(Deformable Part Model)原理详解(汇总)
写在前面: DPM(Deformable Part Model),正如其名称所述,可变形的组件模型,是一种基于组件的检测算法,其所见即其意.该模型由大神Felzenszwalb在2008年提出,并发表 ...
- Journal of Proteomics Research | 利用混合蛋白质组模型对MBR算法中错误转移鉴定率的评估
题目:Evaluating False Transfer Rates from the Match-between-Runs Algorithm with a Two-Proteome Model 期 ...
随机推荐
- Flightgear 编译
一.FlightGear简介 FlightGear 始于1997年,是一个开源的多平台飞行模拟器. 二.FlightGear编译过程 FlightGear平台的说明文档见:http://wiki.fl ...
- 使用程序往Neo4j导入CSV报错
今天在用程序向Neo4j导入csv文件时,报以下错误: java.net.ConnectException: Connection refused: connect java.rmi.ConnectE ...
- html的input输入框提示信息 点击隐藏
<input type="text" <!-- 文本框 --> name="textfield" value="请输入...& ...
- input file文件上传样式
<style> .file-group { position: relative; width: 200px; height: 80px; ...
- 阿里云 centos vim 显示中文 乱码
开始以为是vim 设置编码的问题 :网上搜 改 .vimrc 无效!!! 后转战 是不是系统里面没有中文字体 1.先从你本机 C:\Windows\Fonts 拷贝或者网络上下载你想要安装的 ...
- YII Query Builder
今天遇到一个Query Builder 联合查询问题: 查询关联表某个字段的总数
- Lua 5.1 for Delphi 2010
This is a Lua 5.1 Wrapper for Delphi 2009 and Delphi 2010 which automatically creates OOP callback f ...
- Day11 Memcached、Redis和RabbitMQ安装
修改Ubuntu镜像源: sudo vi /etc/apt/sources.list 全部替换为以下内容: ---------------------------------------------- ...
- Python Tutorial 学习(四)--More Control Flow Tools
4.1 if 表达式 作为最为人熟知的if.你肯定对这样的一些表达式不感到陌生: >>> x = int(raw_input("Please enter an intege ...
- [BZOJ 1207] [HNOI 2004] 打鼹鼠 【DP】
题目链接:BZOJ - 1207 题目分析 每一次打鼹鼠一定是从上一次打某只鼹鼠转移过来的,从打第 j 只鼹鼠能不能转移到打第 i 只鼹鼠,算一下曼哈顿距离和时间差就知道了. 那么就有一个 DP ,用 ...