Android实用代码模块集锦

简介: 1、 精确获取屏幕尺寸(例如:3.5、4.0、5.0寸屏幕)123456public static double getScreenPhysicalSize(Activity ctx) {      DisplayMetrics dm = new DisplayMetrics();      ctx.

1、 精确获取屏幕尺寸(例如:3.5、4.0、5.0寸屏幕)

1
2
3
4
5
6
public static double getScreenPhysicalSize(Activity ctx) {
      DisplayMetrics dm = new DisplayMetrics();
      ctx.getWindowManager().getDefaultDisplay().getMetrics(dm);
      double diagonalPixels = Math.sqrt(Math.pow(dm.widthPixels, 2) + Math.pow(dm.heightPixels, 2));
      return diagonalPixels / (160 * dm.density);
}

2、 判断是否是平板(官方用法)

1
2
3
public static boolean isTablet(Context context) {
     return (context.getResources().getConfiguration().screenLayout & Configuration.SCREENLAYOUT_SIZE_MASK) >=Configuration.SCREENLAYOUT_SIZE_LARGE;
}

3、 文字根据状态更改颜色 android:textColor

1
2
3
4
5
6
<selector xmlns:android="http://schemas.android.com/apk/res/android">
      <item android:color="#53c1bd" android:state_selected="true"/>
      <item android:color="#53c1bd" android:state_focused="true"/>
      <item android:color="#53c1bd" android:state_pressed="true"/>
      <item android:color="#777777"/>
</selector>

放在res/color/目录下

4、背景色根据状态更改颜色 android:backgroup

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_selected="true">
         <shape>
             <gradient
                   android:angle="0"
                   android:centerColor="#00a59f"
                   android:endColor="#00a59f"
                   android:startColor="#00a59f" />
         </shape>
    </item>
    <item android:state_focused="true">
         <shape>
              <gradient
                   android:angle="0"
                   android:centerColor="#00a59f"
                   android:endColor="#00a59f"
                   android:startColor="#00a59f" />
         </shape>
     </item>
     <item android:state_pressed="true">
         <shape>
              <gradient
                   android:angle="0"
                   android:centerColor="#00a59f"
                   android:endColor="#00a59f"
                   android:startColor="#00a59f" />
          </shape>
      </item>
      <item>
          <shape>
              <gradient
                  android:angle="0"
                  android:centerColor="#00ff00"
                  android:endColor="00ff00"
                  android:startColor="00ff00" />
          </shape>
      </item>
</selector>

如果直接给背景色color会报错。

5、 启动APK的默认Activity

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public static void startApkActivity(final Context ctx, String packageName) {
     PackageManager pm = ctx.getPackageManager();
     PackageInfo pi;
     try {
          pi = pm.getPackageInfo(packageName, 0);
          Intent intent = new Intent(Intent.ACTION_MAIN, null);
          intent.addCategory(Intent.CATEGORY_LAUNCHER);
          intent.setPackage(pi.packageName);List<ResolveInfo> apps = pm.queryIntentActivities(intent, 0);
          ResolveInfo ri = apps.iterator().next();
          if (ri != null) {
                String className = ri.activityInfo.name;
                intent.setComponent(new ComponentName(packageName, className));
                ctx.startActivity(intent);
          }
     } catch (NameNotFoundException e) {
          Log.e("startActivity", e);
     }
}

6、计算字宽

1
2
3
4
5
public static float GetTextWidth(String text, float Size) {
     TextPaint FontPaint = new TextPaint();
     FontPaint.setTextSize(Size);
     return FontPaint.measureText(text);
}

注意如果设置了textStyle,还需要进一步设置TextPaint。

7、获取应用程序下所有Activity

1
2
3
4
5
6
7
8
9
public static ArrayList<String> getActivities(Context ctx) {
      ArrayList<String> result = new ArrayList<String>();
      Intent intent = new Intent(Intent.ACTION_MAIN, null);
      intent.setPackage(ctx.getPackageName());
      for (ResolveInfo info : ctx.getPackageManager().queryIntentActivities(intent, 0)) {
           result.add(info.activityInfo.name);
      }
      return result;
}

8、检测字符串中是否包含汉字

1
2
3
4
5
6
7
8
public static boolean checkChinese(String sequence) {
     final String format = "[\\u4E00-\\u9FA5\\uF900-\\uFA2D]";
     boolean result = false;
     Pattern pattern = Pattern.compile(format);
     Matcher matcher = pattern.matcher(sequence);
     result = matcher.find();
     return result;
}

9、检测字符串中只能包含:中文、数字、下划线(_)、横线(-)

1
2
3
4
5
6
public static boolean checkNickname(String sequence) {
     final String format = "[^\\u4E00-\\u9FA5\\uF900-\\uFA2D\\w-_]";
     Pattern pattern = Pattern.compile(format);
     Matcher matcher = pattern.matcher(sequence);
     return !matcher.find();
}

10、检查有没有应用程序来接受处理你发出的intent

1
2
3
4
5
6
public static boolean isIntentAvailable(Context context, String action) {
    final PackageManager packageManager = context.getPackageManager();
    final Intent intent = new Intent(action);
    List<ResolveInfo> list = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
    return list.size();
}

11、使用TransitionDrawable实现图片显示的时候有渐变效果

1
2
3
4
5
6
7
8
private void setImageBitmap(ImageView imageView, Bitmap bitmap) {
      // Use TransitionDrawable to fade in.
      final TransitionDrawable td = new TransitionDrawable(new Drawable[] { new ColorDrawable(android.R.color.transparent), newBitmapDrawable(mContext.getResources(), bitmap) });
      //noinspection deprecation
      imageView.setBackgroundDrawable(imageView.getDrawable());
      imageView.setImageDrawable(td);
      td.startTransition(200);
}

比使用AlphaAnimation效果要好,可避免出现闪烁问题。

12、 扫描指定的文件

1
sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, uri));

