我试图让AsyncStorage在react-native中工作,只是为了节省一些用户输入的数据。虽然我通过在线资源和SO社区找到了一些帮助,但这些东西让我发疯了。
即使是React文档似乎也存在缺陷。下面的代码是taken straight from the AsyncStorage page of the react docs并粘贴到一个新的,干净的应用程序中(我已经添加了最后一行代码来注册应用程序,没有它就无法工作)。
下面的大块代码将作为参考发布。我相信重要的是这一行:
{this.state.messages.map(function(m) {
return <Text key={m}>{m}</Text>;
})}
虽然我有一个密钥集,但我得到了唯一的密钥错误。
'use strict';
var React = require('react');
var ReactNative = require('react-native');
var {
AppRegistry,
AsyncStorage,
PickerIOS,
Text,
View
} = ReactNative;
var PickerItemIOS = PickerIOS.Item;
var STORAGE_KEY = '@AsyncStorageExample:key';
var COLORS = ['red', 'orange', 'yellow', 'green', 'blue'];
var BasicStorageExample = React.createClass({
componentDidMount() {
this._loadInitialState().done();
},
async _loadInitialState() {
try {
var value = await AsyncStorage.getItem(STORAGE_KEY);
if (value !== null){
this.setState({selectedValue: value});
this._appendMessage('Recovered selection from disk: ' + value);
} else {
this._appendMessage('Initialized with no selection on disk.');
}
} catch (error) {
this._appendMessage('AsyncStorage error: ' + error.message);
}
},
getInitialState() {
return {
selectedValue: COLORS[0],
messages: [],
};
},
render() {
var color = this.state.selectedValue;
return (
<View>
<PickerIOS
selectedValue={color}
onValueChange={this._onValueChange}>
{COLORS.map((value) => (
<PickerItemIOS
key={value}
value={value}
label={value}
/>
))}
</PickerIOS>
<Text>
{'Selected: '}
<Text style={{color}}>
{this.state.selectedValue}
</Text>
</Text>
<Text>{' '}</Text>
<Text onPress={this._removeStorage}>
Press here to remove from storage.
</Text>
<Text>{' '}</Text>
<Text>Messages:</Text>
{this.state.messages.map(function(m) {
return <Text key={m}>{m}</Text>;
})}
</View>
);
},
async _onValueChange(selectedValue) {
this.setState({selectedValue});
try {
await AsyncStorage.setItem(STORAGE_KEY, selectedValue);
this._appendMessage('Saved selection to disk: ' + selectedValue);
} catch (error) {
this._appendMessage('AsyncStorage error: ' + error.message);
}
},
async _removeStorage() {
try {
await AsyncStorage.removeItem(STORAGE_KEY);
this._appendMessage('Selection removed from disk.');
} catch (error) {
this._appendMessage('AsyncStorage error: ' + error.message);
}
},
_appendMessage(message) {
this.setState({messages: this.state.messages.concat(message)});
},
});
exports.title = 'AsyncStorage';
exports.description = 'Asynchronous local disk storage.';
exports.examples = [
{
title: 'Basics - getItem, setItem, removeItem',
render(): ReactElement { return <BasicStorageExample />; }
},
];
AppRegistry.registerComponent('BasicStorageExample', () => BasicStorageExample);
上述代码一直有效,直到您选择之前选择的元素。然后,您会发现有关唯一子键的控制台错误。我还没有弄清楚如何避免这种情况。
控制台错误:
index.ios.bundle:24458警告:flattenChildren(...):遇到两个孩子用相同的键,&#39; 6:$ Saved selection to disk:orange&#39;。子键必须是唯一的;当两个孩子共用一把钥匙时,只会使用第一个孩子。
答案 0 :(得分:1)
在地图功能中利用索引而不是值。
{this.state.messages.map(function(m, i) {
return <Text key={'message-' + i}>{m}</Text>;
})}
答案 1 :(得分:1)
如果您使用消息文本作为密钥,并且您有两条具有相同文本的消息,则该密钥不是唯一的。您应该使用数组索引作为键:
{this.state.messages.map((message, i) => (
<Text key={i}>
{message}
</Text>
)}
作为旁注,您应该对代码进行以下更改:
React.createClass
import
代替require()
'use strict'
在ES6中完全是多余的,因为默认情况下它是严格的