[BJOI2019]光线——递推

Posted dummyummy

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[BJOI2019]光线——递推相关的知识,希望对你有一定的参考价值。

原题看我

题解

如果能把多块玻璃合并到一起多棒啊,但是直接把多块合并似乎不太可能,考虑两两合并
也就是已知上下层玻璃的透光率分别为\\(a_1\\%\\)\\(a_2\\%\\),反射率分别为\\(b_1\\%\\)\\(b_2\\%\\),求一块等效的玻璃,使得它和两块玻璃叠在一起的效果相同
(有一点要注意一下,就是合并后的玻璃上下的反射率和透光率不一定相同,我们需要根据合并的方向来判断要计算的是哪一个
也就是下面这两幅图,为了方便,将\\(x\\%\\)省略成\\(x\\)
1.技术图片
即合并后的透光率为\\(a_1a_2\\sum\\limits_{i=0}^{\\infty}(b_1b_2)^i=\\frac{a_1a_2}{1-b_1b_2}\\)
2.技术图片
即合并后的(向下的)反射率为\\(b_2+a_2^2b_1\\sum\\limits_{i=0}^{\\infty}(b_1b_2)^i=\\frac{a_2^2b_1}{1-b_1b_2}+b_2\\)
(P.S. 图中左下角的那个应该是\\(b_2\\))
这样我们就可以\\(O(n)\\)的递推了
代码如下:

#include <algorithm>
#include  <iostream>
#include   <cstdlib>
#include   <cstring>
#include    <cstdio>
#include    <string>
#include    <vector>
#include     <cmath>
#include     <ctime>
#include     <queue>
#include       <map>
#include       <set>

using namespace std;

#define ull unsigned long long
#define pii pair<int, int>
#define uint unsigned int
#define mii map<int, int>
#define lbd lower_bound
#define ubd upper_bound
#define INF 0x3f3f3f3f
#define IINF 0x3f3f3f3f3f3f3f3fLL
#define vi vector<int>
#define ll long long
#define mp make_pair
#define pb push_back
#define re register
#define il inline

#define MOD 1000000007

int n;
int p100;
int A, B;

int fpow(int x, int p) {
  int ret = 1;
  while(p) {
    if(p&1) ret = 1LL*ret*x%MOD;
    x = 1LL*x*x%MOD;
    p >>= 1;
  }
  return ret;
}

int read() {
  int t;
  scanf("%d", &t);
  return 1LL*t*p100%MOD;
}

int main() {
  scanf("%d", &n);
  p100 = fpow(100, MOD-2);
  A = read(), B = read();
  for(int i = 2, a, b, t; i <= n; ++i) {
    a = read(), b = read();
    t = fpow((1-1LL*B*b%MOD+MOD)%MOD, MOD-2);
    A = 1LL*A*a%MOD*t%MOD;
    B = (1LL*a*a%MOD*B%MOD*t%MOD+b)%MOD;
  }
  printf("%d\\n", A);
  return 0;
}

以上是关于[BJOI2019]光线——递推的主要内容,如果未能解决你的问题,请参考以下文章

[BJOI2019] 光线

#loj3093 [BJOI2019]光线

[BJOI2019]总结

#loj3090 [BJOI2019] 勘破神机

光线行进时的 GLSL 伪影

如何为光线投射生成相机光线