引言

有时我们需要使用CCR测评器(CCR-Plus是一个开源的信息学竞赛测评软件,Github链接https://github.com/sxyzccr/CCR-Plus)进行SpecialJudge(以下简称SPJ)。例如判断选手输出与标准输出的差距,大于一定的值就算错,这时就需要用SpecialJudge了。

在CCR测评器中,SPJ是用一项叫做自定义校验器的功能实现的。CCR的文档没有写明校验器的语法,网上也没有这一类的信息。于是,我在CCR的源代码中找到了CCR的默认校验器(全文比较),并将校验器的写法写成此篇博客。

正文

SPJ程序的位置

编译好的SPJ程序放在\data\prob\目录下(prob是题目名)。

如何写SPJ校验器

查看CCR默认全文比较校验器源代码:

// https://github.com/sxyzccr/CCR-Plus/blob/master/src/tools/checker/fulltext_utf8.cpp
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
using namespace std; string In, Out, Ans, Log;
FILE* fout, *fans, *flog; void End(const string& info, double x, int state = 0)
{
fprintf(flog, "%.3lf\n%s\n", x, info.c_str());
exit(state);
} inline void filter(string& s)
{
for (; s.size() && isspace(s[s.size() - 1]); s.erase(s.size() - 1));
} string elided(const string& s, int p)
{
string pre = "", suf = "";
for (int i = 0; i < p; i++) pre.push_back(s[i]);
if (pre.size() > 3) pre = string("…") + pre.substr(pre.size() - 3, 3);
int l = s.size() - p;
if (pre.size() + l >= 13) l = 11 - pre.size(), suf = "…";
for (int i = 0; i < l; i++) pre.push_back(s[p + i]);
return pre + suf;
} int compare(const string& a, const string& b)
{
int la = a.length(), lb = b.length();
for (int i = 0; i < la && i < lb; i++) if (a[i] != b[i]) return i;
return la != lb ? min(la, lb) : -1;
} void Open()
{
if (Log.size()) flog = fopen(Log.c_str(), "w"); else flog = stdout;
if (flog == NULL) exit(1);
if ((fans = fopen(Ans.c_str(), "r")) == NULL) exit(1);
if ((fout = fopen(Out.c_str(), "r")) == NULL) exit(1);
} void Check()
{
char s[256];
for (int i = 1; !feof(fout) || !feof(fans); i++)
{
string s1 = "", s2 = "";
char c1 = -1, c2 = -1;
for (; !feof(fans) && (c1 = fgetc(fans)) != '\n';) if (c1 != -1) s1.push_back(c1);
for (; !feof(fout) && (c2 = fgetc(fout)) != '\n';) if (c2 != -1) s2.push_back(c2);
if (feof(fout) && s1 != "" && s2 == "")
{
if (i == 1) End("选手输出为空", 0);
sprintf(s, "第%d行 标准输出:\"%s\" 选手输出已结尾", i, elided(s1, 0).c_str());
End(s, 0);
}
if (feof(fans) && s1 == "" && s2 != "")
{
sprintf(s, "第%d行 标准输出已结尾 选手输出:\"%s\"", i, elided(s2, 0).c_str());
End(s, 0);
}
filter(s1), filter(s2);
int p = compare(s1, s2);
if (p >= 0)
{
sprintf(s, "第%d行 标准输出:\"%s\" 选手输出:\"%s\"", i, elided(s1, p).c_str(), elided(s2, p).c_str());
End(s, 0);
}
}
} int main(int argc, char* argv[])
{
In = "";
Ans = argc < 3 ? "" : argv[2];
Out = argc < 4 ? "" : argv[3];
Log = argc < 5 ? "" : argv[4];
Open();
Check();
End("", 1);
}

SPJ程序需要两个必要的函数:

FILE* fout, *fans, *flog;
void End(const string& info, double x, int state = 0){
fprintf(flog, "%.3lf\n%s\n", x, info.c_str());
exit(state);
}
void Open(){
if (Log.size()) flog = fopen(Log.c_str(), "w"); else flog = stdout;
if (flog == NULL) exit(1);
if ((fans = fopen(Ans.c_str(), "r")) == NULL) exit(1);
if ((fout = fopen(Out.c_str(), "r")) == NULL) exit(1);
}

  

Open()是进行程序的初始化,End则是返回分数和备注。

当CCR需要调用校验器时,它会向SPJ程序传递一个参数数组argv[]。数组的第二项为标准答案,第三项为选手答案,第四项为一些日志。

当程序开始时,我们先获取这几个参数。

int main(int argc, char* argv[])
{
string In, Out, Ans, Log;
In = "";
Ans = argc < 3 ? "" : argv[2];
Out = argc < 4 ? "" : argv[3];
Log = argc < 5 ? "" : argv[4];
//Ans和Out变量存储的是标准答案和选手答案的路径
}

  

接下来我们需要读入标准答案和选手答案。我们使用freopen重定向到答案文件。

Open();//初始化,这句很重要。
double answer,output;//标准输出和选手输出
freopen(Ans.c_str(),"r",stdin);//重定向到标准输出
cin>>answer;//读入
freopen(Out.c_str(),"r",stdin);//重定向到选手输出
cin>>output;//读入

  

接下来是判断,具体的读入和判断过程因题目而异。

判断完成后需要使用End函数返回结果。End函数的使用很简单,第一个参数是要显示在测评记录上的字符串,第二个参数是double类型的,表示分数百分比,是一个0-1的值(例如这个测试点为10分,这个参数为0.6,那么这个测试点最终得分就是6分)。

if (abs(ans-output)>0.02){
End("与标准答案相差过大",0);
}else{
End("",1);
}

  

最终的SPJ代码就是这样:

#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
FILE* fout, *fans, *flog;
void End(const string& info, double x, int state = 0){
fprintf(flog, "%.3lf\n%s\n", x, info.c_str());
exit(state);
}
void Open(){
if (Log.size()) flog = fopen(Log.c_str(), "w"); else flog = stdout;
if (flog == NULL) exit(1);
if ((fans = fopen(Ans.c_str(), "r")) == NULL) exit(1);
if ((fout = fopen(Out.c_str(), "r")) == NULL) exit(1);
}
int main(int argc, char* argv[]){
string In, Out, Ans, Log;
In = "";
Ans = argc < 3 ? "" : argv[2];
Out = argc < 4 ? "" : argv[3];
Log = argc < 5 ? "" : argv[4];
//Ans和Out变量存储的是标准答案和选手答案的路径
Open();//初始化,这句很重要。
double answer,output;//标准输出和选手输出
freopen(Ans.c_str(),"r",stdin);//重定向到标准输出
cin>>answer;//读入
freopen(Out.c_str(),"r",stdin);//重定向到选手输出
cin>>output;//读入
if (abs(ans-output)>0.02){
End("与标准答案相差过大",0);
}else{
End("",1);
}
return 0;
}

  

在CCR的高级配置中从下拉菜单选择这个校验器,就可以使用这个校验器测评这道SPJ题了。

关于CCR测评器的自定义校验器(Special Judge)的更多相关文章

  1. cena评测系统:自定义校验器(自定义评测插件编写)

    Cena评测系统,最受欢迎的信息学竞赛离线评测系统. 它是开放源程序的信息学竞赛评测系统,能满足大多数程序设计竞赛的测评需求. 特色功能: 通过局域网自动收取选手程序. 高效率的数据文件配置工具. 自 ...

  2. CXF添加拦截器和自定义拦截器

    前面讲了如何采用CXF开发webservice,现在来讲如何添加拦截器和自定义拦截器. 服务端代码: HelloWorld implementor=new HelloWorldImpl(); Stri ...

  3. flask之web网关、三件套、配置、路由(参数、转化器及自定义转化器)、cbv、模板语言、session

    目录 1.wsgiref.py 2.werzeug.py 3.三件套 4.配置文件 5.路由本质 6.cbv.py 7.路由转化器 8.自定义转化器 9.模板语言 10.session原理 11.te ...

  4. Elasticsearch修改分词器以及自定义分词器

    Elasticsearch修改分词器以及自定义分词器 参考博客:https://blog.csdn.net/shuimofengyang/article/details/88973597

  5. struts系列:校验(二)自定义校验器

    一.自定义校验类 public class PasswordValidator extends FieldValidatorSupport { @Override public void valida ...

  6. Bean Validation完结篇:你必须关注的边边角角(约束级联、自定义约束、自定义校验器、国际化失败消息...)

    每篇一句 没有任何技术方案会是一种银弹,任何东西都是有利弊的 相关阅读 [小家Java]深入了解数据校验:Java Bean Validation 2.0(JSR303.JSR349.JSR380)H ...

  7. 【参数校验】 自定义校验器 (实现ConstraintValidator)

    日常工作中写接口时,往往需要校验前端传来的枚举状态码,例如"1","2"等等, 这里使用java 303规范的参数校验框架封装一个自定义参数校验器: /** * ...

  8. struts2(五)之struts2拦截器与自定义拦截器

    前言 前面介绍了struts2的输入验证,如果让我自己选的话,肯定是选择xml配置校验的方法,因为,能使用struts2中的一些校验规则,就无需自己编写了, 不过到后面应该都有其他更方便的校验方法,而 ...

  9. struts2内置拦截器和自定义拦截器详解(附源码)

    一.Struts2内置拦截器 Struts2中内置类许多的拦截器,它们提供了许多Struts2的核心功能和可选的高级特 性.这些内置的拦截器在struts-default.xml中配置.只有配置了拦截 ...

随机推荐

  1. mybatis之注解方式实现

    * 使用mybatis举例,使用注解方式实现* 不需要针对UserMapperI接口去编写具体的实现类代码,这个具体的实现类由MyBatis帮我们动态构建出来,我们只需要直接拿来使用即可.* 1.导入 ...

  2. MySQL5.7 的编译安装

    转: 5.7的安装: https://www.insp.top/article/make-install-mysql-5-7 5.6的安装: https://www.chenyudong.com/ar ...

  3. HGOI 20181030晚 题解

    Problem:给出全班人的个数总分和小明的分数(满分100分),求小明最低排名和最高排名 sol:假设小明的排名为k,总分为sum,小明的分数是r, 贪心求解, 最坏情况下,小明前面的比小明高一分( ...

  4. Libre 6005 「网络流 24 题」最长递增子序列 / Luogu 2766 最长递增子序列问题(网络流,最大流)

    Libre 6005 「网络流 24 题」最长递增子序列 / Luogu 2766 最长递增子序列问题(网络流,最大流) Description 问题描述: 给定正整数序列x1,...,xn . (1 ...

  5. mac os x 之通过远程主机在nginx上部署web静态页面

    1.mac使用ssh命令登陆远程主机 因为苹果mac os x自带ssh命令,所以我们只需打开终端输入 $ ssh user@remote 在这之前最好在服务器上上传自己的ssh key,避免每次登陆 ...

  6. java基础知识代码-------枚举类型

    package com.mon10.day22; /** * 类说明 :枚举类型,案例二 * * @author 作者 : chenyanlong * @version 创建时间:2017年10月22 ...

  7. vue+webpack开发(一)

    一开始接触这个vue+webpack的时候,实在是摸不着头脑,根本无从下手. 但是经过这两天的研究,其实你会发现vue其实并不难,难度都在webpack你对webpack的理解. webpack顾名思 ...

  8. S折交叉验证(S-fold cross validation)

    S折交叉验证(S-fold cross validation) 觉得有用的话,欢迎一起讨论相互学习~Follow Me 仅为个人观点,欢迎讨论 参考文献 https://blog.csdn.net/a ...

  9. jQuery总结或者锋利的jQuery笔记二

    第三章  jQuery 中 DOM 操作 , 进入这一章,你必须先要有 选择器的基础, 最好是基本选择器 (id,class,*,div,p 组合等) ,  层次选择器(div ul),(div> ...

  10. AngularJs-$parsers自我理解-解析

    $parsers 首先先了解下它具体的作用,当用户与控制器进行交互的时候.ngModelController中的$setViewValue()方法就会被调用,$parsers的数组中函数就会以流水线的 ...