• 基于C++的Latex表格代码自动生成


    目录

    一. 前言

    二.代码实现

    三.生成结果


    一. 前言

    相信做科研的同学一定经常接触到Latex。Latex做为脚本控制的自动排版系统,其生成的文本整洁,清楚,对于图表的自动布局,能够有效的安排页面空间,使用户的文本可以直接获得印刷出版物级别的质量。但是Latex有一个非常蛋疼的地方,就是表格。Latex表格是通过格式化的表示形式实现其排版的,其过程繁琐,容易出错。每次想到要将一个大的表格排到Latex里,就头疼。鉴于表格生成的复杂性,一些表格代码自动生成工具应运而生。我个人最经常用的是Table Generator。在线访问Table Generator,将excel数据粘贴到该网站,通过一个非常简洁的交互页面,就能够直接生成latex表格代码,操作界面如下图:

    但是使用该工具仍然有问题。比如一些调整表格风格的代码还需要后续添加,一些数据按照大小极值需要加粗,在Table Generator中需要人工的标定,依然费时费力。鉴于以后要经常使用,所以决定基于C++写一个表格代码生成程序,以方便后续使用。


    二.代码实现

    首先我们要先将需要表格化的数据,存储在txt。我们需要给出对应的信息。这里我们举两个例子:

    caption for table
    1 15 0
    3
    Method1 Method2 Method3 Method4 Method5 Method6 Method7 Method8 Method9 Method10 Method11 Method12 Method13 Method14 Method15
    M1 M2 M3
    Max
    1.2 1.1 1.4 5.6 5.3 5.6 2.2 3.1 0.4 15.6 25.3 45.6 11.2 51.1 21.4
    0.2 0.1 0.4 15.6 15.3 15.6 72.2 33.1 0.4 65.6 125.3 145.6 1.2 351.1 221.4
    21.1 31.1 51.4 75.6 95.3 25.6 22.2 23.1 20.4 215.6 225.3 245.6 211.2 251.1 221.4

    /*
    第一行:行label的基本格式,p1:数据列数,p2:第一行label数,p3:第二行label数;
    第二行:列label的基本格式,p1:数据行数;
    第三行:行label第一行label的string list;
    第四行:行label第二行label的string list;
    第五行:列label的string list
    第六行:Max,加粗每个对应数据组最大值;Min,加粗每个对应数据组最小值
    第七行及其他: 数据本身,默认float类型
    */

    这是一个表格数据在txt存储的形式,包括行的标识以及列的标识,都有清楚的对应关系。通过指定Max或者Min,程序能够自动加粗每一行的最大或者最小数据,而不再需要人工查找。这里默认的数据类型是float。

    caption for table
    2 5 3
    3
    Method1 Method2 Method3 Method4 Method5
    Q1 Q2 Q3
    M1 M2 M3
    Max
    1.2 1.1 1.4 5.6 5.3 5.6 2.2 3.1 0.4 15.6 25.3 45.6 11.2 51.1 21.4
    0.2 0.1 0.4 15.6 15.3 15.6 72.2 33.1 0.4 65.6 125.3 145.6 1.2 351.1 221.4
    21.1 31.1 51.4 75.6 95.3 25.6 22.2 23.1 20.4 215.6 225.3 245.6 211.2 251.1 221.4

    /*
    第一行:行label的基本格式,p1:数据列数,p2:第一行label数,p3:第二行label数;
    第二行:列label的基本格式,p1:数据行数;
    第三行:行label第一行label的string list;
    第四行:行label第二行label的string list;
    第五行:列label的string list
    第六行:Max,加粗每个对应数据组最大值;Min,加粗每个对应数据组最小值
    第七行及其他: 数据本身,默认float类型
    */

    与第一个例子类似,这个表格稍微复杂,行标识有两级。在实际使用中,因为对每一个方法可能有不同的度量(Q1\Q2\Q3),要放在一个表格中,那么支持两级行标识就变的比较实用。

    有了数据的存储方法,剩下的就是写一个程序,读取这些数据,并生成Latex表格代码了。数据载入以及生成的C++代码如下:

    1. #pragma once
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. using namespace std;
    9. class LatexTable {
    10. private:
    11. vector firstRowLabel;
    12. vector subRowLabel;
    13. vector columnLabel;
    14. string orderName;//Max, Min, Non;
    15. vectorfloat>> data;
    16. string caption;
    17. string fileName;
    18. public:
    19. void LatexTable_init(string fileName_input) {
    20. fileName = fileName_input;
    21. LatexTable_Load();
    22. Latextable_Write();
    23. }
    24. private:
    25. void LatexTable_Load() {
    26. int r1, r2, r3;
    27. int r_final;
    28. int c1;
    29. fstream out2;
    30. out2.open(fileName, ios::in);
    31. getline(out2, caption);
    32. out2 >> r1>>r2>>r3>>c1;
    33. for (int i = 0; i < r2; i++) {
    34. string firstRowLabel_i;
    35. out2 >> firstRowLabel_i;
    36. firstRowLabel.push_back(firstRowLabel_i);
    37. }
    38. r_final = r2;
    39. if (r1 == 2) {
    40. for (int i = 0; i < r3; i++) {
    41. string subRowLabel_i;
    42. out2 >> subRowLabel_i;
    43. subRowLabel.push_back(subRowLabel_i);
    44. }
    45. r_final = r_final* r3;
    46. }
    47. for (int i = 0; i < c1; i++) {
    48. string columnLabel_i;
    49. out2 >> columnLabel_i;
    50. columnLabel.push_back(columnLabel_i);
    51. }
    52. //load order
    53. out2 >> orderName;
    54. //load data
    55. for (int i = 0; i < c1; i++) {
    56. vector<float> data_i;
    57. for (int j = 0; j < r_final; j++) {
    58. float r_final_i;
    59. out2 >> r_final_i;
    60. data_i.push_back(r_final_i);
    61. }
    62. data.push_back(data_i);
    63. }
    64. cout << "Data loaded finished." << endl;
    65. }
    66. void Latextable_Write() {
    67. string fileName_Save;
    68. int findTxt = fileName.find_last_of(".txt");
    69. fileName_Save = fileName.substr(0, findTxt-3) + "_Latex.txt";
    70. ofstream f1(fileName_Save, ios::app);
    71. int r_final;
    72. if (subRowLabel.size() == 0) {
    73. r_final = firstRowLabel.size();
    74. }
    75. else {
    76. r_final = firstRowLabel.size() * subRowLabel.size();
    77. }
    78. //1.首先写入label的序列
    79. f1 << "\\begin{table*}[]" << endl;
    80. f1 << "\\centering" << endl;
    81. f1 << "\\caption{" << caption << ".}" << endl;
    82. f1 << "\\begin{tabular}{";
    83. f1 << "c|";
    84. for (int i = 0; i < r_final; i++) {
    85. f1 << "c";
    86. }
    87. f1 << "}";
    88. f1 << "\\toprule[1pt]" << endl;
    89. if (subRowLabel.size() == 0) {
    90. f1 << "\\textbf{Methods} & ";
    91. for (int i = 0; i < firstRowLabel.size() - 1; i++) {
    92. f1 << "\\textbf{" << firstRowLabel[i] << "} &";
    93. }
    94. f1 << "\\textbf{" << firstRowLabel[firstRowLabel.size() - 1] << "}\\\\\\midrule[1pt]" << endl;
    95. }
    96. else {
    97. f1 << "\\textbf{Methods} & ";
    98. for (int i = 0; i < firstRowLabel.size() - 1; i++) {
    99. f1 << "\\multicolumn{" << subRowLabel.size() << "}{c}{\\textbf{" << firstRowLabel[i] << "}} &";
    100. }
    101. f1 << "\\multicolumn{" << subRowLabel.size() << "}{c}{\\textbf{" << firstRowLabel[firstRowLabel.size() - 1] << "}}\\\\" << endl;
    102. f1 << "\\textbf{Models} & ";
    103. for (int i = 0; i < firstRowLabel.size(); i++) {
    104. for (int j = 0; j < subRowLabel.size(); j++) {
    105. if (i == firstRowLabel.size() - 1 && j == subRowLabel.size() - 1) {
    106. f1 << subRowLabel[j] << " \\\\\\midrule[1pt] " << endl;
    107. }
    108. else {
    109. f1 << subRowLabel[j] << " & ";
    110. }
    111. }
    112. }
    113. }
    114. //2.写入数据
    115. if (subRowLabel.size() > 0) {
    116. for (int i = 0; i < data.size(); i++) {
    117. vector<float> data_i = data[i];
    118. vector<bool> label_i(data_i.size(), false);
    119. for (int j = 0; j < subRowLabel.size(); j++) {
    120. int laebl_ij_index_max = j;
    121. int laebl_ij_index_min = j;
    122. float extrmedata_max = data_i[j];
    123. float extrmedata_min = data_i[j];
    124. for (int k = j; k < data_i.size(); k = k + subRowLabel.size()) {
    125. if (data_i[k] > extrmedata_max) {
    126. extrmedata_max = data_i[k];
    127. laebl_ij_index_max = k;
    128. }
    129. if (data_i[k] < extrmedata_min) {
    130. extrmedata_min = data_i[k];
    131. laebl_ij_index_min = k;
    132. }
    133. }
    134. if (orderName == "Max") {
    135. label_i[laebl_ij_index_max] = true;
    136. }
    137. if (orderName == "Min") {
    138. label_i[laebl_ij_index_min] = true;
    139. }
    140. }
    141. f1 << "\\textbf{" << columnLabel[i] << "} & ";
    142. for (int j = 0; j < data_i.size() - 1; j++) {
    143. if (label_i[j]) {
    144. f1 << "\\textbf{"<"} & ";
    145. }
    146. else {
    147. f1 << data_i[j] << " & ";
    148. }
    149. }
    150. f1 << data_i[data_i.size() - 1] << "\\\\" << endl;
    151. }
    152. f1 << "\\bottomrule[1pt] " << endl;
    153. }
    154. else {
    155. for (int i = 0; i < data.size(); i++) {
    156. vector<float> data_i = data[i];
    157. int laebl_ij_index_max = 0;
    158. int laebl_ij_index_min = 0;
    159. float extrmedata_max = data_i[0];
    160. float extrmedata_min = data_i[0];
    161. for (int j = 0; j < data_i.size(); j++) {
    162. float data_ij = data_i[j];
    163. if (data_ij > extrmedata_max) {
    164. extrmedata_max = data_ij;
    165. laebl_ij_index_max = j;
    166. }
    167. if (data_ij < extrmedata_min) {
    168. extrmedata_min = data_ij;
    169. laebl_ij_index_min = j;
    170. }
    171. }
    172. int label_p;
    173. if (orderName == "Max") {
    174. label_p = laebl_ij_index_max;
    175. }
    176. if (orderName == "Min") {
    177. label_p = laebl_ij_index_min;
    178. }
    179. f1 << "\\textbf{" << columnLabel[i] << "} & ";
    180. for (int j = 0; j < data_i.size() - 1; j++) {
    181. if (j == label_p) {
    182. f1 << "\\textbf{" << data_i[j] << "} & ";
    183. }
    184. else {
    185. f1 << data_i[j] << " & ";
    186. }
    187. }
    188. if (label_p == data_i.size() - 1) {
    189. f1 << "\\textbf{" << data_i[data_i.size() - 1] << "}\\\\" << endl;
    190. }
    191. else {
    192. f1 << data_i[data_i.size() - 1] << "\\\\" << endl;
    193. }
    194. }
    195. f1 << "\\bottomrule[1pt]" << endl;
    196. }
    197. //3.收尾
    198. f1 << "\\end{tabular}"<
    199. f1 << "\\label{T1}" << endl;
    200. f1 << "\\end{table*}" << endl;
    201. f1 << endl;
    202. f1.close();
    203. }
    204. };

    调用LatexTable_init方法,传入txt文件的路径名字,就可以直接输出Latex表格代码。我写的比较粗糙,够我自己用的了。如果需要改数据类型,只要找到对应的位置,改变量即可。


    三.生成结果

    注:这里使用了包:\usepackage{booktabs}

    1. \begin{table*}[]
    2. \centering
    3. \caption{caption for table.}
    4. \begin{tabular}{c|ccccccccccccccc}\toprule[1pt]
    5. \textbf{Methods} & \multicolumn{3}{c}{\textbf{Method1}} &\multicolumn{3}{c}{\textbf{Method2}} &\multicolumn{3}{c}{\textbf{Method3}} &\multicolumn{3}{c}{\textbf{Method4}} &\multicolumn{3}{c}{\textbf{Method5}}\\
    6. \textbf{Models} & Q1 & Q2 & Q3 & Q1 & Q2 & Q3 & Q1 & Q2 & Q3 & Q1 & Q2 & Q3 & Q1 & Q2 & Q3 \\\midrule[1pt]
    7. \textbf{M1} & 1.2 & 1.1 & 1.4 & 5.6 & 5.3 & 5.6 & 2.2 & 3.1 & 0.4 & \textbf{15.6} & 25.3 & \textbf{45.6} & 11.2 & \textbf{51.1} & 21.4\\
    8. \textbf{M2} & 0.2 & 0.1 & 0.4 & 15.6 & 15.3 & 15.6 & \textbf{72.2} & 33.1 & 0.4 & 65.6 & 125.3 & 145.6 & 1.2 & \textbf{351.1} & 221.4\\
    9. \textbf{M3} & 21.1 & 31.1 & 51.4 & 75.6 & 95.3 & 25.6 & 22.2 & 23.1 & 20.4 & \textbf{215.6} & 225.3 & \textbf{245.6} & 211.2 & \textbf{251.1} & 221.4\\
    10. \bottomrule[1pt]
    11. \end{tabular}
    12. \label{T1}
    13. \end{table*}

    1. \begin{table*}[]
    2. \centering
    3. \caption{caption for table.}
    4. \resizebox{\textwidth}{!}{
    5. \begin{tabular}{c|ccccccccccccccc}\toprule[1pt]
    6. \textbf{Methods} & \textbf{Method1} &\textbf{Method2} &\textbf{Method3} &\textbf{Method4} &\textbf{Method5} &\textbf{Method6} &\textbf{Method7} &\textbf{Method8} &\textbf{Method9} &\textbf{Method10} &\textbf{Method11} &\textbf{Method12} &\textbf{Method13} &\textbf{Method14} &\textbf{Method15}\\\midrule[1pt]
    7. \textbf{M1} & 1.2 & 1.1 & 1.4 & 5.6 & 5.3 & 5.6 & 2.2 & 3.1 & 0.4 & 15.6 & 25.3 & 45.6 & 11.2 & \textbf{51.1} & 21.4\\
    8. \textbf{M2} & 0.2 & 0.1 & 0.4 & 15.6 & 15.3 & 15.6 & 72.2 & 33.1 & 0.4 & 65.6 & 125.3 & 145.6 & 1.2 & \textbf{351.1} & 221.4\\
    9. \textbf{M3} & 21.1 & 31.1 & 51.4 & 75.6 & 95.3 & 25.6 & 22.2 & 23.1 & 20.4 & 215.6 & 225.3 & 245.6 & 211.2 & \textbf{251.1} & 221.4\\
    10. \bottomrule[1pt]
    11. \end{tabular}}
    12. \label{T1}
    13. \end{table*}

  • 相关阅读:
    Halcon 双相机标定与拼图(一)
    安全组问题 访问华为云服务器端口
    热门项目披露:湖南民航置业有限公司70%股权转让
    光点数据可视化解决方案,助力新型智慧城市打造_光点科技
    java中PriorityQueue队列的使用
    最大的观影时间问题
    Hadoop生态系统官网、下载地址、文档
    动画效果:实现卡片翻转抽奖动画(正反面内容不一样)
    【附源码】计算机毕业设计JAVA抑抑心理交流平台
    【小沐学前端】Node.js实现基于Protobuf协议的WebSocket通信
  • 原文地址:https://blog.csdn.net/aliexken/article/details/126407673