Android与服务“.NET研究”器端数据交互

本文涉及的产品
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
简介:   上一节中我们通过http协议,采用HttpClient向服务器端action请求数据。当然调用服务器端方法获取数据并不止这一种。WebService也可以为我们提供所需数据,那么什么是webService呢?,它是一种基于SAOP协议的远程调用标准,通过webservice可以将不同操作系统平台,不同语言,不同技术整合到一起。

  上一节中我们通过http协议,采用HttpClient向服务器端action请求数据。当然调用服务器端方法获取数据并不止这一种。WebService也可以为我们提供所需数据,那么什么是webService呢?,它是一种基于SAOP协议的远程调用标准,通过webservice可以将不同操作系统平台,不同语言,不同技术整合到一起。

  我们在PC机器java客户端中,需要一些库,比如XFire,Axis2,CXF等等来支持访问WebService,但是这些库并不适合我们资源有限的android手机客户端,做过JAVA ME的人都知道有KSOAP这个第三方的类库,可以帮助我们获取服务器端webService调用,当然KSOAP已经提供了基于android版本的jar包了,那么我们就开始吧:

  首先下载KSOAP包:ksoap2-android-assembly-2.5.2-jar-with-dependencies.jar包

  然后新建android项目:并把下载的KSOAP包放在android项目的lib目录下:右键->build path->configure build path--选择Libraries,如图:

  以下分为七个步骤来调用WebService方法:

  第一:实例化SoapObject 对象,指定webService的命名空间(从相关WSDL文档中可以查看命名空间),以及调用方法名称。如:

 
  
// 命名空间
private static final String serviceNameSpace = " http://WebXml.com.cn/ " ;
// 调用方法(获得支持的城市)
private static final String getSupportCity = " getSupportCity " ;

// 实例化SoapObject对象
SoapObject request = new SoapObject(serviceNameSpace, getSupportCity);
第二步:假设方法有参数的话,设置调用方法参数:
 
 
request.addProperty( " 参数名称 " , " 参数值 " );

  第三步:设置SOAP请求信息(参数部分为SOAP协议版本号,与你要调用的webService中版本号一致):

 
 
// 获得序列化的Envelope
SoapSerializationEnvelope envelope = new SoapSerializationEnvelope(SoapEnvelope.VER11);
envelope.bodyOut
= request;

  第四步:注册Envelope:

 
 
( new MarshalBase64()).register(envelope);

  第五步:构建传输对象,并指明WSDL文档URL:

 
 
// 请求URL
private static final String serviceURL = " http://www.webxml.com.cn/webservices/weatherwebservice.asmx " ;
// Android传输对象
AndroidHttpTransport transport = new AndroidHttpTransport(serviceURL);
transport.debug
= true ;

  第六步:调用WebService(其中参数为1:命名空间+方法名称,2:Envelope对象):

 
 
transport.call(serviceNameSpace + getWeatherbyCityName, envelope);

  第七步:解析返回数据:

 
 
if (envelope.getResponse() != null ){
return parse(envelope.bodyIn.toString());
}
/** ************
* 解析XML
*
@param str
*
@return
*/
private static List < String > parse(String str){
String temp;
List
< String > list = new ArrayList < String > ();
if (str != null && str.length() > 0 ){
int start = str.indexOf( " string " );
int end = str.lastIndexOf( " ; " );
temp
= str.substring(start, end - 3 );
String []test
= temp.split( " ; " );

for ( int i = 0 ;i < test.length;i ++ ){
if (i == 0 ){
temp
= test[i].substring( 7 );
}
else {
temp
= test[i].substring( 8 );
}
int index = temp.indexOf( " , " );
list.add(temp.substring(
0 , index));
}
}
return list;
}

  这样就成功啦。那么现在我们就来测试下吧,这里有个地址提供webService天气预报的服务的,我这里只提供获取城市列表:

 
 
// 命名空间
private static final String serviceNameSpace = " http://WebXml.com.cn/ " ;
// 请求URL
private static final String serviceURL = " http://www.webxml.com.cn/webservices/weatherwebservice.asmx " ;
// 调用方法(获得支持的城市)
private static final String getSupportCity = " getSupportCity " ;
// 调用城市的方法(需要带参数)
private static final String getWeatherbyCityName = " getWeatherbyCityName " ;
// 调用省或者直辖市的方法(获得支持的省份或直辖市)
private static final String getSupportProvince = " getSupportProvince " ;

  然后你可以在浏览器中输入地址(WSDL):serviceURL,你会看到一些可供调用的方法:

  我们选择获取国内外主要城市或者省份的方法吧:getSupportProvice,然后调用,你会发现浏览器返回给我们的是xml文档:

 
 
