Skip to content

wangkang007/android_tips

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

3 Commits
 
 
 
 
 
 

Repository files navigation

本文主要是记录一下在日常开发中的一些总结和一些规范性的东西

###目录

##android命名规范 ###1. id控件命名规范

控件名称 缩写
LinearLayout ll_
RelativeLayout rl_
TextView tv_
Button btn_
ImageButton ibtn_
ImageView iv_
CheckBox cb_
RadioButton rbtn_
EditText et_
ProgressBar proBar_
SeekBar skBar_
AutoCompleteTextView autoTxt_
WebView wbv_
RantingBar ratBar_
Spinner spn_
ScollView slV_
TextSwitch txtSwt_
ListView lv_
ExpandableListView epdLv_
DatePicker dtPk_
TimePicker tmPk_

###2. 资源文件命名规范

资源 命名规范
colors.xml 项目简称_color_颜色英文_色值,比如:cs_color_yellow_ff484a
string.xml 项目简称_string_具体英文,比如:cs_string_cancle
styles.xml 针对TextView来说可以用 tv_颜色英文_色值_size字体大小,比如:tv_grey808080_size12

###3. 资源布局规范

Activity activity_模块名称.xml,例如:activity_main.xml、activity_user.xml
Fragment fragment_模块名称.xml,例如:fragment_main.xml、fragment_user.xml
列表项 item_模块名称.xml
通用布局 common_布局的名称.xml,比如User的通用item,叫做common_user_item.xml
###4. drawable命名规范
</thead>
<tbody>
 <tr>
  <td>Button的背景图命名</td>
  <td>
	图片:
	<table>
	 <tr>
	  <td>命名规范</td>
	  <td>状态</td>
     </tr>
	 <tr>
	  <td>图片名_normal或者直接图片名</td>
	  <td>(default state)</td>
     </tr>
	  <tr>
	  <td>图片名_pressed</td>
	  <td>state_pressed</td>
     </tr>
	  <tr>
	  <td>图片名_focused</td>
	  <td>state_focused</td>
     </tr>
	  <tr>
	  <td>图片名_disabled</td>
	  <td>state_enabled (false)</td>
     </tr>
	  <tr>
	  <td>图片名_selected</td>
	  <td>state_selected</td>
     </tr>
	 <tr>
	  <td>图片名_hovered</td>
	  <td>state_hovered</td>
     </tr>
	 <tr>
	  <td>图片名_checkable</td>
	  <td>state_checkable</td>
     </tr>
	 <tr>
	  <td>图片名_activated</td>
	  <td>state_activated</td>
     </tr>
	  <tr>
	  <td>图片名_windowfocused</td>
	  <td>state_windowfocused</td>
     </tr>
	</table>
  </td>
 </tr>
 <tr>
  <td>Button通用的shape命名</td>
  <td>
	shape前缀规范:根据里面的属性来进行命名前缀,比如: btn_redff6d6d_radius2.xml, btn_redff6d6d_radius2_1strokef8f8f8_press.xml(如果项目里面已经有了调色板,整个系统button都使用统一的边框和红色或者其他颜色,可以进行简化,例如btn_red_press)
	<table>
	 <tr>
	  <td>命名规范</td>
	  <td>状态</td>
     </tr>
	 <tr>
	  <td>shape前缀_normal或者是直接shape前缀即可例如btn_red_radius2.xml</td>
	  <td>(default state)</td>
     </tr>
	  <tr>
	  <td>shape前缀_pressed</td>
	  <td>state_pressed</td>
     </tr>
	  <tr>
	  <td>shape前缀_focused</td>
	  <td>state_focused</td>
     </tr>
	  <tr>
	  <td>shape前缀_disabled</td>
	  <td>state_enabled (false)</td>
     </tr>
	  <tr>
	  <td>shape前缀_selected</td>
	  <td>state_selected</td>
     </tr>
	 <tr>
	  <td>shape前缀_hovered</td>
	  <td>state_hovered</td>
     </tr>
	 <tr>
	  <td>shape前缀_checkable</td>
	  <td>state_checkable</td>
     </tr>
	 <tr>
	  <td>shape前缀_activated</td>
	  <td>state_activated</td>
     </tr>
	  <tr>
	  <td>shape前缀_windowfocused</td>
	  <td>state_windowfocused</td>
     </tr>
	</table>
  </td>
 </tr>
 <tr>
  
  <td>Fragment</td>
  <td>fragment_模块名称.xml,例如:fragment_main.xml、fragment_user.xml</td>
