飞道的博客

<flutter>跨平台开发新手入坑指南 dart dio pubspec.yaml json_annotation 打包 小坑指南

406人阅读  评论(0)

1.资源文件和依赖三方包(pubspec.yaml)

pubspec.yaml文件可以说是和安卓的gradle文件差不多,它用来描述版本号、sdk、依赖等的。

在资源导入方面同安卓不一样的是,flutter需要在pubspec.yaml中声名,不然在dart文件中是引用不到的,但是pubspec.yaml中不论是资源声名还是依赖都需要固定格式,例如:

 就类似这样,有几个空格、对不对齐都决定着你是不是能依赖和声明成功。

我们可以对资源声明目录,但是批量指定并不递归,只有在该目录下的文件才可以被包括,如果下面还有子目录的话,需要单独声明子目录下的文件。

2.适配

图片资源适配也是和安卓差不多,分为2.0x,3.0x,4.0x,系统会自动根据屏幕选择最接近的分辨率,这里需要注意的是,比如说目录是在assets/images/下面创建的分辨率包,那images根目录下就是1.0x的分辨率,而且这里也必须要有对应的资源,因为它是资源的标识符。

 字体大小和宽高等的适配也有是对应框架的,我们可以添加依赖:flutter_screenutil: ^0.4.2,然后在入口里对它添加对应的设计尺寸进行初始化:

ScreenUtil.instance = ScreenUtil(width: 375, height: 893)..init(context);

 然后在dart里使用ScreenUtil.setXXX就可以了。

3.网络请求和json解析

经常用到的应该就是dio了,它的请求贴下代码吧:

  static Future<Map<String, dynamic>> requestPost<T>(
      String url, Map<String, String> map) async {
    String path = HttpConfig.baseUrl + url;
    //添加头部信息
    BaseOptions options = BaseOptions();
    options.headers["Accept-Encoding"] = "gzip";
    options.headers["Accept"] = "application/json";
    options.contentType = "application/json; charset=utf-8";
    // options.headers["User-Agent"] = getUserAgent();
    options.connectTimeout = HttpConfig.timeOut;
    options.headers["Authorization"] = "Bearer";

    String currentTimeMillis =
        DateTime.now().microsecondsSinceEpoch.toString().substring(0, 10);

    //公参配置
    map["sign"] = "";
    map["device_code"] = device_code;
    map["timestamp"] = currentTimeMillis;
    map["uid"] = uid;
    Dio dio = Dio(options);
    //添加拦截
    dio.interceptors.add(
      InterceptorsWrapper(
          onRequest: (RequestOptions options) {

          },
          onResponse: (Response response) {
            //拦截处理后台的错误格式数据
            var errData= response.data.toString();
             xxx ...
            return response;
          },
          onError: (DioError dioError) {

          }),
    );
    Response response = await dio.post(path);
    var str = jsonDecode(response.data); //处理一下返回Future数据
    return str;
  }

 

请求很简单,我想讲的主要点是解析数据,我用到了json_annotation,一开始我是用了网络大家都推荐的FlutterJsonBeanFactory 插件,插件中心下载安装后发现创建File->New>JsonToDartAction里面的JsonToDartAction点击没有任何反应,最后怀疑就是因为as版本或者其他什么环境的问题,那捷径没走成,那就找到了另一个工具插件JsonToDartClass,它的用法流程其实和所谓的jsonFormart差不多,首先新创建一个dart类,然后在类里右键选择Generate,点击JsonToDartClass,会出现下图:

 将正确无误的json格式的json数据填入里面点ok(注意是正确格式的json,否则会出错),我们就会生成这么一个类:

接着我们会发现报错了,那是因为我们还有一步操作,也是容易出问题的地方,这里就和咱们原生安卓不一样了,我们需要在项目根目录下运行命令:

flutter packages pub run build_runner build,如果你运气够好,那会在你这个dart文件的同级文件夹内生成一个.g.dart的文件,此时所有的报错就都会消失了,这时候我们就需要:

var datas = JSDataBean.fromJson(result); 来获取到data就可以各种赋值了。

但是上边说运气好的情况,运气不好,你就会遇到以下问题:

