ssr

  1. 服务端不能识别前端的window。特别是首屏渲染的数据需要用到window对象(比如href += location.search);
  2. 服务端不能加载图片,css文件。
require.extensions['.less'] = function () {
    return null;
};
global.__DEV__ = !bytedEnv.isProd();
global.__SERVER__ = true; // 代码服务端环境
global.window = {
    location: {
        search: ''
    }
};
//做了个中间件事实修改window的属性。
module.exports = () => async (ctx, next) => {
  global.window = {
    location: {
      search: ctx.search,
      host: ctx.host,
      protocol: ctx.protocol
    }
  };
  await next();
};

客户端需要全局变量来表示是否是客户端环境来选择加载某些库。

  1. 首屏从localstorage取数据,并不能服务端渲染
    isSsr ? <div className="channel-bar">: <ChannelBar />

这里用了hydrate而不是render渲染react组件,导致首刷的dom被尽可能的复用。
导致的问题是ChannelBar的最外层dom被复用。如果服务端渲染为空div占位的话,上面的class channel-bar就会丢失。

所以首屏渲染最好还是要保证首刷客户端和服务端一样才行。
这种问题官方推荐的做法是在react生命周期里渲染首刷不同的部分。不过这种做法必然会导致组件rerender。

解决方案:ChannelBar的useEffect里去localstorage拿数据,这样保证首刷一致。

react 16 hooks问题

function OnePxBorder({ color }) {

  const style = {
    backgroundColor: color,
    height: '1px'
  };
  return <div className="one-px" style={style} />;
}

对于这样一个组件,const style语句等语句(定义函数等)会在组件rerender的时候重新执行。这样会导致定义在组件内的函数多次定义。

解决方案:无关state的函数定义在react组件外。

function Channel() {
  // Declare a new state variable, which we'll call "count"
  const [count, setCount] = useState(0);

  useEffect(() => {
    setCount(1);
  });
  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>
        Click me
      </button>
    </div>
  );
}

useEffect会在每次render的时候重新执行,以上代码,count再点也是1;
useEffect第二个参数传空数组可保证函数只执行一遍。

 useEffect(() => {
    setCount(1);
  }, []);

这样实现初始化为1的效果。

function Channel() {
  const [count, setCount] = useState(0);

  useEffect(() => {
    console.log('bind')
    const timer = setInterval(() => {
     setCount(count + 1);
    }, 1000);

    return () => {
      console.log('re bind')
      clearInterval(timer);
    };
  });
  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>
        Click me
      </button>
    </div>
  );
}

如前,可以看到setInterval被触发,被解绑,被触发,再解绑。
如果加上useEffect(() => {}, []);
会导致count 永远是1。
原因是useEffect如果不重新执行,拿不到最新的state。拿到的state总是最早的state也就是0。

暂时没有好的方法。现在写为useEffect(() => {}, [count]);
如果遇到只能初始化一次的,比如某些插件。可以用事件系统每次拿最新的函数去执行:

import { EventEmitter } from 'events';
const event = new EventEmitter();

 new xxx({ // 此插件只运行一次。回调总执行新的doSomething
    onFinish: () => {
      event.emit('some_event');
    }
  });

function Components({ doSomething }) {
 useEffect(() => {
      event.on('some_event', doSomething);
      return () => {
        event.removeListener('some_event', doSomething);
      };
  });
}

render函数被重复执行

有这样一个组件,loading是个state,每次setloadding会导致list重新执行。
listEl需要数据处理,渲染。每次setloadding,就会处理一遍数据。

{ listEl }
{
   loading && <div>'loading.....'</div>
}

处理方案:useMemo Hooks
useMemo能让我们保存一段dom。

const createMemo = function() {
     return useMemo(() => {
        // handle data
     }, [data.length])
}

这样使得data.length变了之后useMemo hooks里的代码才会重新执行。
useMemo的第二个参数是用Object.is判断的,不能直接写个Object,react不会帮你deep assert。

react 16

如官网所说:react16需要set和map环境。而import的包不会被babel编译,所以需要手动引入map/set

import 'core-js/es6/map';
import 'core-js/es6/set';

import React from 'react';
import ReactDOM from 'react-dom';

ReactDOM.render(
  <h1>Hello, world!</h1>,
  document.getElementById('root')
);

React 16 depends on the collection types Map and Set. If you support older browsers and devices which may not yet provide these natively (e.g. IE < 11) or which have non-compliant implementations (e.g. IE 11), consider including a global polyfill in your bundled application, such as core-js or babel-polyfill.

服务端http请求

服务端请求不带header。需要把header带上。
首次请求set到客户端的cookie(用户标识),在服务端首刷请求的时候并不能取得。需要手动修改设置cookie。

 if (!reqestHeader.cookie) {
    reqestHeader.cookie = `uid=${ctx.u_id}; utm_source=${utmSource}`;
}

css module问题

css全局作用域。不想用插件。
靠规范解决。

编译

gulu要求目录文件是app,需要编译要按gulu目录来。
prod环境找问题比较麻烦,对gulu这种框架,需要看他源码来找,没得文档。

