如何使用 redux 中的 axios 库为发布到 API 服务器的数据自动生成唯一 ID

Posted

技术标签:

【中文标题】如何使用 redux 中的 axios 库为发布到 API 服务器的数据自动生成唯一 ID【英文标题】:How to generate a unique id automatically for the data that is posted to an API server using axios library in redux 【发布时间】:2018-06-10 09:41:21 【问题描述】:

我有一个 redux-form,用于从用户接收数据并将其发布到后端 API 服务器。在将数据发布到服务器时,我想为每组数据生成一个唯一的 id被发布到服务器,以便我以后可以使用该 ID 来引用该特定集合并将其显示给用户。那么,如何在将一组数据发布到服务器时自动生成该唯一 ID?

我的 redux-form 的代码如下:

import React,  Component  from 'react';
import  Field, reduxForm  from 'redux-form';
import  Link  from 'react-router-dom';
import  connect  from 'react-redux';
import  createPosts  from '../actions/posts_action';

class CreatePost extends Component 
  constructor() 
    super();
    this.state = 
      selectValue : ''
  ;
  this.handleChange = this.handleChange.bind(this);
  this.renderCategory = this.renderCategory.bind(this);


  renderField(field) 
      return(
        <div className="title-design">
            <label className="label-design"> field.label </label>
            <input
              type="text"
              className="title-input"
              ...field.input
            />
            <div className="text-help  has-danger">
              field.meta.touched ? field.meta.error : ''
            </div>
      </div>
      );
  

  handleChange(e) 
    const value=e.target.value;
    this.props.change("categories",value);
    this.setState(selectValue: value, () => 
      console.log(value)
    );
  

  renderCategory(field) 
    return(
      <div className="title-design">
        <label className="label-design">field.label </label>
          <Field name="category" className="title-input" component="select">
            <option></option>
            <option value="react">React</option>
            <option value="redux">Redux</option>
            <option value="udacity">Udacity</option>
          </Field>

          <div className="text-help has-danger">
            field.meta.touched ? field.meta.error : ''
          </div>
      </div>
    );
  

    onSubmit(values) 
      this.props.createPosts(values, () => 
          this.props.history.push('/');
      );
    



    render() 
      const  handleSubmit  = this.props;

      return (
        <form onSubmit=handleSubmit(this.onSubmit.bind(this))>
          <Field
            label="Title for Post"
            name="title"
            component=this.renderField
          />

          <Field
            label="Post Content"
            name="body"
            component=this.renderField
          />

          <Field
            label="Category"
            name="category"
            component=this.renderCategory
            />



          <button type="submit" className="btn btn-primary">Submit</button>
          <Link  to="/">
            <button className="cancel-button">Cancel</button>
          </Link>
        </form>
      );
    


function validate(values) 
  const errors =  ;

  if (!values.title) 
      errors.title = "Enter a title";
  

  if (!values.body) 
    errors.body = "Enter some content";
    

  if(!values.category) 
    errors.category = "Please select a category";
  

  if (!values.id) 
      errors.id = "ID not generated";
  

  return errors;


export default reduxForm(
  validate : validate,          //validate
  form : 'CreatePostForm'
)(
  connect(null, createPosts )(CreatePost)
);

创建新帖子的动作创建者如下:

//Action Creator for creating posts
export function createPosts(values, callback) 

  return dispatch => 
    return axios.post(`$API/posts`,values,headers)
      .then((data) => 
        callback();
        console.log(data)
      dispatch(
        type: CREATE_POST,
        payload: data
      )
    )
  

Reducer 用于创建帖子:

import _ from 'lodash';
import  FETCH_POSTS, FETCH_POST, CREATE_POST  from '../actions/posts_action';

export default function(state = , action) 
  switch (action.type) 
    case FETCH_POST:
      // const post = action.payload.data;
      // const newState  =  ...state,  ;
      // newState[post.id] = post;
      // return newState;
      return ...state, [action.payload.id]: action.payload;

    case FETCH_POSTS:
     return posts:  ...state.posts, ...action.payload ;

    case CREATE_POST:
      return posts:  ...state, ...action.payload;

     default:
      return state;
  


那么,如何为发送到服务器的每组数据生成id?

编辑 1:

我正在尝试在我的 onSubmit() 方法中添加 uuid,如下所示,但它不起作用。我不知道应该如何完成,或者我必须以其他方式完成?

  onSubmit(values) 

          this.props.createPosts(values, () => 
            uuidv1();
          this.props.history.push('/');
      );
    

编辑 2:

帖子的服务器文件:

const clone = require('clone')

let db = 

const defaultData = 
  "8xf0y6ziyjabvozdd253nd": 
    id: '8xf0y6ziyjabvozdd253nd',
    timestamp: 1467166872634,
    title: 'Udacity is the best place to learn React',
    body: 'Everyone says so after all.',
    author: 'thingtwo',
    category: 'react',
    voteScore: 6,
    deleted: false,
    commentCount: 2
  ,
  "6ni6ok3ym7mf1p33lnez": 
    id: '6ni6ok3ym7mf1p33lnez',
    timestamp: 1468479767190,
    title: 'Learn Redux in 10 minutes!',
    body: 'Just kidding. It takes more than 10 minutes to learn technology.',
    author: 'thingone',
    category: 'redux',
    voteScore: -5,
    deleted: false,
    commentCount: 0
  