1.因为json格式有误导致的生成出错,改正格式即可

2.有些报错(比如:Failed to build build_runner:build_runner:)发现和build_runner的版本号有关系,需要调整版本号

3.报错Flutter Conflicting outputs were detected and the build is unable to prompt for permiss,这个时候是生成的dart_tool目录下缺少了之前生成的文件导致的报错,可以依次执行下边命令即可:
1、flutter packages pub run build_runner clean
2、flutter packages pub run build_runner build --delete-conflicting-outputs
3、flutter packages pub run build_runner build

当然也有一劳永逸的办法,运行flutter packages pub run build_runner watch可以以后自动生成。

4.打包体验流畅的app

说到flutter都知道debug版本可以热更,HotReload和HotRestart,只需要ctril+s就可以直接预览界面,非常的方便有效率,由于debug和relase编译机制不一样(debug是jit即时编译,relase是aot提前编译),所带来的问题就是debug版本性能阉割太厉害,调试的时候可以感受得到卡的像老年机,那我们打正式包:

1.生成key文件:执行命令:

keytool -genkey -v -keystore E:\_as_project_me\flutter_app\flutter_app_key.jks -keyalg RSA -keysize 2048 -validity 10000 -alias key

此时会提示让填入各种信息,依次填写就行,完成后就可以生成key文件了。

2.配置安卓gradle

signingConfigs {
    release {
        keyAlias 'key'
        keyPassword 'xxx'
        storeFile file('E:\\_as_project_me\\flutter_app\\flutter_app_key.jks')//此种写法默认key文件在android-app文件夹下
        storePassword 'xxx'
    }
    debug {
        keyAlias 'key'
        keyPassword 'xxx'
        storeFile file('E:\\_as_project_me\\flutter_app\\flutter_app_key.jks')//此种写法默认key文件在android-app文件夹下
        storePassword 'xxx'
    }
}

 

3.打包执行命令:flutter build apk,如果报安全问题,则执行:flutter build apk --no-sound-null-safety 即可。

 关于苹果打包,我也尝试装了个黑苹果,安装了xcode和as导入项目,不过还没获得苹果开发者账号,所以没把流程走到最后,有个文章可以大家参考下:

Flutter IOS 真机调试,无需开发者账号! - Kxmrg的开发日记

5.dart的语法等一些细节

1.expand:填充满剩余空间,比如:children中的子控件想要拉伸,例如spaceBetween这种的,需要对children外部的Container嵌套一个Expanded才行,先把外部容器拉伸。

2.SingleChildScrollView嵌套容器时候 如果是listview则设置   physics:NeverScrollableScrollPhysics(),  shrinkWrap: true,    否则就给父组件设置固定高度:constraints: BoxConstraints(maxHeight: 100),

3.NestedScrollView 同安卓 折叠组件SliverAppBar

4.ListView中Axis.horizontal时,需要父view指定高度

5.在GridView中的元素无法设置其宽高,主要通过childAspectRatio来设置其比例,通过比例来显示其大小

6.显示省略号overflow: TextOverflow.ellipsis

7.如果你适应了java的new对象写法,写一个new Container时,他会有黄色警告,去掉new就好了

8.const:当一个变量在编译时就确定而且以后不会再发生改变的时候,我们将使用const来修饰,节省build时间

9.color 和 decoration 是互斥的,如果同时设置它们会报错,因为当指定 color 属性时,在 Container 内会自动创建一个 decoration。

10.bottomNavigationBar中,onTab切换会把页面重置,这时候需要页面继承AutomaticKeepAliveClientMixin ,重写 @override   bool get wantKeepAlive => true; 可解决。

11.late JSDataBean jsData;为稍后初始化,可去掉强制让初始化的限制 ,在使用前被初始化

12  .?(类型后面跟操作符 ? 表示当前变量可为null。)
        !(在使用可为null的变量时对可为null变量的另一种处理方式)

13.Stack相当于RelativeLayout,children里面层叠,要居中给Stack设置alignment

.........

有时间会再更新,不足之处多多指正。


转载:https://blog.csdn.net/csdn_lg_one/article/details/128457621
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场