剥开比原看代码13:比原是如何通过/list-balances显示帐户余额的?

简介: 作者:freewind比原项目仓库:Github地址:https://github.com/Bytom/bytomGitee地址:https://gitee.com/BytomBlockchain/bytom在前几篇里,我们研究了比原是如何通过web api接口来创建密钥、帐户和地址的,今天我们继续看一下,比原是如何显示帐户余额的。

作者:freewind

比原项目仓库:

Github地址:https://github.com/Bytom/bytom

Gitee地址:https://gitee.com/BytomBlockchain/bytom

在前几篇里,我们研究了比原是如何通过web api接口来创建密钥、帐户和地址的,今天我们继续看一下,比原是如何显示帐户余额的。

在Dashboard中,左侧有一栏名为"Balances"(余额),点击后,我们可以看到每个帐户当前有多少余额,如下图:

img_e6d69794da1a5a72d02fd92364d0b7d8.png

这又是怎么实现的呢?我们还是和以前一样,把它分成两个部分:

  1. 前端是如何向后端发送请求的
  2. 后端接收到请求数据后,是如何去查询出帐户余额的

前端是如何向后端发送请求的

对应这个功能的前端代码远比想像中复杂,我花了很多功夫才把逻辑理清楚,主要原因是它是一种通用的展示方式:以表格的形式来展示一个数组中多个元素的内容。不过在上图所展示的例子中,这个数组只有一个元素而已。

首先需要提醒的是,这里涉及到ReduxRedux-router的很多知识,如果不熟悉的话,最好能先去找点文档和例子看看,把里面的一些基本概念弄清楚。比如,

  1. 在Redux中,通常会有一个叫store的数据结构,像一个巨大的JSON对象,持有整个应用所有需要的数据;
  2. 我们需要写很多reducer,它们就是store的转换器,根据当前传入的store返回一个新的内容不同的store,store在不同时刻的内容可以看作不同的state
  3. action是用来向reducer传递数据的,reducer将根据action的类型和参数来做不同的转换
  4. dispatch是Redux提供的,我们一般不能直接调用reducer,而是调用dispatch,把action传给它,它会帮我们拿到当前的store,并且把它(或者一部分)和action一起传给reducer去做转换
  5. redux-router会提供一个reduxConnect函数,帮我们把store跟react的组件连接起来,使得我们在React组件中,可以方便的去dispatch

另外,在Chrome中,有两个插件可以方便我们去调试React+Redux:

  1. React DevTools: https://chrome.google.com/webstore/detail/react-developer-tools/fmkadmapgofadopljbjfkapdkoienihi
  2. Redux DevTools: https://chrome.google.com/webstore/detail/redux-devtools/lmhkpmbekcpmknklioeibfkpmmfibljd

下面将结合前端源代码来分析一下过程,因为在逻辑上可以看作存在几条线,所以我们将分开追踪。

reducers

首先我们发现在启动的地方,初始化了store

src/app.js#L17-L18

// Start app
export const store = configureStore()

并且在里面创建store的时候,还创建了reducer:

src/configureStore.js#L13-L37