function getData (token) 
  let data = db[token]
  if (data == null) 
    data = db[token] = clone(defaultData)
  
  return data


function getByCategory (token, category) 
  return new Promise((res) => 
    let posts = getData(token)
    let keys = Object.keys(posts)
    let filtered_keys = keys.filter(key => posts[key].category === category && !posts[key].deleted)
    res(filtered_keys.map(key => posts[key]))
  )


function get (token, id) 
  return new Promise((res) => 
    const posts = getData(token)
    res(
      posts[id].deleted
        ? 
        : posts[id]
    )
  )


function getAll (token) 
  return new Promise((res) => 
    const posts = getData(token)
    let keys = Object.keys(posts)
    let filtered_keys = keys.filter(key => !posts[key].deleted)
    res(filtered_keys.map(key => posts[key]))
  )


function add (token, post) 
  return new Promise((res) => 
    let posts = getData(token)

    posts[post.id] = 
      id: post.id,
      timestamp: post.timestamp,
      title: post.title,
      body: post.body,
      author: post.author,
      category: post.category,
      voteScore: 1,
      deleted: false,
      commentCount: 0
    

    res(posts[post.id])
  )


function vote (token, id, option) 
  return new Promise((res) => 
    let posts = getData(token)
    post = posts[id]
    switch(option) 
        case "upVote":
            post.voteScore = post.voteScore + 1
            break
        case "downVote":
            post.voteScore = post.voteScore - 1
            break
        default:
            console.log(`posts.vote received incorrect parameter: $option`)
    
    res(post)
  )


function disable (token, id) 
    return new Promise((res) => 
      let posts = getData(token)
      posts[id].deleted = true
      res(posts[id])
    )


function edit (token, id, post) 
    return new Promise((res) => 
        let posts = getData(token)
        for (prop in post) 
            posts[id][prop] = post[prop]
        
        res(posts[id])
    )


function incrementCommentCounter(token, id, count) 
  const data = getData(token)
  if (data[id]) 
    data[id].commentCount += count
  


module.exports = 
  get,
  getAll,
  getByCategory,
  add,
  vote,
  disable,
  edit,
  getAll,
  incrementCommentCounter

【问题讨论】:

一般由服务器负责生成唯一的 id 并将其与响应数据一起传递 @ShubhamKhatri 当我尝试创建新帖子时,我没有看到帖子的任何“id”属性。我在编辑中粘贴了上面的服务器文件。你能看看它。 【参考方案1】:

好吧,通常您会让服务器生成该 ID,然后将其作为 axios 调用响应的一部分发送回去,但如果您真的想在客户端上生成 ID,我建议使用 uuid 和一个它的各种 UUID 生成器: https://www.npmjs.com/package/uuid

文档中的示例:

const uuidv4 = require('uuid/v4');
uuidv4(); // -> '110ec58a-a0f2-4ac4-8393-c866d813b8d1' 

您添加了更多代码,我认为在您的服务器上使用 uuidv4() 会更好地为您服务;类似:

function add (token, post) 
  return new Promise((res) => 
    let posts = getData(token)
    let id = uuidv4();

    posts[id] = 
      id: id,
      timestamp: post.timestamp,
      title: post.title,
      body: post.body,
      author: post.author,
      category: post.category,
      voteScore: 1,
      deleted: false,
      commentCount: 0
    

    res(posts[id])
  )

这样,id就在服务器上生成了。客户端不再需要依赖于 uuid,并且您不会“信任”客户端生成有效的 id。 id 将作为响应的一部分呈现给客户端,然后您可以在进行编辑等时从客户端使用它。

【讨论】:

感谢您的回复。让我试试看。几分钟后会更新您的结果。 我在导入后尝试在我的onSubmit方法中添加uuidv1(),但我没有看到它添加了。我在上面的编辑中添加了代码。你能看一下吗并告诉我如何进行? 我以为你想提交身份证?您只是在调用该方法,而不是使用生成的 ID 值。您可能希望将 uuidv1() 的结果添加到被推送的值中,以便服务器可以将其作为结果的一部分立即发送回来。 “值[id] = uuidv1();”在进行 axios 调用之前。 @Rashomon 是的,有......每次你的 redux 代码重新启动时(当你重新部署时,或其他),你的计数器回到 0,你得到你上次创建的重复时间。 @Rashomon 我有点假设生成 ID 的主要原因是将对象存储在某种数据库中,以便以后能够引用它。如果它只是一个临时标识符(如会话),那么您的计数器将起作用。毕竟,这就是 DB 序列的作用。

以上是关于如何使用 redux 中的 axios 库为发布到 API 服务器的数据自动生成唯一 ID的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 redux 和 redux-promise 将多个 axios 调用合并到一个有效负载中?

如何将 axios 添加到 react-redux-universal-hot-example 入门套件?

如何从服务器端 express/mongo/mongoose 中的 URL,客户端的 axios/react/redux 中获取参数 ID

如何使用 redux-axios-middleware 测试 Redux 异步操作

如何在反应中使用redux取消以前的axios

使用 axios 和 redux-thunk 以 redux-form 发布请求