</tr>
 <tr>
  <td>列表项</td>
  <td>item_模块名称.xml</td>
</tr>
<tr>
  <td>通用布局</td>
  <td>common_布局的名称.xml,比如User的通用item,叫做common_user_item.xml</td>
</tr>
 <tr>
  <td>selector</td>
  <td>以selector结尾</td>
</tr>
###5. 编码命名规范     1. 类成员变量命名规范:android中的Activity,Fragment,Adapter和一些自定义View等以及包括一些业务处理的类(不包括model的属性)类成员变量命名规则如下所示:
 public ImageView mIv_question;
 public TextView mTv_car_num;

    2. android组件类命名规范:

命名规范
Activity 例如:主Activity叫做MainActivity
Adapter 例如:如果主Activity叫做MainActivity,那么Adapter叫做MainAdapter,取Activity的前缀
Fragment 例如:根据业务名称来命名(业务名Fragment)

###6. 代码注释规范     1.class头注释

/**
 * @ClassName: 类名(例如:MainActivity)
 * @Description: (类的描述)
 * @Author (姓名,可以把写这个类的人的拼音或者英文名写入到这,通常android studio会根据系统用户名自动生成)
 * @Date: 2015/7/10 0010 (日期格式)
 */

如下模板可以通过打开android studio->settings->File and Code Templates中,打开Class设置项进行设置,把头部放入以下模板

/**
 * @ClassName: ${NAME}
 * @Description: 
 * @Author  ${USER}(如果系统有用户名可以用${USER}变量取,如果没有可以写死 用户名)
 * @Date ${DATE}
 */

    2.方法注释

/**
 * @Description:
 * @param datas  (参数描述)
 * @return  (返回值描述)
 */

##android注意事项 ###1. Handler的使用     Handler机制有一个特点是不会随着Activity、Service的生命周期结束而结束。也就是说,如果你Post了一个Delay的Runnable,然后在Runnable执行之前退出了Activity,Runnable到时间之后还是要执行的。如果Runnable里面包含更新View的操作,程序崩溃了。

    解决方案:

    1.采用EventBus的开发方式,线程中的数据都可以通过消息的形式传到主Activity中。

    2.采用android-weak-handler,具体使用方式可以点击连接进行查看。

    3.@Override protected void onDestroy() { super.onDestroy(); handler.removeCallbacksAndMessages(null); }

###2. ListView的使用     1. 使用ListView时如果用到removeHeaderView,一定要确定ListView已经使用了setAdapter方法,不然会报NullPointException。

    2. 如果ListView没有HeaderView或者FooterView的时候,与ListView相关联的Adapter就是传进来的参数Adapter。如果有,则原来的Adapter将被包装成HeaderViewListAdapter。

    3. 使用ListView的addHeaderView时候,切记要放在setAdapter之前。由2可如果先setAdapter则与ListView想关联的Adapter是设置 的adapter,没有进行包装,在调用addHeaderView方法的时候会把adapter强制转换成HeaderViewListAdapter。这个时候就会报ClassCastException

    4. 当我们使用给ListView添加了Header的时候,这个时候要注意,我们自定义adapter里面的getitem方法里面返回的position是不包括header的,也就是我们数据集合中的位置,而listview的onitemclick方法中返回的position是HeaderViewListAdapter中的位置,是带了header的。所以注意操作数据的时候记得-1

    5. 使用ListView的时候,布局尽量使用fill_parent或者写死,如果使用wrap_content,它初始化的时候需要测量,会重复调用adapter的getView方法。在getView()方法中不要有实例化对象的操作。