用途:从本软件新增、修改、删除图片、文件某一个文件(音频、视频)需要更新系统媒体库时使用,不必扫描整个SD卡。

13、Dip转px

1
2
3
public static int dipToPX(final Context ctx, float dip) {
      return (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dip, ctx.getResources().getDisplayMetrics());
}

用途:难免在Activity代码中设置位置、大小等,本方法就很有用了!

14、获取已经安装APK(列表)的路径

1
2
3
4
PackageManager pm = getPackageManager();
for (ApplicationInfo app : pm.getInstalledApplications(0)) {
      Log.d("PackageList", "package: " + app.packageName + ", sourceDir: " + app.sourceDir);
}

输出:

1
2
package: com.tmobile.thememanager, sourceDir: /system/app/ThemeManager.apk
package: com.touchtype.swiftkey, sourceDir: /data/app/com.touchtype.swiftkey-1.apk

15、 多进程Preferences数据共享

1
2
3
4
5
6
7
8
9
10
public static void putStringProcess(Context ctx, String key, String value) {
     SharedPreferences sharedPreferences = ctx.getSharedPreferences("preference_mu", Context.MODE_MULTI_PROCESS);
     Editor editor = sharedPreferences.edit();
     editor.putString(key, value);
     editor.commit();
}
public static String getStringProcess(Context ctx, String key, String defValue) {
     SharedPreferences sharedPreferences = ctx.getSharedPreferences("preference_mu", Context.MODE_MULTI_PROCESS);
     return sharedPreferences.getString(key, defValue);
}

16、泛型ArrayList转数组

1
2
3
4
5
6
7
@SuppressWarnings("unchecked")
public static <T> T[] toArray(Class<?> cls, ArrayList<T> items) {
     if (items == null || items.size() == 0) {
          return (T[]) Array.newInstance(cls, 0);
     }
     return items.toArray((T[]) Array.newInstance(cls, items.size()));
}