<? xml version="1.0" encoding="utf-8" ?>
-
< ArrayOfString xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd ="http://www.w3.org/2001/XMLSchema" xmlns ="http://WebXml.com.cn/" >
< string > 直辖市 </ string >
< string > 特别行政区 </ string >
< string > 黑龙江 </ string >
< string > 吉林 </ string >
< string > 辽宁 </ string >
< string > 内蒙古 </ string >
< string > 河北 </ string >
< string > 河南 </ string >
< string > 山东 </ string >
< string > 山西 </ string >
< string > 江苏 </ string >
< string > 安徽 </ string >
< string > 陕西 </ string >
< string > 宁夏 </ string >
< string > 甘肃 </ string >
< string > 青海 </ string >
< string > 湖北 </ string >
< string > 湖南 </ string >
< string > 浙江 </ string >
< string > 江西 </ string >
< string > 福建 </ string >
< string > 贵州 </ string >
< string > 四川 </ string >
< string > 广东 </ string >
< string > 广西 </ string >
< string > 云南 </ string >
< 上海企业网站设计与制作le="color: #800000;">string > 海南 </ string >
< string > 新疆 </ string >
< string > 西藏 </ string >
< string > 台湾 </ string >
< string > 亚洲 </ string >
< string > 欧洲 </ string >
< string > 非洲 </ string >
< string > 北美洲 </ string >
< string > 南美洲 </ string >
< string > 大洋洲 </ string >
</ ArrayOfString >

  我们可以用 listview来显示:

  那么下面我将给出全部代码:

 
 