###3. TabLayout的使用     在使用support-design包的时候偶然发现的问题(这玩意肯定是个bug,当时使用的版本是22.2.1):

	mViewPager.setAdapter(modelPagerAdapter);
    tabLayout.setupWithViewPager(mViewPager);
    for (int i = 0; i < titles.length; i++) {
        TabLayout.Tab tab = tabLayout.newTab();
        tab.setText(titles[i]);
        tabLayout.addTab(tab);
    }

    一眼看去上面的代码是没有问题的,但其实不然,这样顺序会让tabLayout重复生成Title,如图所示:

    img

    正确顺序

	mViewPager.setAdapter(modelPagerAdapter);
    for (int i = 0; i < titles.length; i++) {
        TabLayout.Tab tab = tabLayout.newTab();
        tab.setText(titles[i]);
        tabLayout.addTab(tab);
    }
	tabLayout.setupWithViewPager(mViewPager);

##android小技巧 ###1. SharedPreferences小技巧     SharedPreferences.Editor.commit这个方法是同步的,一直到把数据同步到Flash上面之后才会返回,由IO操作的不可控,尽量使用apply方法代替。apply只在API Level>=9才会支持,需要做兼容。 ###2. ADB 命令技巧1     如果你用的是mac或者是linux再或者是windows上面得git shell,那么你就可以使用如下文件adb.sh,把下面的内容放入到adb.sh中,然后通过执行./adb.sh+空格+包名的形式去过滤定位到当前应用:

#!/bin/bash
packageName=$1
pid=`adb shell ps | grep $packageName | awk '{print $2 }'`
real_pid=`echo $pid | awk '{print $1}'`
echo $real_pid+"#######"
adb logcat | grep --color=auto $real_pid

###2. ADB 命令技巧2     当你发现你的adb命令不能使用或者是打不开的时候,可以试着采用如下命令去查看下是否有进程占用了你的端口:

    linux or mac:

    netstat -ano | grep "5037"

    window:

    netstat -ano | findstr "5037"

###3.string.xml技巧     在string.xml中定义html代码,可以采用如下的方式:

    

    <string name='effect'>影响力:<![CDATA[<font color='#ff484a'>%1$s</font>]]></string>

    在string.xml中使用空格可以通过代码 &#160;:&#160; 去实现。

###4.android studio使用技巧     1. 检查无用的资源,在项目中点击右键,在出现的右键菜单中有“Analyze” --> “run inspaction by Name ...”。在弹出的搜索窗口中输入想执行的检查类型,如“Unused Resources” ###5.防止EditText自动获取焦点并且弹出输入法     方案1 android:windowSoftInputMode="adjustUnspecified|stateHidden" 通过这段代码设置关闭

   <activity  android:name="指定Activity"
		      android:label="@string/app_name"
		      android:windowSoftInputMode="adjustUnspecified|stateHidden"
		      android:configChanges="orientation|keyboardHidden">
  		<intent-filter>
  			<action android:name="android.intent.action.MAIN" />
  			<category android:name="android.intent.category.LAUNCHER" />
  		</intent-filter>
  < /activity>

    方案二:

EditText edit=(EditText)findViewById(R.id.edit);
InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);
imm.hideSoftInputFromWindow(edit.getWindowToken(),0);

##android兼容性bug ###1.避免快捷方式创建bug,一般创建快捷方式代码如下所述:

 private void addShortcut(String name) {
    Intent addShortcutIntent = new Intent(ACTION_ADD_SHORTCUT);

    // 不允许重复创建
    addShortcutIntent.putExtra("duplicate", false);
 

    // 名字
    addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_NAME, name);

    // 图标
    addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE,
            Intent.ShortcutIconResource.fromContext(MainActivity.this,
                    R.drawable.ic_launcher));

    // 设置关联程序
    Intent launcherIntent = new Intent(Intent.ACTION_MAIN);
    launcherIntent.setClass(MainActivity.this, MainActivity.class);
    launcherIntent.addCategory(Intent.CATEGORY_LAUNCHER);

    addShortcutIntent
            .putExtra(Intent.EXTRA_SHORTCUT_INTENT, launcherIntent);

    // 发送广播
    sendBroadcast(addShortcutIntent);
}

经过测试这段代码是存在问题的addShortcutIntent.putExtra("duplicate", false); 这个代码在一些机型上面是没有效果的。所以只能选择其他方案:

private boolean hasInstallShortcut(String name) {

    boolean hasInstall = false;

    final String AUTHORITY = "com.android.launcher2.settings";
    Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY
            + "/favorites?notify=true");

    // 这里总是failed to find provider info
    // com.android.launcher2.settings和com.android.launcher.settings都不行
    Cursor cursor = this.getContentResolver().query(CONTENT_URI,
            new String[] { "title", "iconResource" }, "title=?",
            new String[] { name }, null);

    if (cursor != null && cursor.getCount() > 0) {
        hasInstall = true;
    }

    return hasInstall;

}

在创建之前首先判断下数据库里面是否存在了

###2.相机调用

/**
 * 专为Android4.4设计的从Uri获取文件绝对路径,以前的方法已不好使
 */
@SuppressLint("NewApi")
public static String getPath(final Context context, final Uri uri) {

    final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;

    // DocumentProvider
    if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
        // ExternalStorageProvider
        if (isExternalStorageDocument(uri)) {
            final String docId = DocumentsContract.getDocumentId(uri);
            final String[] split = docId.split(":");
            final String type = split[0];

            if ("primary".equalsIgnoreCase(type)) {
                return Environment.getExternalStorageDirectory() + "/" + split[1];
            }

            // TODO handle non-primary volumes
        }
        // DownloadsProvider
        else if (isDownloadsDocument(uri)) {

            final String id = DocumentsContract.getDocumentId(uri);
            final Uri contentUri = ContentUris.withAppendedId(
                    Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));

            return getDataColumn(context, contentUri, null, null);
        }
        // MediaProvider
        else if (isMediaDocument(uri)) {
            final String docId = DocumentsContract.getDocumentId(uri);
            final String[] split = docId.split(":");
            final String type = split[0];

            Uri contentUri = null;
            if ("image".equals(type)) {
                contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
            } else if ("video".equals(type)) {
                contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
            } else if ("audio".equals(type)) {
                contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
            }

            final String selection = "_id=?";
            final String[] selectionArgs = new String[] { split[1] };

            return getDataColumn(context, contentUri, selection, selectionArgs);
        }
    }
    // MediaStore (and general)
    else if ("content".equalsIgnoreCase(uri.getScheme())) {
        return getDataColumn(context, uri, null, null);
    }
    // File
    else if ("file".equalsIgnoreCase(uri.getScheme())) {
        return uri.getPath();
    }

    return null;
}
/**
 * Get the value of the data column for this Uri. This is useful for
 * MediaStore Uris, and other file-based ContentProviders.
 * 
 * @param context
 *            The context.
 * @param uri
 *            The Uri to query.
 * @param selection
 *            (Optional) Filter used in the query.
 * @param selectionArgs
 *            (Optional) Selection arguments used in the query.
 * @return The value of the _data column, which is typically a file path.
 */
public static String getDataColumn(Context context, Uri uri, String selection,
        String[] selectionArgs) {

    Cursor cursor = null;
    final String column = "_data";
    final String[] projection = { column };

    try {
        cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
                null);
        if (cursor != null && cursor.moveToFirst()) {
            final int column_index = cursor.getColumnIndexOrThrow(column);
            return cursor.getString(column_index);
        }
    } finally {
        if (cursor != null)
            cursor.close();
    }
    return null;
}

/**
 * @param uri
 *            The Uri to check.
 * @return Whether the Uri authority is ExternalStorageProvider.
 */
public static boolean isExternalStorageDocument(Uri uri) {
    return "com.android.externalstorage.documents".equals(uri.getAuthority());
}

/**
 * @param uri
 *            The Uri to check.
 * @return Whether the Uri authority is DownloadsProvider.
 */
public static boolean isDownloadsDocument(Uri uri) {
    return "com.android.providers.downloads.documents".equals(uri.getAuthority());
}

/**
 * @param uri
 *            The Uri to check.
 * @return Whether the Uri authority is MediaProvider.
 */
public static boolean isMediaDocument(Uri uri) {
    return "com.android.providers.media.documents".equals(uri.getAuthority());
}

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published