17、 保存恢复ListView当前位置

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
private void saveCurrentPosition() {
      if (mListView != null) {
            int position = mListView.getFirstVisiblePosition();
            View v = mListView.getChildAt(0);
            int top = (v == null) ? 0 : v.getTop();
            //保存position和top
       }
}
private void restorePosition() {
       if (mFolder != null && mListView != null) {
             int position = 0;//取出保存的数据
             int top = 0;//取出保存的数据
             mListView.setSelectionFromTop(position, top);
       }
}

18、调用 便携式热点和数据共享 设置

1
2
3
4
5
6
7
public static Intent getHotspotSetting() {
     Intent intent = new Intent();
     intent.setAction(Intent.ACTION_MAIN);
     ComponentName com = new ComponentName("com.android.settings", "com.android.settings.TetherSettings");
     intent.setComponent(com);
     return intent;
}

19、 格式化输出IP地址

1
2
3
4
5
public static String getIp(Context ctx) {
     return Formatter.formatIpAddress((WifiManager)ctx.getSystemService(Context.WIFI_SERVICE).getConnectionInfo().getIpAddress());
}

20、 ip地址转成8位十六进制串

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
/** ip转16进制 */
public static String ipToHex(String ips) {
     StringBuffer result = new StringBuffer();
     if (ips != null) {
          StringTokenizer st = new StringTokenizer(ips, ".");
          while (st.hasMoreTokens()) {
                String token = Integer.toHexString(Integer.parseInt(st.nextToken()));
                if (token.length() == 1)
                      token = "0" + token;
                result.append(token);
          }
     }
     return result.toString();
}
/** 16进制转ip */
public static String texToIp(String ips) {
     try {
          StringBuffer result = new StringBuffer();
          if (ips != null && ips.length() == 8) {
               for (int i = 0; i < 8; i += 2) {
                    if (i != 0)
                          result.append('.');
                    result.append(Integer.parseInt(ips.substring(i, i + 2), 16));
               }
          }
          return result.toString();
     } catch (NumberFormatException ex) {
          Logger.e(ex);
     }
     return "";
}
ip:192.168.68.128 16 =>hex :c0a84480

21、 文件夹排序(先文件夹排序,后文件排序)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public static void sortFiles(File[] files) {
     Arrays.sort(files, new Comparator<File>() {
          @Override
          public int compare(File lhs, File rhs) {
               //返回负数表示o1 小于o2,返回0 表示o1和o2相等,返回正数表示o1大于o2。
               boolean l1 = lhs.isDirectory();
               boolean l2 = rhs.isDirectory();
               if (l1 && !l2)
                    return -1;
               else if (!l1 && l2)
                    return 1;
               else {
                    return lhs.getName().compareTo(rhs.getName());
               }
          }
     });
}

22、代码设置TextView的样式

使用过自定义Dialog可能马上会想到用如下代码:

1
new TextView(this,null,R.style.text_style);

但你运行这代码你会发现毫无作用!正确用法:

1
new TextView(new ContextThemeWrapper(this, R.style.text_style))