react 16 ssr的重构踩坑的更多相关文章

  1. React Native Android配置部署踩坑日记

    万事开头难 作为一只进入ECMAScript世界不久的菜鸟,已经被React Native的名气惊到了,开源一周数万星勾起了我浓烈的兴趣.新年新气象,来个HellWorld压压惊吧^_^(故意少打个' ...

  2. ant.design React使用Echarts,实力踩坑

    最近项目用到Echarts(以下用ec代替),于是照猫画虎得引入到团队的antd项目中,但是遇到2个棘手问题: 1. ec对dom不渲染,检查后发现,原来是全局存在id重复,所以使用React时,最好 ...

  3. React 16 升级时遇到的一个坑,分享一下

    遇到的坑 今天在跟着dva.js官网上面的一个教程写东西的时候,照着教程上面的代码写之后,运行总是报错:TypeError: Cannot read property 'object' of unde ...

  4. react中的路由配置踩坑记

    react 路由配置中,如果根路由(/)匹配一个组件,另一个路由(/list)在进行匹配的时候也会匹配到根路由(/),即在 /list 页面展示的时候 / 页面总是展示在上方. 此时如果想进行严格匹配 ...

  5. 使用 jest 测试 react component 的配置,踩坑。

    首先安装依赖 npm i jest -g npm i jest babel-jest identity-obj-proxy enzyme enzyme-adapter-react-15.4 react ...

  6. 利用vue-router和compoment重构代码--踩坑(一)

    业务主要功能 获取所有的数据库列表 点击某一个数据库列表的时候,右侧分页展示数据 点击右侧某一条数据的时候,现实数据详情 以下是之前的页面,存在以下问题: 前段开发没有工程化(webpack) 主要功 ...

  7. React Native踩坑日记 —— tailwind-rn

    项目背景 在项目的初始阶段,我们需要建立自己的design system,我们spike了一些方案,tailwind-rn就是其中一种,如果有用到或者即将用到tailwind-rn的,可以进来看一看, ...

  8. React Native踩坑Tip

    最近在使用React Native(以下简称RN)中踩了个坑,RN只能异步调用原生方法,所以在原生方法直接调用UI刷新操作需要将任务递交到主线程才可以. RCT_EXPORT_METHOD(finis ...

  9. Ubuntu 16.04 安装Mysql 5.7 踩坑小记

    title:Ubuntu 16.04 安装Mysql 5.7 踩坑小记 date: 2018.02.03 安装mysql sudo apt-get install mysql-server mysql ...

随机推荐

  1. 初学Shiro

    Shiro Shiro是什么? Apache Shiro是Java的一个安全(权限)框架. Shiro可以非常容易的开发出足够好的应用,其不仅可以用在JavaSE环境下,也可以用者JavaEE环境下 ...

  2. MongoDB学习(操作集合中的文档)

    文档概念 文档的数据结构和JSON基本一样. 所有存储在集合中的数据都是BSON格式. BSON是一种类json的一种二进制形式的存储格式,简称Binary JSON. 插入文档 insert()方法 ...

  3. java基础(三):谈谈java异常的处理

    1.知识点总结 1.1.异常分类 异常就是java中出现的不正常的现象(错误与异常),按照继承的体系结构,可以分类如下 Throwable: 它是所有错误与异常的超类(祖宗类) |- Error 错误 ...

  4. Android studio怎么使用自定义的framework而避免冲突报错和点不进去报红。

    文件:xx\project_abc\video\build.gradle保证可以运行到自定义的framework而不报错,可能因为project和module名字相同所以导致下面的路径是绝对路径,其他 ...

  5. 我们距离AI编程还有多远?

    近几年来,人工智能的信息以不同形式霸占着我们的眼球,我们知道AlphaGo.微软小冰.Sophia,了解过自动驾驶.无人机.智能家居等,深知人工智能是在记忆力.学习力.运算力方面都远超人类的存在,但人 ...

  6. C#中USB转串口的拔插捕获

    // usb消息定义 public const int WM_DEVICE_CHANGE = 0x219; public const int DBT_DEVICEARRIVAL = 0x8000; p ...

  7. Neo4j 全文检索

    全文检索基本概念 搜索 搜索这个行为是用户与搜索引擎的一次交互过程,用户需要找一些数据,他提供给搜索引擎一些约束条件.搜索引擎通过约束条件抽取一些结果给用户 搜索引擎 搜索引擎存在的目的是存储,查找和 ...

  8. UNIX DOMAIN SOCKET效率

    关于UNIX DOMAIN SOCKET和普通udp socket的对比 在TX1(4核A57 1.7GHz)的板卡上进行测试,每个包大小设置为1024,全速收发,UDS的速度在90Mbps左右,UD ...

  9. zabbix利用SNMPTrap接收交换机主动告警

    zabbix接收trap的工作流程: snmptrapd 收到trap snmptrapd将trap传递给SNMPTT或调用Perl接收器 SNMPTT或Perl trap接收器解析,格式化并将tra ...

  10. Java基础系列--02_运算符和程序的语句

    运算符: (1)算术运算符: +,-,*,/,%,++,--(加.减.乘.除.取余.自增,自减) ++和--的注意事项: a:他们的作用是自增或者自减 b:使用 1.单独使用 放在操作数据的前面和后面 ...