export default function() {
  const store = createStore(
    makeRootReducer(),
    ...

  return store
}

进入makeRootReducer

src/reducers.js#L18-L62

// ...
import { reducers as balance } from 'features/balances'
// ...
const makeRootReducer = () => (state, action) => {
  // ...
  return combineReducers({
    // ...
    balance,
    // ...
  })(state, action)
}

这个函数的最后实际上会把多个组件需要的reducer合并在一起,但是我把其它的都省略了,只留下了今天要研究的balance

而这个balance是来自于'features/balances'暴露出来的reducers

src/features/balances/index.js#L5-L9

import reducers from './reducers'

export {
  actions,
  reducers,
  routes
}

可以看到除了reducers,它还暴露了别的,那些我们一会儿再研究。先看reducers,它对应于reducers.js

src/features/balances/reducers.js#L30-L33

export default combineReducers({
  items: itemsReducer,
  queries: queriesReducer
})

可以看到,它是把两种作用的reducer合并起来了,一个是跟操作元素相关的,另一个是用来记录查询状态的(是否查询过)。

我们先看元素相关的itemsReducer

src/features/balances/reducers.js#L3-L17

const itemsReducer = (state = {}, action) => {
  if (action.type == 'APPEND_BALANCE_PAGE') {
    const newState = {}
    action.param.data.forEach((item, index) => {
      const id = `balance-${index}`
      newState[id] = {
        id: `balance-${index}`,
        ...item
      }
    })

    return newState
  }
  return state
}

可以看到,当传过来的参数actiontypeAPPEND_BALANCE_PAGE时,就会把action.param.data中包含的元素放到一个新创建的state中,并且以索引顺序给它们起了id,且在id前面加了balance-方便追踪。比如我们在Chrome的Redux DevTools插件中就可以看到:

img_aa1cf0da9c0d20576b18c518df463fc1.png

经过这个reducer处理后产生的新store中就包含了与balance相关的数据,它们可以用于在别处拿出来显示在React组件中。这点我们在后面会看到。

再看另一个与查询相关的queriesReducer

src/features/balances/reducers.js#L19-L27

const queriesReducer = (state = {}, action) => {
  if (action.type == 'APPEND_BALANCE_PAGE') {
    return {
      loadedOnce: true
    }
  }
  return state
}

这个比较简单,它关心的action.type跟前面一样,也是APPEND_BALANCE_PAGE。返回的loadedOnce的作用是告诉前端有没有向后台查询过,这样可以用于控制比如提示信息的显示等。

与balance相关的reducer就只有这些了,看起来还是比较简单的。

actions

在前面,我们看到在balance中除了reducer,还定义了actions:

src/features/balances/index.js#L5-L9

import actions from './actions'
// ...

export {
  actions,
  reducers,
  routes
}

其中的actions对应的是actions.js

src/features/balances/actions.js#L1-L2

import { baseListActions } from 'features/shared/actions'
export default baseListActions('balance')

可以看到,它实际上是利用了一个项目内共享的action来产生自己的action,让我们找到baseListActions

src/features/shared/actions/index.js#L1-L9

// ...
import baseListActions from './list'

export {
  // ...
  baseListActions,
}

继续,先让我们省略掉一些代码,看看骨架:

src/features/shared/actions/list.js#L4-L147

// 1. 
export default function(type, options = {}) {

  // 2. 
  const listPath  = options.listPath || `/${type}s`

  // 3. 
  const clientApi = () => options.clientApi ? options.clientApi() : chainClient()[`${type}s`]

  // 4. 
  const fetchItems = (params) => {
    // ...
  }

  const fetchPage = (query, pageNumber = 1, options = {}) => {
    // ...
  }

  const fetchAll = () => {
    // ...
  }

  const _load = function(query = {}, list = {}, requestOptions) {
    // ...
  }

  const deleteItem = (id, confirmMessage, deleteMessage) => {
    // ...
  }

  const pushList = (query = {}, pageNumber, options = {}) => {
    // ...
  }

  // 5.
  return {
    fetchItems,
    fetchPage,
    fetchAll,
    deleteItem,
    pushList,
    didLoadAutocomplete: {
      type: `DID_LOAD_${type.toUpperCase()}_AUTOCOMPLETE`
    },
  }
}

这个函数比较大,它是一个通用的用来分页分元素来展示数据的。为了方便理解,我们先把一些细节代码注释掉了,只留下了骨架,并且标注了6块内容:

  1. 第1处需要关注的是,这是一个函数,可以被外界调用,所以前面才可以baseListActions('balance'),传进来的第一个参数是用来表示这是什么类型的数据,其它地方可以根据这个类型发送不同的请求或进行不同的操作
  2. 第2处是定义前台列出数据(就是常用的list页面)的router路径,默认就type的复数,比如balance就是/balances,它会被redux-router处理,并且转到相应的组件
  3. 第3处是找到相应的用于向后台传送数据的对象,名为clientApi,封装了后台提供的web api接口
  4. 第4处是与显示数据相关的通用函数定义,比如取数据,按页取,删除等
  5. 第5处是把前面定义的各种操作函数组合成一个对象,返回给调用者

其实我觉得这些函数的细节在这里都不用怎么展示,因为在代码分析的时候,难度不在一个具体的函数是怎么实现的,而是在于骨架和流程是怎么样的。这里列出了多个函数的名字,我还不清楚哪些会用到,所以先不讲解,等后面遇到了再把代码贴出来讲解。

routes

再看前面剩下的routes是怎么实现的:

src/features/balances/index.js#L5-L9

// ...
import routes from './routes'

export {
  actions,
  reducers,
  routes
}

这个routes对应的是routes.js文件:

src/features/balances/routes.js#L1-L4

import { List } from './components'
import { makeRoutes } from 'features/shared'

export default (store) => makeRoutes(store, 'balance', List)

跟前面的action类似,它也是通过调用一个通用的函数再传入一些具体的参数过去实现的,那么在那边的makeRoutes肯定做了大量的工作。让我们进入features/shared/index.js

src/features/shared/index.js#L1-L9

// ...
import makeRoutes from './routes'
// ...

export {
  actions,
  reducers,
  makeRoutes
}

只聚焦于makeRoutes

src/features/shared/routes.js#L5-L44

const makeRoutes = (store, type, List, New, Show, options = {}) => {
  // 1. 
  const loadPage = () => {
    store.dispatch(actions[type].fetchAll())
  }

  // 2.
  const childRoutes = []

  if (New) {
    childRoutes.push({
      path: 'create',
      component: New
    })
  }

  if (options.childRoutes) {
    childRoutes.push(...options.childRoutes)
  }

  if (Show) {
    childRoutes.push({
      path: ':id',
      component: Show
    })
  }

  // 3.
  return {
    path: options.path || type + 's',
    component: RoutingContainer,
    name: options.name || humanize(type + 's'),
    name_zh: options.name_zh,
    indexRoute: {
      component: List,
      onEnter: (nextState, replace) => {
        loadPage(nextState, replace)
      },
      onChange: (_, nextState, replace) => { loadPage(nextState, replace) }
    },
    childRoutes: childRoutes
  }
}

分成了4块:

  1. 第1处定义了loadPage的操作,它实际上要是调用该type对应的action的fetchAll方法(还记得前面action骨架中定义了fetchAll函数吗)
  2. 第2处根据传入的参数来确定这个router里到底有哪些routes,比如是否需要“新建”,“显示”等等
  3. 第3处就是返回值,返回了一个对象,它是可以被redux-router理解的。可以看到它里面有path, 对应的组件component,甚至首页中某些特别时刻如进入或者改变时,要进行什么操作。

由于这里调用了fetchAll,那我们便把前面action里的fetchAll贴出来:

src/features/shared/actions/list.js#L58-L60

  const fetchAll = () => {
    return fetchPage('', -1)
  }

又调用到了fetchPage

src/features/shared/actions/list.js#L39-L55

  const fetchPage = (query, pageNumber = 1, options = {}) => {
    const listId =  query.filter || ''
    pageNumber = parseInt(pageNumber || 1)

    return (dispatch, getState) => {
      const getFilterStore = () => getState()[type].queries[listId] || {}

      const fetchNextPage = () =>
        dispatch(_load(query, getFilterStore(), options)).then((resp) => {
          if (!resp || resp.type == 'ERROR') return

          return Promise.resolve(resp)
        })

      return dispatch(fetchNextPage)
    }
  }

在中间又调用了_load

src/features/shared/actions/list.js#L62-L101

  const _load = function(query = {}, list = {}, requestOptions) {
    return function(dispatch) {
      // ...
      // 1.
      if (!refresh && latestResponse) {
        let responsePage
        promise = latestResponse.nextPage()
          .then(resp => {
            responsePage = resp
            return dispatch(receive(responsePage))
          })
          // ...
      } else {
        // 2. 
        const params = {}
        if (query.filter) params.filter = filter
        if (query.sumBy) params.sumBy = query.sumBy.split(',')
        promise = dispatch(fetchItems(params))
      }

      // 3. 
      return promise.then((response) => {
        return dispatch({
          type: `APPEND_${type.toUpperCase()}_PAGE`,
          param: response,
          refresh: refresh,
        })
      })
      // ...
    }
  }

这个函数还比较复杂,我进行了适当简化,并且分成了3块:

  1. 第1处的if分支处理的是第2页的情况。拿到数据后,会通过receive这个函数定义了一个action传给dispatch进行操作。这个receive在前面被我省略了,其实就是定义了一个typeRECEIVED_${type.toUpperCase()}_ITEMS的action,也就是说,拿到数据后,还需要有另一个地方对它进行处理。我们晚点再来讨论它。
  2. 第2处的else处理的是查询情况,拿到其中的过滤条件等,传给fetchItems函数
  3. 第3处的promise就是前面两处中的一个,也就是拿到数据后再进行APPEND_${type.toUpperCase()}_PAGE的操作

我们从这里并没有看到它到底会向比原后台的哪个接口发送请求,它可能被隐藏在了某个函数中,比如nextPage或者fetchItems等。我们先看看nextPage:

src/sdk/page.js#L17-L24

  nextPage(cb) {
    let queryOwner = this.client
    this.memberPath.split('.').forEach((member) => {
      queryOwner = queryOwner[member]
    })

    return queryOwner.query(this.next, cb)
  }

可以看到它最后调用的是clientquery方法。其中的client对应的是balanceAPI

src/sdk/api/balances.js#L3-L9

const balancesAPI = (client) => {
  return {
    query: (params, cb) => shared.query(client, 'balances', '/list-balances', params, {cb}),

    queryAll: (params, processor, cb) => shared.queryAll(client, 'balances', params, processor, cb),
  }
}

可以看到,query最后将调用后台的/list-balances接口。

fetchItems最终也调用的是同样的方法:

src/features/shared/actions/list.js#L15-L35

const fetchItems = (params) => {
    // ...
    return (dispatch) => {
      const promise = clientApi().query(params)

      promise.then(
        // ...
      )

      return promise
    }
  }

所以我们一会儿在分析后台的时候,只需要关注/list-balances就可以了。

这里还剩下一点,就是从后台拿到数据后,前端怎么处理,也就是前面第1块和第3块中拿到数据后的操作。

我们先看一下第1处中的RECEIVED_${type.toUpperCase()}_ITEMS的action是如何被处理的。通过搜索,发现了:

src/features/shared/reducers.js#L6-L28

export const itemsReducer = (type, idFunc = defaultIdFunc) => (state = {}, action) => {
  if (action.type == `RECEIVED_${type.toUpperCase()}_ITEMS`) {
    const newObjects = {}

    const data = type.toUpperCase() !== 'TRANSACTION' ? action.param.data : action.param.data.map(data => ({
      ...data,
      id: data.txId,
      timestamp: data.blockTime,
      blockId: data.blockHash,
      position: data.blockIndex
    }));

    (data || []).forEach(item => {
      if (!item.id) { item.id = idFunc(item) }
      newObjects[idFunc(item)] = item
    })
    return newObjects
  } else // ...
  return state
}

可以看到,当拿到数据后,如果是“转帐”则进行一些特殊的操作,否则就直接用。后面的操作,也主要是给每个元素增加了一个id,然后放到store里。

那么第3步中的APPEND_${type.toUpperCase()}_PAGE呢?我们找到一些通用的处理代码:

src/features/shared/reducers.js#L34-L54

export const queryCursorReducer = (type) => (state = {}, action) => {
  if (action.type == `APPEND_${type.toUpperCase()}_PAGE`) {
    return action.param
  }
  return state
}

export const queryTimeReducer = (type) => (state = '', action) => {
  if (action.type == `APPEND_${type.toUpperCase()}_PAGE`) {
    return moment().format('h:mm:ss a')
  }
  return state
}

export const autocompleteIsLoadedReducer = (type) => (state = false, action) => {
  if (action.type == `DID_LOAD_${type.toUpperCase()}_AUTOCOMPLETE`) {
    return true
  }

  return state
}

这里没有什么复杂的操作,主要是把前面送过来的参数当作store新的state传出去,或者在queryTimeReducer是传出当前时间,可以把它们理解为一些占位符(默认值)。如果针对某一个具体类型,还可以定义具体的操作。比如我们这里是balance,所以它还会被前面最开始讲解的这个函数处理:

src/features/balances/reducers.js#L3-L17

const itemsReducer = (state = {}, action) => {
  if (action.type == 'APPEND_BALANCE_PAGE') {
    const newState = {}
    action.param.data.forEach((item, index) => {
      const id = `balance-${index}`
      newState[id] = {
        id: `balance-${index}`,
        ...item
      }
    })

    return newState
  }
  return state
}

这个前面已经讲了,这里列出来仅供回忆。

那么到这里,我们基本上就已经把比原前端中,如何通过分页列表形式展示数据的流程弄清楚了。至于拿到数据后,最终如何在页面上以table的形式展示出来,可以参看https://github.com/freewind/bytom-dashboard-v1.0.0/blob/master/src/features/balances/components/ListItem.jsx,我觉得这里已经不需要再讲解了。

那么我们准备进入后端。

后端是如何通过/list-balances接口查询出帐户余额的

跟之前一样,我们可以很快的找到定义web api接口的地方:

api/api.go#L164-L244

func (a *API) buildHandler() {
    // ...
    if a.wallet != nil {
        // ...
        m.Handle("/list-balances", jsonHandler(a.listBalances))
        // ...

    // ...
}

可以看到,/list-balances对应的handler是a.listBalances(外面的jsonHandler是用于处理http方面的东西,以及在Go对象与JSON之间做转换的)

api/query.go#L60-L67

// POST /list-balances
func (a *API) listBalances(ctx context.Context) Response {
    balances, err := a.wallet.GetAccountBalances("")
    if err != nil {
        return NewErrorResponse(err)
    }
    return NewSuccessResponse(balances)
}

这个方法看起来很简单,因为它不需要前端传入任何参数,然后再调用wallet.GetAccountBalances并传入空字符串(表示全部帐户)拿到结果,并且返回给前端即可:

wallet/indexer.go#L544-L547

// GetAccountBalances return all account balances
func (w *Wallet) GetAccountBalances(id string) ([]AccountBalance, error) {
    return w.indexBalances(w.GetAccountUTXOs(""))
}

这里分成了两步,首先是调用w.GetAccountUTXOs得到帐户对应的UTXO,然后再根据它计算出来余额balances。

UTXOUnspent Transaction Output,是比特币采用的一个概念(在比原链中对它进行了扩展,支持多种资产)。其中Transaction可看作是一种数据结构,记录了一个交易的过程,包括若干个资金输入和输出。在比特币中没有我们通常熟悉的银行帐户那样有专门的地方记录余额,而是通过计算属于自己的所有未花费掉的输出来算出余额。关于UTXO网上有很多文章讲解,可以自行搜索。

我们继续看w.GetAccountUTXOs

wallet/indexer.go#L525-L542

// GetAccountUTXOs return all account unspent outputs
func (w *Wallet) GetAccountUTXOs(id string) []account.UTXO {
    var accountUTXOs []account.UTXO

    accountUTXOIter := w.DB.IteratorPrefix([]byte(account.UTXOPreFix + id))
    defer accountUTXOIter.Release()
    for accountUTXOIter.Next() {
        accountUTXO := account.UTXO{}
        if err := json.Unmarshal(accountUTXOIter.Value(), &accountUTXO); err != nil {
            hashKey := accountUTXOIter.Key()[len(account.UTXOPreFix):]
            log.WithField("UTXO hash", string(hashKey)).Warn("get account UTXO")
        } else {
            accountUTXOs = append(accountUTXOs, accountUTXO)
        }
    }

    return accountUTXOs
}

这个方法看起来不是很复杂,它主要是从数据库中搜索UTXO,然后返回给调用者继续处理。这里的w.DB是指名为wallet的leveldb,我们这段时间一直在用它。初始化的过程今天就不看了,之前做过多次,大家有需要的话应该能自己找到。

然后就是以UTXOPreFix(常量ACU:,表示StandardUTXOKey prefix)作为前缀对数据库进行遍历,把取得的JSON格式的数据转换为account.UTXO对象,最后把它们放到数组里返回给调用者。

我们再看前面GetAccountBalances方法中的w.indexBalances

wallet/indexer.go#L559-L609

func (w *Wallet) indexBalances(accountUTXOs []account.UTXO) ([]AccountBalance, error) {
    // 1. 
    accBalance := make(map[string]map[string]uint64)
    balances := make([]AccountBalance, 0)

    // 2.
    for _, accountUTXO := range accountUTXOs {
        assetID := accountUTXO.AssetID.String()
        if _, ok := accBalance[accountUTXO.AccountID]; ok {
            if _, ok := accBalance[accountUTXO.AccountID][assetID]; ok {
                accBalance[accountUTXO.AccountID][assetID] += accountUTXO.Amount
            } else {
                accBalance[accountUTXO.AccountID][assetID] = accountUTXO.Amount
            }
        } else {
            accBalance[accountUTXO.AccountID] = map[string]uint64{assetID: accountUTXO.Amount}
        }
    }

    // 3. 
    var sortedAccount []string
    for k := range accBalance {
        sortedAccount = append(sortedAccount, k)
    }
    sort.Strings(sortedAccount)

    for _, id := range sortedAccount {
        // 4.
        var sortedAsset []string
        for k := range accBalance[id] {
            sortedAsset = append(sortedAsset, k)
        }
        sort.Strings(sortedAsset)

        // 5.
        for _, assetID := range sortedAsset {
            alias := w.AccountMgr.GetAliasByID(id)
            targetAsset, err := w.AssetReg.GetAsset(assetID)
            if err != nil {
                return nil, err
            }

            assetAlias := *targetAsset.Alias
            balances = append(balances, AccountBalance{
                Alias: alias,
                AccountID: id,
                AssetID: assetID,
                AssetAlias: assetAlias,
                Amount: accBalance[id][assetID],
                AssetDefinition: targetAsset.DefinitionMap,
            })
        }
    }

    return balances, nil
}

这个方法看起来很长,但是实际上做的事情没那么多,只不过是因为Go低效的语法让它看起来非常庞大。我把它分成了5块:

  1. 第1块分别定义了后面要用到的一些数据结构,其中accBalance是一个两级的map(AccountID -> AssetID -> AssetAmount),通过对参数accountUTXOs进行遍历,把相同account和相同asset的数量累加在一起。balances是用来保存结果的,是一个AccountBalance的切片
  2. 第2块就是累加assetAmount,放到accBalance
  3. 对accountId进行排序,
  4. 对assetId也进行排序,这两处的排序是想让最后的返回结果稳定(有利于查看及分页)
  5. 经过双层遍历,拿到了每一个account的每一种asset的assetAmount,然后再通过w.AccountMgr.GetAliasByID拿到缺少的alias信息,最后生成一个切片返回。其中GetAliasByID就是从wallet数据库中查询,比较简单,就不贴代码了。

看完这一段代码之后,我的心情是比较郁闷的,因为这里的代码看着多,但实际上都是一些比较低层的逻辑(构建、排序、遍历),在其它的语言中(尤其是支持函数式的),可能只需要十来行代码就能搞定,但是这么要写这么多。而且,我还发现,GO语言通过它独特的语法、错误处理和类型系统,让一些看起来应该很简单的事情(比如抽出来一些可复用的处理数据结构的函数)都变得很麻烦,我试着重构,居然发现无从下手。

今天的问题就算是解决了,下次再见。

相关文章
|
5月前
|
Java
用JAVA架建List集合为树形结构的代码方法
这段代码定义了一个表示树形结构的 `Node` 类和一个用于构建树形结构的 `TreeController`。`Node` 类包含基本属性如 `id`、`pid`、`name` 和 `type`,以及子节点列表 `children`。`TreeController` 包含初始化节点列表并将其转换为树形结构的方法。通过过滤和分组操作实现树形结构的构建。详情可见:[代码示例链接1](http://www.zidongmutanji.com/zsjx/43551.html),[代码效果参考链接2](https://www.257342.com/sitemap/post.html)。
51 5
|
4月前
|
Java API 开发者
代码小妙招:用Java轻松获取List交集数据
在Java中获取两个 `List`的交集可以通过 `retainAll`方法和Java 8引入的流操作来实现。使用 `retainAll`方法更为直接,但会修改原始 `List`的内容。而使用流则提供了不修改原始 `List`、更为灵活的处理方式。开发者可以根据具体的需求和场景,选择最适合的方法来实现。了解和掌握这些方法,能够帮助开发者在实际开发中更高效地处理集合相关的问题。
116 1
|
5月前
|
存储 Java API
【Java高手必备】揭秘!如何优雅地对List进行排序?掌握这几种技巧,让你的代码瞬间高大上!
【8月更文挑战第23天】本文深入探讨了Java中对List集合进行排序的各种方法,包括使用Collections.sort()、自定义Comparator以及Java 8的Stream API。通过示例代码展示了不同情况下如何选择合适的方法:从简单的整数排序到自定义类对象的排序,再到利用Comparator指定特殊排序规则,最后介绍了Stream API在排序操作中的简洁应用。理解这些技术的区别与应用场景有助于提高编程效率。
152 4
|
6月前
|
运维 关系型数据库 Java
PolarDB产品使用问题之使用List或Range分区表时,Java代码是否需要进行改动
PolarDB产品使用合集涵盖了从创建与管理、数据管理、性能优化与诊断、安全与合规到生态与集成、运维与支持等全方位的功能和服务,旨在帮助企业轻松构建高可用、高性能且易于管理的数据库环境,满足不同业务场景的需求。用户可以通过阿里云控制台、API、SDK等方式便捷地使用这些功能,实现数据库的高效运维与持续优化。
|
7月前
|
Java
JAVA构建List集合为树形结构的方法和代码
JAVA构建List集合为树形结构的方法和代码
35 0
|
8月前
|
存储 C++ 容器
【C++】STL容器——list类的使用指南(含代码演示)(13)
【C++】STL容器——list类的使用指南(含代码演示)(13)
|
8月前
|
Java
Java【代码分享 11】yaml配置List和Map参数对象的配置信息及类文件实例分享(效仿GatewayDynamic+DynamicDataSource的注入方法)
Java【代码分享 11】yaml配置List和Map参数对象的配置信息及类文件实例分享(效仿GatewayDynamic+DynamicDataSource的注入方法)
296 0
|
8月前
|
安全 Java
Java【代码分享 09】多线程处理List数据核心代码说明(下标越界、数据丢失及效率问题)
Java【代码分享 09】多线程处理List数据核心代码说明(下标越界、数据丢失及效率问题)
131 0
|
8月前
|
Oracle Java 关系型数据库
Java【代码分享 06】Lamda表达式将List对象中的Map对象的key全部转化为大写或者小写(去除外层循环:可用于Map对象中的key全部转化为大写或者小写)
Java【代码分享 06】Lamda表达式将List对象中的Map对象的key全部转化为大写或者小写(去除外层循环:可用于Map对象中的key全部转化为大写或者小写)
370 0
|
前端开发
前端项目实战柒拾柒react-admin+material ui-踩坑-List的基本用法最少代码<Datagrid>
前端项目实战柒拾柒react-admin+material ui-踩坑-List的基本用法最少代码<Datagrid>
46 0