23、获取网络类型名称

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
public static String getNetworkTypeName(Context context) {
     if (context != null) {
          ConnectivityManager connectMgr = (ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE);
          if (connectMgr != null) {
                  NetworkInfo info = connectMgr.getActiveNetworkInfo();
                  if (info != null) {
                       switch (info.getType()) {
                            case ConnectivityManager.TYPE_WIFI:
                                  return "WIFI";
                            case ConnectivityManager.TYPE_MOBILE:
                                  return getNetworkTypeName(info.getSubtype());
                       }
                  }
          }
     }
     return getNetworkTypeName(TelephonyManager.NETWORK_TYPE_UNKNOWN);
}
public static String getNetworkTypeName(int type) {
     switch (type) {
          case TelephonyManager.NETWORK_TYPE_GPRS:
               return "GPRS";
          case TelephonyManager.NETWORK_TYPE_EDGE:
               return "EDGE";
          case TelephonyManager.NETWORK_TYPE_UMTS:
               return "UMTS";
          case TelephonyManager.NETWORK_TYPE_HSDPA:
               return "HSDPA";
          case TelephonyManager.NETWORK_TYPE_HSUPA:
               return "HSUPA";
          case TelephonyManager.NETWORK_TYPE_HSPA:
               return "HSPA";
          case TelephonyManager.NETWORK_TYPE_CDMA:
               return "CDMA";
          case TelephonyManager.NETWORK_TYPE_EVDO_0:
               return "CDMA - EvDo rev. 0";
          case TelephonyManager.NETWORK_TYPE_EVDO_A:
               return "CDMA - EvDo rev. A";
          case TelephonyManager.NETWORK_TYPE_EVDO_B:
               return "CDMA - EvDo rev. B";
          case TelephonyManager.NETWORK_TYPE_1xRTT:
               return "CDMA - 1xRTT";
          case TelephonyManager.NETWORK_TYPE_LTE:
               return "LTE";
          case TelephonyManager.NETWORK_TYPE_EHRPD:
               return "CDMA - eHRPD";
          case TelephonyManager.NETWORK_TYPE_IDEN:
               return "iDEN";
          case TelephonyManager.NETWORK_TYPE_HSPAP:
               return "HSPA+";
          default:
               return "UNKNOWN";
     }
}

24、Android解压Zip包

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
/**
* 解压一个压缩文档 到指定位置
*
* @param zipFileString 压缩包的名字
* @param outPathString 指定的路径
* @throws Exception
*/
public static void UnZipFolder(String zipFileString, String outPathString) throws Exception {
      java.util.zip.ZipInputStream inZip = new java.util.zip.ZipInputStream(new java.io.FileInputStream(zipFileString));
      java.util.zip.ZipEntry zipEntry;
      String szName = "";while ((zipEntry = inZip.getNextEntry()) != null) {
           szName = zipEntry.getName();
           if (zipEntry.isDirectory()) {
                // get the folder name of the widget
                szName = szName.substring(0, szName.length() - 1);
                java.io.File folder = new java.io.File(outPathString + java.io.File.separator + szName);
                folder.mkdirs();
          } else {
                java.io.File file = new java.io.File(outPathString + java.io.File.separator + szName);
                file.createNewFile();
                // get the output stream of the file
                java.io.FileOutputStream out = new java.io.FileOutputStream(file);
                int len;
                byte[] buffer = new byte[1024];
                // read (len) bytes into buffer
                while ((len = inZip.read(buffer)) != -1) {
                      // write (len) byte from buffer at the position 0
                      out.write(buffer, 0, len);
                      out.flush();
               }
               out.close();
         }
    }//end of while
    inZip.close();
}//end of func

25、 从assets中读取文本和图片资源

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
/** 从assets 文件夹中读取文本数据 */
public static String getTextFromAssets(final Context context, String fileName) {
     String result = "";
     try {
          InputStream in = context.getResources().getAssets().open(fileName);
          // 获取文件的字节数
          int lenght = in.available();
          // 创建byte数组
          byte[] buffer = new byte[lenght];
          // 将文件中的数据读到byte数组中
          in.read(buffer);
          result = EncodingUtils.getString(buffer, "UTF-8");
          in.close();
     } catch (Exception e) {
          e.printStackTrace();
     }
     return result;
}
/** 从assets 文件夹中读取图片 */
public static Drawable loadImageFromAsserts(final Context ctx, String fileName) {
      try {
            InputStream is = ctx.getResources().getAssets().open(fileName);
            return Drawable.createFromStream(is, null);
      } catch (IOException e) {
            if (e != null) {
                 e.printStackTrace();
            }
      } catch (OutOfMemoryError e) {
            if (e != null) {
                 e.printStackTrace();
            }
      } catch (Exception e) {
            if (e != null) {
                 e.printStackTrace();
            }
      }
      return null;
}