public class WebServiceHelper {

// WSDL文档中的命名空间
private static final String targetNameSpace = " http://WebXml.com.cn/ " ;
// WSDL文档中的URL
private static final String WSDL = " http://webservice.webxml.com.cn/WebServices/WeatherWebService.asmx?wsdl " ;

// 需要调用的方法名(获得本天气预报Web Services支持的洲、国内外省份和城市信息)
private static final String getSupportProvince = " getSupportProvince " ;
// 需要调用的方法名(获得本天气预报Web Services支持的城市信息,根据省份查询城市集合:带参数)
private static final String getSupportCity = " getSupportCity " ;
// 根据城市或地区名称查询获得未来三天内天气情况、现在的天气实况、天气和生活指数
private static final String getWeatherbyCityName = " getWeatherbyCityName " ;

/* *******
* 获得州,国内外省份和城市信息
* @return
*/
public List < String > getProvince(){
List
< String > provinces = new ArrayList < String > ();
String str
= "" ;
SoapObject soapObject
= new SoapObject(targetNameSpace,getSupportProvince);
// request.addProperty("参数", "参数值");调用的方法参数与参数值(根据具体需要可选可不选)

SoapSerializationEnvelope envelope
= new SoapSerializationEnvelope(SoapEnvelope.VER11);
envelope.dotNet
= true ;
envelope.setOutputSoapObject(soapObject);
// envelope.bodyOut=request;  

AndroidHttpTransport httpTranstation
= new AndroidHttpTransport(WSDL);
// 或者HttpTransportSE httpTranstation=new HttpTransportSE(WSDL);
try {

httpTranstation.call(targetNameSpace
+ getSupportProvince, envelope);
SoapObject result
= (SoapObject)envelope.getResponse();
// 下面对结果进行解析,结构类似json对象
// str=(String) result.getProperty(6).toString();

int count = result.getPropertyCount();
for ( int index = 0 ;index < count;index ++ ){
provinces.add(result.getProperty(index).toString());
}

}
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
catch (XmlPullParserException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return provinces;
}

/* *********
* 根据省份或者直辖市获取天气预报所支持的城市集合
* @param province
* @return
*/
public List < String > getCitys(String province){
List
< String > citys = new ArrayList < String > ();
SoapObject soapObject
= new SoapObject(targetNameSpace,getSupportCity);
soapObject.addProperty(
" byProvinceName " , province);
SoapSerializationEnvelope envelope
= new SoapSerializationEnvelope(SoapEnvelope.VER11);
envelope.dotNet
= true ;
envelope.setOutputSoapObject(soapObject);

AndroidHttpTransport httpTransport
= new AndroidHttpTransport(WSDL);
try {
httpTransport.call(targetNameSpace
+ getSupportCity, envelope);
SoapObject result
= (SoapObject)envelope.getResponse();
int count = result.getPropertyCount();
for ( int index = 0 ;index < count;index ++ ){
citys.add(result.getProperty(index).toString());
}

}
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
catch (XmlPullParserException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return citys;
}

/* **************************
* 根据城市信息获取天气预报信息
* @param city
* @return
**************************
*/
public WeatherBean getWeatherByCity(String city){

WeatherBean bean
= new WeatherBean();

SoapObject soapObject
= new SoapObject(targetNameSpace,getWeatherbyCityName);
soapObject.addProperty(
" theCityName " ,city); // 调用的方法参数与参数值(根据具体需要可选可不选)

SoapSerializationEnvelope envelope
= new SoapSerializationEnvelope(SoapEnvelope.VER11);
envelope.dotNet
= true ;
envelope.setOutputSoapObject(soapObject);
// envelope.bodyOut=request;


AndroidHttpTransport httpTranstation
= new AndroidHttpTransport(WSDL);
// 或者HttpTransportSE httpTranstation=new HttpTransportSE(WSDL);
try {
httpTranstation.call(targetNameSpace
+ getWeatherbyCityName, envelope);
SoapObject result
= (SoapObject)envelope.getResponse();
// 下面对结果进行解析,结构类似json对象
bean = parserWeather(result);

}
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
catch (XmlPullParserException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return bean;
}

/* *
* 解析返回的结果
* @param soapObject
*/
protected WeatherBean parserWeather(SoapObject soapObject){
WeatherBean bean
= new WeatherBean();

List
< Map < String,Object >> list = new ArrayList < Map < String,Object >> ();

Map
< String,Object > map = new HashMap < String,Object > ();

// 城市名
bean.setCityName(soapObject.getProperty( 1 ).toString());
// 城市简介
bean.setCityDescription(soapObject.getProperty(soapObject.getPropertyCount() - 1 ).toString());
// 天气实况+建议
bean.setLiveWeather(soapObject.getProperty( 10 ).toString() + " \n " + soapObject.getProperty( 11 ).toString());

// 其他数据
// 日期,
String date = soapObject.getProperty( 6 ).toString();
// ---------------------------------------------------
String weatherToday = " 今天: " + date.split( " " )[ 0 ];
weatherToday
+= " \n天气: " + date.split( " " )[ 1 ];
weatherToday
+= " \n气温: " + soapObject.getProperty( 5 ).toString();
weatherToday
+= " \n风力: " + soapObject.getProperty( 7 ).toString();
weatherToday
+= " \n " ;

List
< Integer > icons = new ArrayList < Integer > ();

icons.add(parseIcon(soapObject.getProperty(
8 ).toString()));
icons.add(parseIcon(soapObject.getProperty(
9 ).toString()));

map.put(
" weatherDay " , weatherToday);
map.put(
" icons " ,icons);
list.add(map);

// -------------------------------------------------
map = new HashMap < String,Object > ();
date
= soapObject.getProperty( 13 ).toString();
String weatherTomorrow
= " 明天: " + date.split( " " )[ 0 ];
weatherTomorrow
+= " \n天气: " + date.split( " " )[ 1 ];
weatherTomorrow
+= " \n气温: " + soapObject.getProperty( 12 ).toString();
weatherTomorrow
+= " \n风力: " 上海企业网站制作="color: #000000;">+ soapObject.getProperty( 14 ).toString();
weatherTomorrow
+= " \n " ;

icons
= new ArrayList < Integer > ();

icons.add(parseIcon(soapObject.getProperty(
15 ).toString()));
icons.add(parseIcon(soapObject.getProperty(
16 ).toString()));

map.put(
" weatherDay " , weatherTomorrow);
map.put(
" icons " ,icons);
list.add(map);
// --------------------------------------------------------------
map = new HashMap < String,Object > ();

date
= soapObject.getProperty( 18 ).toString();
String weatherAfterTomorrow
= " 后天: " + date.split( " " )[ 0 ];
weatherAfterTomorrow
+= " \n天气: " + date.split( " " )[ 1 ];
weath上海徐汇企业网站设计与制作erAfterTomorrow
+= " \n气温: " + soapObject.getProperty( 17 ).toString();
weatherAfterTomorrow
+= " \n风力: " + soapObject.getProperty( 19 ).toString();
weatherAfterTomorrow
+= " \n " ;

icons
= new ArrayList < Integer >上海闵行企业网站设计与制作span>();
icons.add(parseIcon(soapObject.getProperty(
20).toString()));
icons.add(parseIcon(soapObject.getProperty(
21).toString()));

map.put(
"weatherDay", weatherAfterTomorrow);
map.put(
"icons",icons);
list.add(map);
//--------------------------------------------------------------

bean.setList(list);
return bean;
}

//解析图标字符串
private int parseIcon(String data){
// 0.gif,返回名称0,
int resID=32;
String result
=data.substring(0, data.length()-4).trim();
// String []icon=data.split(".");
// String result=icon[0].trim();
// Log.e("this is the icon", result.trim());

if(!result.equals("nothing")){
resID
=Integer.parseInt(result.trim());
}
return resID;
//return ("a_"+data).split(".")[0];
}
}

  以及帮助类:

 
 
public class WebServiceUtil {

// 命名空间
private static final String serviceNameSpace = " http://WebXml.com.cn/ " ;
// 请求URL
private static final String serviceURL = " http://www.webxml.com.cn/webservices/weatherwebservice.asmx " ;
// 调用方法(获得支持的城市)
private static final String getSupportCity = " getSupportCity " ;
// 调用城市的方法(需要带参数)
private static final String getWeatherbyCityName = " getWeatherbyCityName " ;
// 调用省或者直辖市的方法(获得支持的省份或直辖市)
private static final String getSupportProvince = " getSupportProvince " ;

/* ************
* @return城市列表
************
*/
public static List < String > getCityList(){
// 实例化SoapObject对象
SoapObject request = new SoapObject(serviceNameSpace, getSupportCity);
// 获得序列化的Envelope
SoapSerializationEnvelope envelope = new SoapSerializationEnvelope(SoapEnvelope.VER11);
envelope.bodyOut
= request;
(
new MarshalBase64()).register(envelope);
// Android传输对象
AndroidHttpTransport transport = new AndroidHttpTransport(serviceURL);
transport.debug
= true ;

// 调用
try {
transport.call(serviceNameSpace
+ getWeatherbyCityName, envelope);
if (envelope.getResponse()上海闵行企业网站制作n style="color: #000000;">!= null ){
return parse(envelope.bodyIn.toString());
}

}
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
catch (XmlPullParserException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

return null ;
}

public static List < String > getProviceList(){
// 实例化SoapObject对象
SoapObject request = new SoapObject(serviceNameSpace, getSupportProvince);
// 获得序列化的Envelope
SoapSerializationEnvelope envelope = new SoapSerializationEnvelope(SoapEnvelope.VER11);
envelope.bodyOut
= request;
(
new MarshalBase64()).register(envelope);
// Android传输对象
AndroidHttpTransport transport = new AndroidHttpTransport(serviceURL);
transport.debug
= true ;

// 调用
try {
transport.call(serviceNameSpace
+ getWeatherbyCityName, envelope);
if (envelope.getResponse() != null ){
return null ;
}
}
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
catch (XmlPullParserException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return null ;
}

/* ************
* @param cityName
* @return
************
*/
public static String getWeather(String cityName){

return "" ;
}

/* *************
* 解析XML
* @param str
* @return
*/
private static List < String > parse(String str){
String temp;
List
< String > list = new ArrayList < String > ();
if (str != null && str.length() > 0 ){
int start = str.indexOf( " string " );
int end = str.lastIndexOf( " ; " );
temp
= str.substring(start, end - 3 );
String []test
= temp.split( " ; " );

for ( int i = 0 ;i < test.length;i ++ ){
if (i == 0 ){
temp
= test[i].substring( 7 );
}
else {
temp
= test[i].substring( 8 );
}
int index = temp.indexOf( " , " );
list.add(temp.substring(
0 , index));
}
}
return list;
}

/* ********
* 获取天气
* @param soapObject
*/
private void parseWeather(SoapObject soapObject){
// String date=soapObject.getProperty(6);
}
}

  以上就是我所作的查询天气预报的全部核心代码了,读者可以根据注释以及本文章了解下具体实现,相信很快就搞明白了,运行结果如下:

  到此结束,下一节主要是socket通信了。

目录
相关文章
|
4月前
|
Java Linux Android开发
移动应用开发与操作系统的交互:深入理解Android和iOS
在数字时代,移动应用成为我们日常生活的一部分。本文将深入探讨移动应用开发的核心概念、移动操作系统的工作原理以及它们如何相互作用。我们将通过实际代码示例,展示如何在Android和iOS平台上创建一个简单的“Hello World”应用,并解释其背后的技术原理。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和知识。
|
4月前
|
安全 Java 网络安全
Android远程连接和登录FTPS服务代码(commons.net库)
Android远程连接和登录FTPS服务代码(commons.net库)
61 1
|
4月前
|
SQL XML 关系型数据库
入门指南:利用NHibernate简化.NET应用程序的数据访问
【10月更文挑战第13天】NHibernate是一个面向.NET的开源对象关系映射(ORM)工具,它提供了从数据库表到应用程序中的对象之间的映射。通过使用NHibernate,开发者可以专注于业务逻辑和领域模型的设计,而无需直接编写复杂的SQL语句来处理数据持久化问题。NHibernate支持多种数据库,并且具有高度的灵活性和可扩展性。
72 2
|
5月前
|
JavaScript 前端开发 Android开发
让Vite+Vue3项目在Android端离线打开(不需要起服务)
让Vite+Vue3项目在Android端离线打开(不需要起服务)
185 10
|
5月前
|
调度 Android开发 UED
Android经典实战之Android 14前台服务适配
本文介绍了在Android 14中适配前台服务的关键步骤与最佳实践,包括指定服务类型、请求权限、优化用户体验及使用WorkManager等。通过遵循这些指南,确保应用在新系统上顺畅运行并提升用户体验。
358 6
|
5月前
|
安全 API 开发工具
Android平台RTMP推送|轻量级RTSP服务如何实现麦克风|扬声器声音采集切换
Android平台扬声器播放声音的采集,在无纸化同屏等场景下,意义很大,早期低版本的Android设备,是没法直接采集扬声器audio的(从Android 10开始支持),所以,如果需要采集扬声器audio,需要先做系统版本判断,添加相应的权限。
122 0
|
5月前
|
编解码 开发工具 Android开发
Android平台实现屏幕录制(屏幕投影)|音频播放采集|麦克风采集并推送RTMP或轻量级RTSP服务
Android平台屏幕采集、音频播放声音采集、麦克风采集编码打包推送到RTMP和轻量级RTSP服务的相关技术实现,做成高稳定低延迟的同屏系统,还需要有配套好的RTMP、RTSP直播播放器
|
6月前
|
开发框架 .NET 数据库连接
闲话 Asp.Net Core 数据校验(三)EF Core 集成 FluentValidation 校验数据例子
闲话 Asp.Net Core 数据校验(三)EF Core 集成 FluentValidation 校验数据例子
116 1
|
6月前
|
JSON Android开发 数据格式
Android项目架构设计问题之实现交互响应的结构化处理如何解决
Android项目架构设计问题之实现交互响应的结构化处理如何解决
25 0

热门文章

最新文章

  • 1
    yum install -y net-snmp-devel 安装不成功 zabbix项目安装,Errors during downloading metadata for repository ‘extras-common’:问题解决方案-优雅草卓伊凡
  • 2
    RT-DETR改进策略【Neck】| ArXiv 2023,基于U - Net v2中的的高效特征融合模块:SDI(Semantics and Detail Infusion)
  • 3
    高性能网络SIG双月度动态: 三年磨一剑,virtio-net AF_XDP 完成 Linux 主线零拷贝支持
  • 4
    YOLOv11改进策略【Neck】| ArXiv 2023,基于U - Net v2中的的高效特征融合模块:SDI(Semantics and Detail Infusion)
  • 5
    YOLOv11改进策略【Backbone/主干网络】| 2023 U-Net V2 替换骨干网络,加强细节特征的提取和融合
  • 6
    RT-DETR改进策略【Backbone/主干网络】| 2023 U-Net V2 替换骨干网络,加强细节特征的提取和融合
  • 7
    如何修复 Android 和 Windows 不支持视频编解码器的问题?
  • 8
    Android历史版本与APK文件结构
  • 9
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 10
    【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
  • 1
    【01】噩梦终结flutter配安卓android鸿蒙harmonyOS 以及next调试环境配鸿蒙和ios真机调试环境-flutter项目安卓环境配置-gradle-agp-ndkVersion模拟器运行真机测试环境-本地环境搭建-如何快速搭建android本地运行环境-优雅草卓伊凡-很多人在这步就被难倒了
    14
  • 2
    Cellebrite UFED 4PC 7.71 (Windows) - Android 和 iOS 移动设备取证软件
    28
  • 3
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    34
  • 4
    Android历史版本与APK文件结构
    121
  • 5
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    29
  • 6
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
    23
  • 7
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
    60
  • 8
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
    37
  • 9
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
    73
  • 10
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
    121