digital root = n==0 ? 0 : n%9==0 ? 9:n%9;
可以简单证明一下
n = a0*n^0 + a1*n^1 + ... + ak * n^k
n%9 = a0+a1+..+ak
然后,数学归纳易知结论是正确的。
因此9个状态就够了,表示%9的结果。
这里需要特殊处理0, 表示状态为0。
1 /* 4351 */ 2 #include <iostream> 3 #include <sstream> 4 #include <string> 5 #include <map> 6 #include <queue> 7 #include <set> 8 #include <stack> 9 #include <vector> 10 #include <deque> 11 #include <algorithm> 12 #include <cstdio> 13 #include <cmath> 14 #include <ctime> 15 #include <cstring> 16 #include <climits> 17 #include <cctype> 18 #include <cassert> 19 #include <functional> 20 #include <iterator> 21 #include <iomanip> 22 using namespace std; 23 //#pragma comment(linker,"/STACK:102400000,1024000") 24 25 #define sti set<int> 26 #define stpii set<pair<int, int> > 27 #define mpii map<int,int> 28 #define vi vector<int> 29 #define pii pair<int,int> 30 #define vpii vector<pair<int,int> > 31 #define rep(i, a, n) for (int i=a;i<n;++i) 32 #define per(i, a, n) for (int i=n-1;i>=a;--i) 33 #define clr clear 34 #define pb push_back 35 #define mp make_pair 36 #define fir first 37 #define sec second 38 #define all(x) (x).begin(),(x).end() 39 #define SZ(x) ((int)(x).size()) 40 #define lson l, mid, rt<<1 41 #define rson mid+1, r, rt<<1|1 42 43 typedef struct { 44 short sum; 45 short lst, rst, st; 46 } node_t; 47 48 const int maxn = 1e5+5; 49 int zn[maxn]; 50 bool visit[10]; 51 int a[maxn]; 52 node_t nd[maxn<<2]; 53 int Mod9[105]; 54 short sum_st[20][512]; 55 short st_st[512][512]; 56 int n; 57 58 void init() { 59 rep(i, 0, 105) 60 Mod9[i] = i % 9; 61 62 rep(st, 0, 512) { 63 rep(i, 0, 20) { 64 int nst = 0; 65 66 rep(j, 0, 9) { 67 if (st & (1<<j)) { 68 int k = Mod9[i + j]; 69 nst |= (1<<k); 70 } 71 } 72 sum_st[i][st] = nst; 73 } 74 } 75 76 rep(lst, 0, 512) { 77 rep(rst, lst, 512) { 78 int nst = 0; 79 rep(i, 0, 9) { 80 if ((lst & (1<<i)) == 0) 81 continue; 82 rep(j, 0, 9) { 83 if (rst & (1<<j)) { 84 int k = Mod9[i+j]; 85 nst |= (1<<k); 86 } 87 } 88 } 89 st_st[lst][rst] = st_st[rst][lst] = nst; 90 } 91 } 92 } 93 94 95 void PushUp(int rt) { 96 int lb = rt << 1; 97 int rb = lb | 1; 98 99 nd[rt].sum = Mod9[nd[lb].sum + nd[rb].sum]; 100 nd[rt].lst = nd[lb].lst | sum_st[nd[lb].sum][nd[rb].lst]; 101 nd[rt].rst = nd[rb].rst | sum_st[nd[rb].sum][nd[lb].rst]; 102 nd[rt].st = nd[lb].st | nd[rb].st | st_st[nd[lb].rst][nd[rb].lst]; 103 } 104 105 void Build(int l, int r, int rt) { 106 if (l == r) { 107 if (zn[l] == zn[l-1]) { 108 nd[rt].sum = a[l]; 109 nd[rt].lst = nd[rt].rst = nd[rt].st = (1 << a[l]); 110 } else { 111 nd[rt].sum = a[l]; 112 nd[rt].lst = nd[rt].rst = nd[rt].st = 0; 113 } 114 return ; 115 } 116 117 int mid = (l + r) >> 1; 118 119 Build(lson); 120 Build(rson); 121 122 PushUp(rt); 123 } 124 125 node_t Query(int L, int R, int l, int r, int rt) { 126 if (L==l && R==r) { 127 return nd[rt]; 128 } 129 130 int mid = (l + r) >> 1; 131 132 if (R <= mid) { 133 return Query(L, R, lson); 134 } else if (L > mid) { 135 return Query(L, R, rson); 136 } else { 137 node_t lnd = Query(L, mid, lson); 138 node_t rnd = Query(mid+1, R, rson); 139 node_t ret; 140 141 ret.sum = Mod9[lnd.sum + rnd.sum]; 142 ret.lst = lnd.lst | sum_st[lnd.sum][rnd.lst]; 143 ret.rst = rnd.rst | sum_st[rnd.sum][lnd.rst]; 144 ret.st = lnd.st | rnd.st | st_st[lnd.rst][rnd.lst]; 145 146 return ret; 147 } 148 } 149 150 void solve() { 151 Build(1, n, 1); 152 153 int q; 154 node_t d; 155 int l, r; 156 157 scanf("%d", &q); 158 while (q--) { 159 scanf("%d %d", &l, &r); 160 memset(visit, false, sizeof(visit)); 161 d = Query(l, r, 1, n, 1); 162 rep(i, 0, 9) { 163 if (d.st & (1<<i)) 164 visit[i] = true; 165 } 166 visit[9] = visit[0]; 167 visit[0] = (zn[r] - zn[l-1]) > 0; 168 int cnt = 5; 169 per(i, 0, 10) { 170 if (visit[i]) { 171 if (cnt == 5) 172 printf("%d", i); 173 else 174 printf(" %d", i); 175 if (--cnt == 0) 176 break; 177 } 178 } 179 180 while (cnt) { 181 printf(" -1"); 182 --cnt; 183 } 184 putchar(' '); 185 } 186 } 187 188 int main() { 189 ios::sync_with_stdio(false); 190 #ifndef ONLINE_JUDGE 191 freopen("data.in", "r", stdin); 192 freopen("data.out", "w", stdout); 193 #endif 194 195 int t; 196 197 init(); 198 scanf("%d", &t); 199 rep(tt, 1, t+1) { 200 scanf("%d", &n); 201 rep(i, 1, n+1) { 202 scanf("%d", &a[i]); 203 zn[i] = zn[i-1] + (a[i] == 0); 204 a[i] %= 9; 205 } 206 printf("Case #%d: ", tt); 207 solve(); 208 if (tt != t) 209 putchar(' '); 210 } 211 212 #ifndef ONLINE_JUDGE 213 printf("time = %d. ", (int)clock()); 214 #endif 215 216 return 0; 217 }
数据发生器。
1 from copy import deepcopy 2 from random import randint, shuffle 3 import shutil 4 import string 5 6 7 def GenDataIn(): 8 with open("data.in", "w") as fout: 9 t = 10 10 bound = 10**9 11 fout.write("%d " % (t)) 12 for tt in xrange(t): 13 n = randint(10, 20) 14 fout.write("%d " % (n)) 15 L = [] 16 for i in xrange(1, n): 17 x = randint(0, bound) 18 L.append(x) 19 L.append(0) 20 shuffle(L) 21 fout.write(" ".join(map(str, L)) + " ") 22 q = (n+1)*n/2 23 fout.write("%d " % (q)) 24 for l in xrange(1, n+1): 25 for r in xrange(l, n+1): 26 fout.write("%d %d " % (l, r)) 27 28 29 def MovDataIn(): 30 desFileName = "F:eclipse_prjworkspacehdojdata.in" 31 shutil.copyfile("data.in", desFileName) 32 33 34 if __name__ == "__main__": 35 GenDataIn() 36 MovDataIn()