26、展开、收起状态栏

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
public static final void collapseStatusBar(Context ctx) {
     Object sbservice = ctx.getSystemService("statusbar");
     try {
          Class<?> statusBarManager = Class.forName("android.app.StatusBarManager");
          Method collapse;
          if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
                collapse = statusBarManager.getMethod("collapsePanels");
          } else {
                collapse = statusBarManager.getMethod("collapse");
          }
          collapse.invoke(sbservice);
     } catch (Exception e) {
           e.printStackTrace();
     }
}
public static final void expandStatusBar(Context ctx) {
      Object sbservice = ctx.getSystemService("statusbar");
      try {
            Class<?> statusBarManager = Class.forName("android.app.StatusBarManager");
            Method expand;
            if (Build.VERSION.SDK_INT >= 17) {
                  expand = statusBarManager.getMethod("expandNotificationsPanel");
            } else {
                  expand = statusBarManager.getMethod("expand");
            }
            expand.invoke(sbservice);
      } catch (Exception e) {
            e.printStackTrace();
      }
}

用途:可用于点击Notifacation之后收起状态栏

27、 获取状态栏高度

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public static int getStatusBarHeight(Context context){
     Class<?> c = null;
     Object obj = null;
     Field field = null;
     int x = 0, statusBarHeight = 0;
     try {
          c = Class.forName("com.android.internal.R$dimen");
          obj = c.newInstance();
          field = c.getField("status_bar_height");
          x = Integer.parseInt(field.get(obj).toString());
          statusBarHeight = context.getResources().getDimensionPixelSize(x);
     } catch (Exception e1) {
          e1.printStackTrace();
     }
     return statusBarHeight;
}

28、ListView使用ViewHolder极简写法

1
2
3
4
5
6
7
8
9
10
11
12
13
public static <T extends View> T getAdapterView(View convertView, int id) {
     SparseArray<View> viewHolder = (SparseArray<View>) convertView.getTag();
     if (viewHolder == null) {
          viewHolder = new SparseArray<View>();
          convertView.setTag(viewHolder);
     }
     View childView = viewHolder.get(id);
     if (childView == null) {
          childView = convertView.findViewById(id);
          viewHolder.put(id, childView);
     }
     return (T) childView;
}

用法:

1
2
3
4
5
6
7
8
9
10
@Override
public View getView(int position, View convertView, ViewGroup parent) {
     if (convertView == null) {
          convertView = LayoutInflater.from(getActivity()).inflate(R.layout.fragment_feed_item, parent, false);
     }
     ImageView thumnailView = getAdapterView(convertView, R.id.video_thumbnail);
     ImageView avatarView =  getAdapterView(convertView, R.id.user_avatar);
     ImageView appIconView = getAdapterView(convertView, R.id.app_icon);
}

用起来非常简练,将ViewHolder隐于无形。

 

29、 设置Activity透明

 

1
2
3
4
5
6
7
<style  name ="TransparentActivity"  parent ="AppBaseTheme">
      <item  name ="android: windowBackground">@android: color/transparent</item>
      <item  name ="android: colorBackgroundCacheHint">@null</item>
      <item  name ="android: windowIsTranslucent">true</item>
      <item  name ="android: windowNoTitle">true</item>
      <item  name ="android: windowContentOverlay">@null</item>
</style>

说明:AppBaseTheme一般是你application指定的Android:theme是啥这里就是啥,否则Activity内部的空间风格可能不一致。

用途:用于模拟Dialog效果,比如再Service中没法用Dialog,就可以用Activity来模拟

30、 代码切换全屏

 

1
2
3
4
5
//切换到全屏
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);
getActivity().getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);//切换到非全屏
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);

 

