Titanium Android 应用在更新 ScrollableView 时崩溃
Titanium Android app crashes upon updating ScrollableView
首先:一切都完全按照 iOS 中的预期进行。
我认为问题是由 titanium 生成的某些 Java 对象没有获得更新可滚动视图大小的参数。
错误日志:
[WARN] : ResourceType: No known package when getting name for resource number 0xffffffff
[WARN] : dalvikvm: threadid=1: thread exiting with uncaught exception (group=0xa4d57b20)
[ERROR] : TiApplication: (main) [674,5584] Sending event: exception on thread: main msg:java.lang.IllegalStateException: The application's PagerAdapter changed the adapter's contents without calling PagerAdapter#notifyDataSetChanged! Expected adapter item count: 1, found: 0 Pager id: ffffffff Pager class: class ti.modules.titanium.ui.widget.TiUIScrollableView Problematic adapter: class ti.modules.titanium.ui.widget.TiUIScrollableView$ViewPagerAdapter; Titanium 3.5.0,2015/01/12 15:33,0014f83
[ERROR] : TiApplication: java.lang.IllegalStateException: The application's PagerAdapter changed the adapter's contents without calling PagerAdapter#notifyDataSetChanged! Expected adapter item count: 1, found: 0 Pager id: ffffffff Pager class: class ti.modules.titanium.ui.widget.TiUIScrollableView Problematic adapter: class ti.modules.titanium.ui.widget.TiUIScrollableView$ViewPagerAdapter
[ERROR] : TiApplication: at android.support.v4.view.ViewPager.populate(ViewPager.java:962)
[ERROR] : TiApplication: at android.support.v4.view.ViewPager.populate(ViewPager.java:914)
[ERROR] : TiApplication: at android.support.v4.view.ViewPager.onMeasure(ViewPager.java:1436)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.constrainChild(TiCompositeLayout.java:398)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.onMeasure(TiCompositeLayout.java:278)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.constrainChild(TiCompositeLayout.java:398)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.onMeasure(TiCompositeLayout.java:278)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.constrainChild(TiCompositeLayout.java:398)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.onMeasure(TiCompositeLayout.java:278)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
[ERROR] : TiApplication: at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
[ERROR] : TiApplication: at com.android.internal.widget.ActionBarOverlayLayout.onMeasure(ActionBarOverlayLayout.java:327)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
[ERROR] : TiApplication: at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
[ERROR] : TiApplication: at com.android.internal.policy.impl.PhoneWindow$DecorView.onMeasure(PhoneWindow.java:2291)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:1912)
[ERROR] : TiApplication: at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1109)
[ERROR] : TiApplication: at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1291)
[ERROR] : TiApplication: at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:996)
[ERROR] : TiApplication: at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5600)
[ERROR] : TiApplication: at android.view.Choreographer$CallbackRecord.run(Choreographer.java:761)
[ERROR] : TiApplication: at android.view.Choreographer.doCallbacks(Choreographer.java:574)
[ERROR] : TiApplication: at android.view.Choreographer.doFrame(Choreographer.java:544)
[ERROR] : TiApplication: at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:747)
[ERROR] : TiApplication: at android.os.Handler.handleCallback(Handler.java:733)
[ERROR] : TiApplication: at android.os.Handler.dispatchMessage(Handler.java:95)
[ERROR] : TiApplication: at android.os.Looper.loop(Looper.java:136)
[ERROR] : TiApplication: at android.app.ActivityThread.main(ActivityThread.java:5001)
[ERROR] : TiApplication: at java.lang.reflect.Method.invokeNative(Native Method)
[ERROR] : TiApplication: at java.lang.reflect.Method.invoke(Method.java:515)
[ERROR] : TiApplication: at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:785)
[ERROR] : TiApplication: at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:601)
[ERROR] : TiApplication: at dalvik.system.NativeStart.main(Native Method)
[INFO] : Process: Sending signal. PID: 1280 SIG: 9
直接设置可滚动视图的 .views 属性 或使用 .setViews() 时会发生崩溃。
相关代码:
//on scrolling of the Years view fill the makes and model views
function populateMake(e){
$.makeScrollableView.scrollToView(0);
removeAllChildren($.makeScrollableView);
//Generating URL
var selectedYear = $.yearScrollableView.currentPage; //selected index
var carYearViews = $.yearScrollableView.getViews(); //array of views
var children = carYearViews[selectedYear].getChildren(); //children of the selected year (aka the label w/ the year in it)
var makeYear = children[0].getText();
var url = "www.example.com";
var makeViews = [];
var client = Ti.Network.createHTTPClient({
onload : function(e){
var yearMakes = [];
yearMakes = parseMakes(this.responseText); //returns an array of strings
for(var i = 0; i < yearMakes.length; i++){
var yearLabel = Ti.UI.createLabel({
title: "year",
text: yearMakes[i]
});
var view = Ti.UI.createView({
backgroundColor: 'blue'
});
view.add(yearLabel);
makeViews.push(view);
}
$.makeScrollableView.setViews(makeViews); //<--- crashes here when scrolling $.yearScrollableView
populateModel();
},
onerror : function (e){
Ti.API.error("HTTP Error populating makes");
}
});
client.open("GET", url);
client.send();
}
并且:
//populate model when year/make is changed
function populateModel(e){
$.modelScrollableView.scrollToView(0);
removeAllChildren($.modelScrollableView);
var selectedYear = $.yearScrollableView.currentPage; //selected index
var carYearViews = $.yearScrollableView.getViews(); //array of views
var childrenYear = carYearViews[selectedYear].getChildren(); //children of the selected year (aka the label w/ the year in it)
var carYear = childrenYear[0].getText();
var url = "www.totallyafakewebsite.net";
Ti.API.info(url);
var modelViews = [];
var client = Ti.Network.createHTTPClient({
onload : function(e){
var carModels = [];
carModels = parseModels(this.responseText); //array of strings
for(var i = 0; i < carModels.length; i++){
var modelLabel = Ti.UI.createLabel({
title: "model",
text: carModels[i]
});
var view = Ti.UI.createView({
backgroundColor: 'blue'
});
view.add(modelLabel);
modelViews.push(view);
}
$.modelScrollableView.setViews(modelViews); //<--- this is the crash when scrolling $.makeScrollableView
//$.modelScrollableView.views = modelViews; //doesn't work either
},
onerror : function (e){
Ti.API.error("HTTP Error populating models: " + e.error);
}
});
client.open("GET", url);
client.send();
}
滚动相应的可滚动视图时触发这些事件。
每个可滚动视图都在 xml 视图中创建。
我发现了我自己的问题:
发生这种情况的原因是我自己的 removeAllChildren 函数不起作用。因为 Titanium 的功能不起作用,所以我不得不创建自己的功能。
函数内部看起来像这样:
$.scrollableView.values = []; //an empty array
相反,我使用了以下代码:
//Remove old data from $.makeScrollableView
$.makeScrollableView.setCurrentPage(0);
makeViews.length = 0;
将当前索引设置为 0,然后将数组长度设置为 0 解决了我的问题。
@starlord7,你代码中的以下说法不正确...
removeAllChildren($.makeScrollableView);
您必须将其更改为...
$.makeScrollableView.removeAllChildren();
此外,我不完全确定将 makeViews 数组的长度设置为 0 是否是个好主意。如果您在需要发布的视图中有一些引用,那么它将被清除以销毁他们一个一个。这可以帮助您发布视频、音频、大图像、事件侦听器、对其他视图的任何引用等资源。
请记住,使用钛 Android 很容易造成巨大的内存泄漏。
首先:一切都完全按照 iOS 中的预期进行。
我认为问题是由 titanium 生成的某些 Java 对象没有获得更新可滚动视图大小的参数。
错误日志:
[WARN] : ResourceType: No known package when getting name for resource number 0xffffffff
[WARN] : dalvikvm: threadid=1: thread exiting with uncaught exception (group=0xa4d57b20)
[ERROR] : TiApplication: (main) [674,5584] Sending event: exception on thread: main msg:java.lang.IllegalStateException: The application's PagerAdapter changed the adapter's contents without calling PagerAdapter#notifyDataSetChanged! Expected adapter item count: 1, found: 0 Pager id: ffffffff Pager class: class ti.modules.titanium.ui.widget.TiUIScrollableView Problematic adapter: class ti.modules.titanium.ui.widget.TiUIScrollableView$ViewPagerAdapter; Titanium 3.5.0,2015/01/12 15:33,0014f83
[ERROR] : TiApplication: java.lang.IllegalStateException: The application's PagerAdapter changed the adapter's contents without calling PagerAdapter#notifyDataSetChanged! Expected adapter item count: 1, found: 0 Pager id: ffffffff Pager class: class ti.modules.titanium.ui.widget.TiUIScrollableView Problematic adapter: class ti.modules.titanium.ui.widget.TiUIScrollableView$ViewPagerAdapter
[ERROR] : TiApplication: at android.support.v4.view.ViewPager.populate(ViewPager.java:962)
[ERROR] : TiApplication: at android.support.v4.view.ViewPager.populate(ViewPager.java:914)
[ERROR] : TiApplication: at android.support.v4.view.ViewPager.onMeasure(ViewPager.java:1436)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.constrainChild(TiCompositeLayout.java:398)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.onMeasure(TiCompositeLayout.java:278)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.constrainChild(TiCompositeLayout.java:398)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.onMeasure(TiCompositeLayout.java:278)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.constrainChild(TiCompositeLayout.java:398)
[ERROR] : TiApplication: at org.appcelerator.titanium.view.TiCompositeLayout.onMeasure(TiCompositeLayout.java:278)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
[ERROR] : TiApplication: at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
[ERROR] : TiApplication: at com.android.internal.widget.ActionBarOverlayLayout.onMeasure(ActionBarOverlayLayout.java:327)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5125)
[ERROR] : TiApplication: at android.widget.FrameLayout.onMeasure(FrameLayout.java:310)
[ERROR] : TiApplication: at com.android.internal.policy.impl.PhoneWindow$DecorView.onMeasure(PhoneWindow.java:2291)
[ERROR] : TiApplication: at android.view.View.measure(View.java:16497)
[ERROR] : TiApplication: at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:1912)
[ERROR] : TiApplication: at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1109)
[ERROR] : TiApplication: at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1291)
[ERROR] : TiApplication: at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:996)
[ERROR] : TiApplication: at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5600)
[ERROR] : TiApplication: at android.view.Choreographer$CallbackRecord.run(Choreographer.java:761)
[ERROR] : TiApplication: at android.view.Choreographer.doCallbacks(Choreographer.java:574)
[ERROR] : TiApplication: at android.view.Choreographer.doFrame(Choreographer.java:544)
[ERROR] : TiApplication: at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:747)
[ERROR] : TiApplication: at android.os.Handler.handleCallback(Handler.java:733)
[ERROR] : TiApplication: at android.os.Handler.dispatchMessage(Handler.java:95)
[ERROR] : TiApplication: at android.os.Looper.loop(Looper.java:136)
[ERROR] : TiApplication: at android.app.ActivityThread.main(ActivityThread.java:5001)
[ERROR] : TiApplication: at java.lang.reflect.Method.invokeNative(Native Method)
[ERROR] : TiApplication: at java.lang.reflect.Method.invoke(Method.java:515)
[ERROR] : TiApplication: at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:785)
[ERROR] : TiApplication: at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:601)
[ERROR] : TiApplication: at dalvik.system.NativeStart.main(Native Method)
[INFO] : Process: Sending signal. PID: 1280 SIG: 9
直接设置可滚动视图的 .views 属性 或使用 .setViews() 时会发生崩溃。
相关代码:
//on scrolling of the Years view fill the makes and model views
function populateMake(e){
$.makeScrollableView.scrollToView(0);
removeAllChildren($.makeScrollableView);
//Generating URL
var selectedYear = $.yearScrollableView.currentPage; //selected index
var carYearViews = $.yearScrollableView.getViews(); //array of views
var children = carYearViews[selectedYear].getChildren(); //children of the selected year (aka the label w/ the year in it)
var makeYear = children[0].getText();
var url = "www.example.com";
var makeViews = [];
var client = Ti.Network.createHTTPClient({
onload : function(e){
var yearMakes = [];
yearMakes = parseMakes(this.responseText); //returns an array of strings
for(var i = 0; i < yearMakes.length; i++){
var yearLabel = Ti.UI.createLabel({
title: "year",
text: yearMakes[i]
});
var view = Ti.UI.createView({
backgroundColor: 'blue'
});
view.add(yearLabel);
makeViews.push(view);
}
$.makeScrollableView.setViews(makeViews); //<--- crashes here when scrolling $.yearScrollableView
populateModel();
},
onerror : function (e){
Ti.API.error("HTTP Error populating makes");
}
});
client.open("GET", url);
client.send();
}
并且:
//populate model when year/make is changed
function populateModel(e){
$.modelScrollableView.scrollToView(0);
removeAllChildren($.modelScrollableView);
var selectedYear = $.yearScrollableView.currentPage; //selected index
var carYearViews = $.yearScrollableView.getViews(); //array of views
var childrenYear = carYearViews[selectedYear].getChildren(); //children of the selected year (aka the label w/ the year in it)
var carYear = childrenYear[0].getText();
var url = "www.totallyafakewebsite.net";
Ti.API.info(url);
var modelViews = [];
var client = Ti.Network.createHTTPClient({
onload : function(e){
var carModels = [];
carModels = parseModels(this.responseText); //array of strings
for(var i = 0; i < carModels.length; i++){
var modelLabel = Ti.UI.createLabel({
title: "model",
text: carModels[i]
});
var view = Ti.UI.createView({
backgroundColor: 'blue'
});
view.add(modelLabel);
modelViews.push(view);
}
$.modelScrollableView.setViews(modelViews); //<--- this is the crash when scrolling $.makeScrollableView
//$.modelScrollableView.views = modelViews; //doesn't work either
},
onerror : function (e){
Ti.API.error("HTTP Error populating models: " + e.error);
}
});
client.open("GET", url);
client.send();
}
滚动相应的可滚动视图时触发这些事件。
每个可滚动视图都在 xml 视图中创建。
我发现了我自己的问题:
发生这种情况的原因是我自己的 removeAllChildren 函数不起作用。因为 Titanium 的功能不起作用,所以我不得不创建自己的功能。
函数内部看起来像这样:
$.scrollableView.values = []; //an empty array
相反,我使用了以下代码:
//Remove old data from $.makeScrollableView
$.makeScrollableView.setCurrentPage(0);
makeViews.length = 0;
将当前索引设置为 0,然后将数组长度设置为 0 解决了我的问题。
@starlord7,你代码中的以下说法不正确...
removeAllChildren($.makeScrollableView);
您必须将其更改为...
$.makeScrollableView.removeAllChildren();
此外,我不完全确定将 makeViews 数组的长度设置为 0 是否是个好主意。如果您在需要发布的视图中有一些引用,那么它将被清除以销毁他们一个一个。这可以帮助您发布视频、音频、大图像、事件侦听器、对其他视图的任何引用等资源。
请记住,使用钛 Android 很容易造成巨大的内存泄漏。