注意:切换到全屏时,底部的虚拟按键仍然是显示的。次方法可多次调用用于切换

用途:播放器界面经常会用到

转自:农民伯伯

若转载请注明出处!若有疑问,请回复交流!
目录
相关文章
|
16天前
|
安全 Java 网络安全
Android远程连接和登录FTPS服务代码(commons.net库)
Android远程连接和登录FTPS服务代码(commons.net库)
16 1
|
29天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异:从代码到用户体验
【10月更文挑战第5天】在移动应用开发的广阔天地中,安卓和iOS两大平台各占半壁江山。它们在技术架构、开发环境及用户体验上有着根本的不同。本文通过比较这两种平台的开发过程,揭示背后的设计理念和技术选择如何影响最终产品。我们将深入探讨各自平台的代码示例,理解开发者面临的挑战,以及这些差异如何塑造用户的日常体验。
|
2月前
|
存储 Java Android开发
🔥Android开发大神揭秘:从菜鸟到高手,你的代码为何总是慢人一步?💻
在Android开发中,每位开发者都渴望应用响应迅速、体验流畅。然而,代码执行缓慢却是常见问题。本文将跟随一位大神的脚步,剖析三大典型案例:主线程阻塞导致卡顿、内存泄漏引发性能下降及不合理布局引起的渲染问题,并提供优化方案。通过学习这些技巧,你将能够显著提升应用性能,从新手蜕变为高手。
26 2
|
3月前
|
JSON JavaScript 前端开发
Android调用Vue中的JavaScript代码
Android调用Vue中的JavaScript代码
34 3
|
3月前
|
编解码 网络协议 前端开发
如何实现Android平台GB28181设备接入模块按需打开摄像头并回传数据
后台采集摄像头,如果想再进一步扩展,可以把android平台gb28181的camera2 demo,都移植过来,实现功能更强大的国标设备侧,这里主要是展示,收到国标平台侧的回传请求后,才打开摄像头,才开始编码打包,最大限度的减少资源的占用
|
3月前
|
编解码 网络协议 Android开发
Android平台GB28181设备接入模块实现后台service按需回传摄像头数据到国标平台侧
我们在做Android平台GB28181设备对接模块的时候,遇到这样的技术需求,开发者希望能以后台服务的形式运行程序,国标平台侧没有视频回传请求的时候,仅保持信令链接,有发起视频回传请求或语音广播时,打开摄像头,并实时回传音视频数据或接收处理国标平台侧发过来的语音广播数据。
|
3月前
|
监控 Java 开发工具
如何快速对接Android平台GB28181接入模块(SmartGBD)
大牛直播SDK推出的Android平台GB28181接入SDK(SmartGBD),可实现不具备国标音视频能力的 Android终端,通过平台注册接入到现有的GB/T28181—2016服务,可用于如执法记录仪、智能安全帽、智能监控、智慧零售、智慧教育、远程办公、明厨亮灶、智慧交通、智慧工地、雪亮工程、平安乡村、生产运输、车载终端等场景,可能是业内为数不多功能齐全性能优异的商业级水准GB28181接入SDK。
|
3月前
|
安全 Java 网络安全
Android远程连接和登录FTPS服务代码(commons.net库)
很多文章都介绍了FTPClient如何连接ftp服务器,但却很少有人说如何连接一台开了SSL认证的ftp服务器,现在代码来了。
97 2
|
4月前
|
存储 Java Android开发
🔥Android开发大神揭秘:从菜鸟到高手,你的代码为何总是慢人一步?💻
【7月更文挑战第28天】在Android开发中,每位开发者都追求极致的用户体验。然而,“代码执行慢”的问题时常困扰着开发者。通过案例分析,我们可探索从新手到高手的成长路径。
39 3
|
3月前
|
Java Android开发
Android项目架构设计问题之要提升代码的可读性和管理性如何解决
Android项目架构设计问题之要提升代码的可读性和管理性如何解决
38 0