EditText默认获取焦点
<EditText
<requestFocus/>//可获取焦点
</EditText>
Gradle解决依赖冲突
gradle ModuleName:dependencies//查看依赖树状图
//定位到冲突的依赖后排除
compile ('com.awesome:someawesomelibrary:1.0.0.0') {
exclude group: 'com.google.android', module: 'android'
}
ListView多类型的坑
//滚动时抛出索引越界异常...
//java.lang.ArrayIndexOutOfBoundsException: length=2; index=2
//注意类型不能随便定义 范围应该为 0...getViewTypeCount-1
@Override
public int getItemViewType(int position) {
DataItem dataItem = map.get(position);
return dataItem!=null?0:1;
}
复制文件
//复制文件
private void copyFile(String oldPath, String newPath) {
try {
int bytesum = 0;
int byteread = 0;
File oldfile = new File(oldPath);
if (oldfile.exists()) { //文件不存在时
InputStream inStream = new FileInputStream(oldPath); //读入原文件
FileOutputStream fs = new FileOutputStream(newPath);
byte[] buffer = new byte[1444];
int length;
while ((byteread = inStream.read(buffer)) != -1) {
bytesum += byteread; //字节数 文件大小
//System.out.println(bytesum);
fs.write(buffer, 0, byteread);
}
inStream.close();
}
} catch (Exception e) {
System.out.println("复制单个文件操作出错");
e.printStackTrace();
}
}
通过一个像素点进行进程保活
//通过一个像素点进行进程保活,被万恶的腾讯QQ带坏了
windowManager = (WindowManager) this.getSystemService(Context.WINDOW_SERVICE);
surfaceView = new SurfaceView(this);
mSurfaceHolder = surfaceView.getHolder();
WindowManager.LayoutParams layoutParams = new WindowManager.LayoutParams(
1, 1,
WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSLUCENT
);
layoutParams.gravity = Gravity.LEFT | Gravity.TOP;//放在左上角
windowManager.addView(surfaceView, layoutParams);
mSurfaceHolder.addCallback(this);//看了下,什么也没做
lv中有checkbox导致无法item点击事件无法获取焦点
cb:
android:focusable="false"
android:clickable="false"
android:focusableInTouchMode="false"
item的viewgroup中:
android:descendantFocusability="blocksDescendants"
beforeDescendants:viewgroup会优先其子类控件而获取到焦点
afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点
blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点
AlertDialog设置圆角背景
res/style
<style name="NoShadowDialog" parent="@android:style/Theme.Dialog">
<item name="android:windowIsFloating">true</item>
<item name="android:windowIsTranslucent">false</item>
<item name="android:windowNoTitle">true</item>
<item name="android:windowBackground">@android:color/transparent</item>
<item name="android:backgroundDimAmount">0.5</item>
</style>
View dialogLayout = LayoutInflater.from(OrderFoodActivity.this)
.inflate(R.layout.dialog_select_food_type, null, false);
AlertDialog alertDialog = new AlertDialog
.Builder(OrderFoodActivity.this,R.style.NoShadowDialog)
.create();
alertDialog.show();
WindowManager.LayoutParams attributes = alertDialog.getWindow().getAttributes();
attributes.width= FrameLayout.LayoutParams.WRAP_CONTENT;
attributes.height= FrameLayout.LayoutParams.WRAP_CONTENT;
alertDialog.getWindow().setAttributes(attributes);
alertDialog.setContentView(dialogLayout);
alertDialog.getWindow().clearFlags(WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM);
FragmentTabHost实现滚动效果
//FragmentTabHost的宽度必须为固定值,否则不显示,应该是测量的问题
<HorizontalScrollView
android:layout_width="1080dp"
android:layout_height="wrap_content"
android:scrollbars="none">
<LinearLayout
android:layout_width="2160dp"
android:layout_height="wrap_content"
android:orientation="vertical">
<android.support.v4.app.FragmentTabHost
android:id="@android:id/tabhost"
android:layout_width="2160dp"
android:layout_height="wrap_content"/>
</LinearLayout>
</HorizontalScrollView>
圆角背景
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="#fff"/>
<corners android:radius="6dp"/>
<stroke android:color="#b5b5b5" android:width="1dp"/>
</shape>
Tablayout
compile 'com.android.support:design:25.3.1'
<android.support.design.widget.TabLayout
android:id="@+id/tablayout_orderfood_act"
android:layout_width="0dp"
android:layout_height="50dp"
android:layout_marginTop="0dp"
app:tabIndicatorColor="#3af"//指示器颜色
app:tabSelectedTextColor="#000"//选中后文字颜色
app:tabTextColor="#000"//tab中文字颜色
app:tabMode="fixed"//是否固定
app:tabGravity="center"//tab的gravity
android:background="#fff" app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toBottomOf="@+id/linearLayout3"/>
tab_ll.addTab(tab_ll.newTab().setText("今日菜单"));
tab_ll.addTab(tab_ll.newTab().setText("我的订餐"));
tab_ll.setOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
@Override
public void onTabSelected(TabLayout.Tab tab) {
}
@Override
public void onTabUnselected(TabLayout.Tab tab) {
}
@Override
public void onTabReselected(TabLayout.Tab tab) {
}
});
ArrayList<String> datas=new ArrayList<>();
for (int i = 0; i < 2; i++) {
datas.add("哈哈"+i);
}
ArrayList<CharSequence> titles=new ArrayList<>();
titles.add("今日菜单");
titles.add("我的订餐");
vp.setAdapter(new CommonPagerAdapter<String>(OrderFoodMainActivity.this,datas,titles) {
@Override
public View getView(ViewGroup container, String data, int position) {
View view = LayoutInflater.from(mContext).inflate(R.layout.pop_scoretype, container, false);
return view;
}
});
tab_ll.setupWithViewPager(vp);
EditText输入类型的限制
<EditText
android:id="@+id/editText_username"
android:layout_width="0dp"
android:layout_height="30dp"
//通过属性, android:digits中的内容才能被输入
android:digits=
"0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
android:layout_weight="1"
android:background="@color/white"
android:inputType="text"
android:ems="10"
android:hint="账 号"
android:padding="5dp"
android:textSize="13dp">
</EditText>
//使用正则来过滤
et.addTextChangedListener(new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {
}
@Override
public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {
String content=et.getText().toString();
String regEx="[^a-zA-Z0-9]";
Pattern pattern = Pattern.compile(regEx);
Matcher matcher = pattern.matcher(content);
String trimedContent = matcher.replaceAll("").trim();
if (!content.equals(trimedContent)){
et.setText(trimedContent);
et.setSelection(trimedContent.length());
}
}
@Override
public void afterTextChanged(Editable editable) {
}
});
android:ems=“10”
android:ems = "10" 设置TextView或者Edittext的宽度为10个字符的宽度。当设置该属性后,控件显示的长度就为10个字符的长度,超出的部分将不显示。
List转换成数组
String[] strings1 = datas.toArray(new String[datas.size()]);
用asynchttpclient进行文件下载
public static void download(Context context){
String url="http://192.168.0.103:8080/SchoolPC/res/Score/11010001/11010001G0C0-1512632769413.xlsx";
String cachePath = CachePathUtils.getCachePath(context);
String fileName = url.substring(url.lastIndexOf("/") + 1);
File file = new File(cachePath,fileName);
client.get(url, new FileAsyncHttpResponseHandler(file) {
@Override
public void onFailure(int i, Header[] headers, Throwable throwable, File file) {
Logger.d("失败"+i);
}
@Override
public void onSuccess(int i, Header[] headers, File file) {
Logger.d("成功"+file.getAbsolutePath());
}
});
}
多个mime的隐式意图
<activity android:name=".iu.gridview.score.PostActivity" android:exported="true">
<intent-filter>
<data android:mimeType="application/msword"/>
<data android:mimeType="application/vnd.openxmlformats-officedocument.wordprocessingml.document"/>
<data android:mimeType="application/vnd.ms-excel"/>
<data android:mimeType="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"/>
<action android:name="android.intent.action.VIEW"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</activity>
exlv的使用方法
public class HomeworkExAdapter extends BaseExpandableListAdapter {
Context mContext;
ArrayList<JsonHomeworkBean> mDatas;
String mLoginName;
deleteCallback mDeleteCallback;
public HomeworkExAdapter(Context context, String loginName, ArrayList<JsonHomeworkBean> datas,deleteCallback callback) {
mContext = context;
mLoginName = loginName;
mDatas = datas;
mDeleteCallback=callback;
}
public void setDatas(ArrayList<JsonHomeworkBean> datas){
mDatas=datas;
notifyDataSetChanged();
}
@Override
public int getGroupCount() {
return mDatas != null ? mDatas.size() : 0;
}
@Override
public int getChildrenCount(int groupPosition) {
return mDatas.get(groupPosition).getDatas() == null ? 0 : mDatas.get(groupPosition).getDatas().size();
}
@Override
public JsonHomeworkBean getGroup(int groupPosition) {
return mDatas.get(groupPosition);
}
@Override
public JsonHomeworkBean.DatasBean getChild(int groupPosition, int childPosition) {
return mDatas.get(groupPosition).getDatas().get(childPosition);
}
@Override
public long getGroupId(int groupPosition) {
return groupPosition;
}
@Override
public long getChildId(int groupPosition, int childPosition) {
return childPosition;
}
//不知道有啥用...算啦不管了
@Override
public boolean hasStableIds() {
return false;
}
ParentViewHolder mParentViewHolder;
@Override
public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.exlv_parent_item_homework_message, parent, false);
mParentViewHolder = new ParentViewHolder(convertView);
convertView.setTag(mParentViewHolder);
} else {
mParentViewHolder = (ParentViewHolder) convertView.getTag();
}
JsonHomeworkBean jsonHomeworkBean = mDatas.get(groupPosition);
if (jsonHomeworkBean.getStatus() == 1) {
mParentViewHolder.count.setText(jsonHomeworkBean.getDatas().size() + "");
mParentViewHolder.title.setText(jsonHomeworkBean.getClassName());
}
return convertView;
}
ChildViewHolder mChildViewHolder;
@Override
public View getChildView(final int groupPosition, final int childPosition, boolean isLastChild, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = LayoutInflater.from(mContext).inflate(R.layout.exlv_child_homework_message, parent, false);
mChildViewHolder = new ChildViewHolder(convertView);
convertView.setTag(mChildViewHolder);
} else {
mChildViewHolder = (ChildViewHolder) convertView.getTag();
}
JsonHomeworkBean.DatasBean datasBean = mDatas.get(groupPosition).getDatas().get(childPosition);
if (datasBean == null)
return convertView;
List<String> imgList = datasBean.getImgList();
final int id = datasBean.getId();
mChildViewHolder.teacherName.setText(datasBean.getT_name());
mChildViewHolder.subjectName.setText(datasBean.getSubject_name());
mChildViewHolder.homeworkContent.setText(datasBean.getContent());
mChildViewHolder.time.setText(TimeUtils.getLocalTimeString(datasBean.getCreate_date().getTime()));
mChildViewHolder.delete.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (mDeleteCallback!=null)
mDeleteCallback.delete(id);
}
});
mChildViewHolder.pics.setAdapter(
new HolderAdapter<String, CommonViewHolder2>(mContext, imgList, R.layout.item_gv_single_iv) {
@Override
protected void bindDatas(CommonViewHolder holder, String data, int position) {
ImageView iv=holder.getView(R.id.iv_single_item);
String url = AsyncHttpAsynchttpUtil.uhtp + "SchoolPC/res/homeworks/" + data;
Picasso.with(mContext)
.load(url)
.fit()
.into(iv);
}
});
FixedViewUtil.setListViewHeightBasedOnChildren(mChildViewHolder.pics, 3);
return convertView;
}
@Override
public boolean isChildSelectable(int groupPosition, int childPosition) {
return false;
}
static class ParentViewHolder {
ImageView iv;
TextView title;
TextView count;
public ParentViewHolder(View rootView) {
iv = (ImageView) rootView.findViewById(R.id.iv_hint_homework_message_act_exlvparent);
title = (TextView) rootView.findViewById(R.id.tv_title_homework_message_act_exlvparent);
count = (TextView) rootView.findViewById(R.id.tv_count_homework_message_act_exlvparent);
}
}
static class ChildViewHolder {
TextView subjectName;
TextView teacherName;
TextView delete;
TextView homeworkContent;
GridView pics;
TextView time;
public ChildViewHolder(View rootView) {
subjectName = (TextView) rootView.findViewById(R.id.tv_subject_name);
teacherName = (TextView) rootView.findViewById(R.id.tv_teacher_name);
delete = (TextView) rootView.findViewById(R.id.tv_delete);
homeworkContent = (TextView) rootView.findViewById(R.id.tv_homework_content);
pics = (GridView) rootView.findViewById(R.id.gv_homework_pictures);
time = (TextView) rootView.findViewById(R.id.tv_time);
}
}
public interface deleteCallback{
void delete(int deleteId);
}
}
picasso压缩图片大小为控件大小
PopupWindow.update()
从当前设置状态更新弹出窗口的状态,如果当前正在显示,
PopupWindow在屏幕下方
public class PopupWindows extends PopupWindow {
public PopupWindows(Context mContext, View parent) {
super(mContext);
View view = View
.inflate(mContext, R.layout.item_popupwindows, null);
view.startAnimation(AnimationUtils.loadAnimation(mContext,
R.anim.fade_ins));
LinearLayout ll_popup = (LinearLayout) view
.findViewById(R.id.ll_popup);
ll_popup.startAnimation(AnimationUtils.loadAnimation(mContext,
R.anim.push_bottom_in_2));
setWidth(LayoutParams.FILL_PARENT);
setHeight(LayoutParams.FILL_PARENT);
setBackgroundDrawable(new BitmapDrawable());
setFocusable(true);
setOutsideTouchable(true);
setContentView(view);
showAtLocation(parent, Gravity.BOTTOM, 0, 0);
update();
Button bt1 = (Button) view
.findViewById(R.id.item_popupwindows_camera);
Button bt2 = (Button) view
.findViewById(R.id.item_popupwindows_Photo);
Button bt3 = (Button) view
.findViewById(R.id.item_popupwindows_cancel);
bt1.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
photo();
dismiss();
}
});
bt2.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
Intent intent = new Intent(PublishedActivity.this,
TestPicActivity.class);
startActivity(intent);
dismiss();
}
});
bt3.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
dismiss();
}
});
}
}
PopupWindow和Dialog的区别
1)AlertDialog是非阻塞线程的,Popupwindow是阻塞线程的。
2)Dialog没法设置宽为整个屏幕宽,总有点边界。Popupwindow可以。
1、Dialog及设置Dialog的动画
设置Dialog的位置和大小与加载的布局文件无关。需自己设置dialog参数。
1)设置Dialog位置:
设置位置时必须先指定Dialog的gravity属性,否则指定大小无用。
/*
* lp.x与lp.y表示相对于原始位置的偏移.
* 当参数值包含Gravity.LEFT时,对话框出现在左边,所以lp.x就表示相对左边的偏移,负值忽略.
* 当参数值包含Gravity.RIGHT时,对话框出现在右边,所以lp.x就表示相对右边的偏移,负值忽略.
* 当参数值包含Gravity.TOP时,对话框出现在上边,所以lp.y就表示相对上边的偏移,负值忽略.
* 当参数值包含Gravity.BOTTOM时,对话框出现在下边,所以lp.y就表示相对下边的偏移,负值忽略.
* 当参数值包含Gravity.CENTER_HORIZONTAL时
* ,对话框水平居中,所以lp.x就表示在水平居中的位置移动lp.x像素,正值向右移动,负值向左移动.
* 当参数值包含Gravity.CENTER_VERTICAL时
* ,对话框垂直居中,所以lp.y就表示在垂直居中的位置移动lp.y像素,正值向右移动,负值向左移动.
* gravity的默认值为Gravity.CENTER,即Gravity.CENTER_HORIZONTAL |
* Gravity.CENTER_VERTICAL.
*
* 本来setGravity的参数值为Gravity.LEFT | Gravity.TOP时对话框应出现在程序的左上角,但在
* 我手机上测试时发现距左边与上边都有一小段距离,而且垂直坐标把程序标题栏也计算在内了,
* Gravity.LEFT, Gravity.TOP, Gravity.BOTTOM与Gravity.RIGHT都是如此,据边界有一小段距离
*/
lp.y=90;
2)去标题:
[java] view plain copy
dialog.requestWindowFeature(Window.FEATURE_NO_TITLE);<span style="color:#cc0000;">//要在创建完dialog后就调用,否则报错
3)设置Dialog的宽和高
[java] view plain copy
WindowManager wm = getWindowManager();
Display display = wm.getDefaultDisplay();
android.view.WindowManager.LayoutParams lp = dialog.getWindow().getAttributes();
lp.width = display.getWidth();
lp.height =LayoutParams.WRAP_CONTENT;
dialog.getWindow().setAttributes(lp);
2、Popupwindow
1)设置显示位置特别方便:
showAsDropDown(View anchor):相对某个控件的位置(正左下方),无偏移。
showAsDropDown(View anchor, int xoff, int yoff):相对某个控件的位置,有偏移。
showAtLocation(View parent, int gravity, int x, int y):相对于父控件的位置(例如正中央Gravity.CENTER,下方Gravity.BOTTOM等),可以设置偏移或无偏移。
2)点击Popupwindow以外区域自动消失
注意一定要设置backgroundDrawable
[java] view plain copy
//参数也可以是下面这俩值
//1、getResources().getDrawable(R.drawable.abc)
//2、getWallpaper()
//当你发现有背景色时,需给布局文件设置背景色,这样即可覆盖系统自带的背景色。
pw.setBackgroundDrawable(new BitmapDrawable());
pw.setOutsideTouchable(true);
有种说法是pw.setFocusable(false);,则不点击区域以外不会消失。经测试,此种说法不对。
递归删除整个文件夹
public static String SDPATH = Environment.getExternalStorageDirectory()
+ "/formats/";
//删除文件夹
public static void deleteDir() {
File dir = new File(SDPATH);
if (dir == null || !dir.exists() || !dir.isDirectory())
return;
//通过递归删除所有的文件和文件夹
for (File file : dir.listFiles()) {
if (file.isFile())
file.delete();
else if (file.isDirectory())
deleteDir();
}
dir.delete();// 删除目录本身
}
AbsListView设置item的点击效果
ablv.setSelector(new ColorDrawable(Color.TRANSPARENT));
将一个project 变成一个module
apply plugin: 'com.android.application'
改为
apply plugin: 'com.android.library'
android {
compileSdkVersion 25
buildToolsVersion "25.0.2"
defaultConfig {
//删除applicationId
applicationId "com.junx.x5demo"
minSdkVersion 19
targetSdkVersion 25
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
与project的编辑版本要一致
Webview手机端设置
//初始化Webview
webView.loadUrl("file:///android_asset/EduMessage.html");
webView.getSettings().setJavaScriptEnabled(true);
WebSettings webSettings = webView.getSettings();
//设置是否支持缩放
webSettings.setSupportZoom(true);
//开启js
webSettings.setJavaScriptEnabled(true);
//设置打开时js可以被打开
webSettings.setJavaScriptCanOpenWindowsAutomatically(true);
//是否使用内置的缩放机制
webSettings.setBuiltInZoomControls(true);
//是否开启插件
webSettings.setPluginState(WebSettings.PluginState.ON);
//是否使用标记的中宽高
webSettings.setUseWideViewPort(true);
//设置是否考虑到屏幕宽高
webSettings.setLoadWithOverviewMode(true);
webSettings.setTextSize(WebSettings.TextSize.LARGEST);
DisplayMetrics metrics = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getMetrics(metrics);
int mDensity = metrics.densityDpi;
if (mDensity == 240) {
webSettings.setDefaultZoom(ZoomDensity.FAR);
} else if (mDensity == 160) {
webSettings.setDefaultZoom(ZoomDensity.MEDIUM);
} else if (mDensity == 120) {
webSettings.setDefaultZoom(ZoomDensity.CLOSE);
} else if (mDensity == DisplayMetrics.DENSITY_XHIGH) {
webSettings.setDefaultZoom(ZoomDensity.FAR);
} else if (mDensity == DisplayMetrics.DENSITY_TV) {
webSettings.setDefaultZoom(ZoomDensity.FAR);
}
webView.addJavascriptInterface(this,"contact");
@JavascriptInterface
public void showContent() {
webView.loadUrl("javascript:showContent('" + mData.getContent() + "')");
}
@JavascriptInterface
public void showTitle() {
webView.loadUrl("javascript:showTitle('" + mData.getTitle() + "')");
}
ListView最外部item margin失效
List中的Item的LayoutParam是直接继承自ViewPager中的LayoutParam。 不包含有margin信息。 所以在ListView中父节点设置的值会失效。
TextView设置文本过长时的省略号
<TextView
android:id="@+id/tv_content_edu_message_item"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
android:ellipsize="end"//...在末尾
android:maxLines="1"
android:text="content"
android:textSize="13dp"
/>
修改TextView的字体
.ttf
typeface = Typeface.createFromAsset(context.assets, "fonts/yummy_bread.ttf")
view.findViewById<TextView>(R.id.message).apply {
typeface = getTypeface(context)
text = message
}
修改系统的源码
修改源码:
打开java文件,切换到project视图,点击project视图上方的瞄准镜,定位到文件的位置,复制黏贴即可
ImageView加载一个文件
BitmapFactory.decodeFile(file.getAbsolutePath());
git merge 和 git merge –no-ff
git merge –no-ff 可以保存你之前的分支历史。能够更好的查看 merge历史,以及branch 状态。
git merge 则不会显示 feature,只保留单条分支记录。
WebView和Html的联合调用
//webview常用方法
webView.loadUrl("file:///android_asset/campusDynami.html");
webView.getSettings().setJavaScriptEnabled(true);
WebSettings webSettings = webView.getSettings();
webSettings.setSupportZoom(true);
webSettings.setJavaScriptEnabled(true);
webSettings.setJavaScriptCanOpenWindowsAutomatically(true);
webSettings.setBuiltInZoomControls(true);//support zoom
webSettings.setPluginState(WebSettings.PluginState.ON);//support flash
webSettings.setUseWideViewPort(true);// 这个很关键
webSettings.setLoadWithOverviewMode(true);
webSettings.setTextSize(WebSettings.TextSize.LARGEST);
DisplayMetrics metrics = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getMetrics(metrics);
int mDensity = metrics.densityDpi;
if (mDensity == 240) {
webSettings.setDefaultZoom(ZoomDensity.FAR);
} else if (mDensity == 160) {
webSettings.setDefaultZoom(ZoomDensity.MEDIUM);
} else if (mDensity == 120) {
webSettings.setDefaultZoom(ZoomDensity.CLOSE);
} else if (mDensity == DisplayMetrics.DENSITY_XHIGH) {
webSettings.setDefaultZoom(ZoomDensity.FAR);
} else if (mDensity == DisplayMetrics.DENSITY_TV) {
webSettings.setDefaultZoom(ZoomDensity.FAR);
}
<html>
<head>
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no"/>
<script type="text/javascript">
<!--第一步,在Html中定义js方法-->
function showContent(jsondata){
alert(title);
document.getElementById("personts").innerHTML=jsondata;
}
function showTitle(jsondata){
alert(title);
document.getElementById("title").innerHTML=jsondata;
}
</script>
<!--修正图片尺寸,设置宽度最大值,高度自适应防变形-->
<style>img{max-width:100% !important; height:auto !important;}</style>
</head>
<!--在加载网页时,调用js的两个方法-->
<body onload="javascript:contact.showContent();javascript:contact.showTitle()" bgcolor="#efeff4">
<!--白色圆角背景-->
<div style=" background-color:white; border-radius:10px; margin:8px; padding-left:10px; padding-right:10px; padding-top:5px; padding-bottom:5px; box-shadow:3px 3px 2px 2px lightgray; ">
<!--标题-->
<p id="title" style="font-size:0.8em;text-align: center; color:#2191FB; margin:0px;" >
</p>
<!--横线-->
<div style="background-color:#efeff4; width:100%; height:1px; margin-top:5px; margin-bottom:5px;"></div>
<div style="font-size:0.5em;" id="personts"></div>
</div>
</body>
</html>
//在java中定义回调类,调用js中方法
public final class JSObjet {
@JavascriptInterface
public void showContent() {
webView.loadUrl("javascript:showContent('" + LogConten + "')");
}
@JavascriptInterface
public void showTitle() {
webView.loadUrl("javascript:showTitle('" + LogTitle + "')");
}
}
//在webview中使用回调类
webView.addJavascriptInterface(new JSObjet(), "contact");
判断版本号
Build.VERSION.SDK_INT >= Build.VERSION_CODES.N
DatePickerDialog
Calendar c = Calendar.getInstance();
DatePickerDialog dialog = new DatePickerDialog(this, new DatePickerDialog.OnDateSetListener() {
@Override
public void onDateSet(DatePicker view, int year, int month, int dayOfMonth) {
month+=1;//因为它是从0开始计算的
}
}, c.get(Calendar.YEAR), c.get(Calendar.MONTH), c.get(Calendar.DAY_OF_MONTH));
dialog.setTitle("请选择");
dialog.setCanceledOnTouchOutside(true);
dialog.setCancelable(true);
dialog.setButton(DialogInterface.BUTTON_NEGATIVE, "取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
}
});
dialog.show();
Gson解析报错:Expected BEGIN_OBJECT but was STRING at line 1 column 39 path $
{"code":1,"info":"success","results":{"id":"1","name":"hehe"}}
results对应的应该是一个实体类,如果这个时候想把他解析为String或者List就会出现异常。
如果参考使用GsonForm处理后的数据模型,几乎不会出现问题;加入result后面的内容可能在请求时会因为某些原因会存在格式上的变化,这个时候就有出现该异常的风险。Gson中,关键字后面出现""引起来的内容将会被只认为是STRING,“{}”只被认为是类,“[]”只被认为是List,这个几乎是强制性的。
就是说如果你的实体预计是获取String的变量,但是关键字后面对应的却出现了“{”或“[”,那么这个转换将被认为是错误的,抛出异常。
解决办法:后台输出稳定的Gson格式。
解决办法2:换到新版本的FastJson
compile 'com.alibaba:fastjson:1.2.41'
用Stetho来对okhttp网络数据进行抓取
1.翻墙 chrome内核浏览器
2.gradle
//抓包工具
compile 'com.facebook.stetho:stetho:1.3.1'
compile 'com.facebook.stetho:stetho-okhttp3:1.3.1'
3. okhttp中添加插值器
okHttpClient=new OkHttpClient.Builder()
.connectTimeout(10, TimeUnit.SECONDS)
.readTimeout(10, TimeUnit.SECONDS)
.writeTimeout(10, TimeUnit.SECONDS)
.addNetworkInterceptor(new StethoInterceptor())
.build();
4. 初始化 Stetho.initializeWithDefaults(this);
5. chrome://inspect/#devices
调用.so 文件时报错has text relocations
说明编译.so文件时使用了较低版本sdk
而project 中的配置 targetSdkVersion22 大于so编译时使用的sdkversion,所以只需要把功能中
的targetSdkVersion降低即可
defaultConfig {
applicationId “com.example”
minSdkVersion 16
targetSdkVersion 22
versionCode 1
versionName “1.0”
}
SparseArray&&ArrayMap
SparseArray比HashMap更省内存,在某些条件下性能更好,主要是因为它避免了对key的自动装箱(int转为Integer类型),它内部则是通过两个数组来进行数据存储的,一个存储key,另外一个存储value,为了优化性能,它内部对数据还采取了压缩的方式来表示稀疏数组的数据,从而节约内存空间,我们从源码中可以看到key和value分别是用数组表示:
private int[] mKeys;
private Object[] mValues;
SparseArray在存储和读取数据时候,使用的是二分查找法
public void put(int key, E value);
public void delete(int key);
public void remove(int key);//内部还是调用了delete
public E get(int key);
public E get(int key, E valueIfKeyNotFound);
public int keyAt(int index);
public E valueAt(int index);
SparseArray应用场景:
虽说SparseArray性能比较好,但是由于其添加、查找、删除数据都需要先进行一次二分查找,所以在数据量大的情况下性能并不明显,将降低至少50%。
满足下面两个条件我们可以使用SparseArray代替HashMap:
数据量不大,最好在千级以内
key必须为int类型,这中情况下的HashMap可以用SparseArray代替:
ArrayMap是一个<key,value>映射的数据结构,它设计上更多的是考虑内存的优化,内部是使用两个数组进行数据存储,一个数组记录key的hash值,另外一个数组记录Value值,它和SparseArray一样,也会对key使用二分法进行从小到大排序,在添加、删除、查找数据的时候都是先使用二分查找法得到相应的index,然后通过index来进行添加、查找、删除等操作,所以,应用场景和SparseArray的一样,如果在数据量比较大的情况下,那么它的性能将退化至少50%。
如果我们要兼容aip19以下版本的话,那么导入的包需要为v4包
import android.support.v4.util.ArrayMap;
1、如果key的类型已经确定为int类型,那么使用SparseArray,因为它避免了自动装箱的过程,如果key为long类型,它还提供了一个LongSparseArray来确保key为long类型时的使用
2、如果key类型为其它的类型,则使用ArrayMap
为Fragment添加与Activity关联的点击事件
public class BlankFragment extends Fragment {
private static final String ARG_PARAM1 = "param1";
private static final String ARG_PARAM2 = "param2";
private String mParam1;
private String mParam2;
private OnFragmentInteractionListener mListener;
public BlankFragment() {
}
public static BlankFragment newInstance(String param1, String param2) {
BlankFragment fragment = new BlankFragment();
Bundle args = new Bundle();
args.putString(ARG_PARAM1, param1);
args.putString(ARG_PARAM2, param2);
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getArguments() != null) {
mParam1 = getArguments().getString(ARG_PARAM1);
mParam2 = getArguments().getString(ARG_PARAM2);
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_blank, container, false);
View fl = view.findViewById(R.id.fl);
int i = new Random().nextInt(5);
int color=0;
switch (i){
case 0:
color= Color.RED;
break;
case 1:
color=Color.BLUE;
break;
case 2:
color=Color.GREEN;
break;
case 3:
color=Color.YELLOW;
break;
case 4:
color=Color.BLACK;
break;
case 5:
color=Color.GRAY;
break;
}
fl.setBackgroundColor(color);
View btn = view.findViewById(R.id.btn);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mListener.onFragmentInteraction("你好啊");
}
});
return view;
}
public void onButtonPressed(String uri) {
if (mListener != null) {
mListener.onFragmentInteraction(uri);
}
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
if (context instanceof OnFragmentInteractionListener) {
mListener = (OnFragmentInteractionListener) context;
} else {
throw new RuntimeException(context.toString()
+ " must implement OnFragmentInteractionListener");
}
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
public interface OnFragmentInteractionListener {
void onFragmentInteraction(String uri);
}
}
FragmentPagerAdapter与FragmentStatePagerAdapter
前者会将所有的Fragment缓存在内存中,后者只缓存当前item的前后各一个Fragment
简单来说前者更流畅但吃内存
在指定位置插入字符串
StringBuffer stringBuffer = new StringBuffer(title);
stringBuffer.insert(2,"\n");
title=stringBuffer.toString();
android studio折叠代码
ctrl +;
ctrl -;
setDisplayHomeAsUpEnabled
//给actionbar添加一个左上角的返回图标
actionBar.setDisplayHomeAsUpEnabled(true)
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
toolbar.setTitle(demo.titleResId);
setSupportActionBar(toolbar);
ActionBar actionBar = getSupportActionBar();
//添加返回图标
if (actionBar!=null)
actionBar.setDisplayHomeAsUpEnabled(true);
Handler跨类通讯
1.
在a类中创建一个静态的mHandler,不进行初始化
2.
在b类中创建a的实例的时候,初始化mHandler
3.
那么就可以在a类中调用mHandler,从而实现跨类通讯
listview设置有新item时自动滚动到新的item
listview.setTranscriptMode(ListView.TRANSCRIPT_MODE_ALWAYS_SCROLL);
获取本机wifi的ip地址
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
public static String getWifiIp (Context context)
{
String wifiStr = "";
WifiManager wifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
if (wifiManager.isWifiEnabled()
&& wifiManager.getWifiState() == wifiManager.WIFI_STATE_ENABLED){
WifiInfo wifiInfo = wifiManager.getConnectionInfo();
if (wifiInfo != null) {
int ipAddress = wifiInfo.getIpAddress();
if (ipAddress == 0){
wifiStr = "";
}
else{
wifiStr = ((ipAddress & 0xff) + "." + (ipAddress >> 8 & 0xff)
+ "." + (ipAddress >> 16 & 0xff) + "." + (ipAddress >> 24 & 0xff));
}
}
}
return wifiStr;
}
显示锁的使用
public void lockMethod() {
ReentrantLock myLock = new ReentrantLock();
myLock.lock();
try{
// 受保护的代码段
//critical section
} finally {
// 可以保证发生异常 锁可以得到释放 避免死锁的发生
myLock.unlock();
}
}
viewpager加载时就实现pagerselected监听
//似乎是需要一点点的时间进行初始化
new Timer().schedule(new TimerTask() {
@Override
public void run() {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
banner.getViewPager().setCurrentItem(banner.getViewPager().getCurrentItem()-1);
banner.getViewPager().setCurrentItem(banner.getViewPager().getCurrentItem()+1);
}
});
}
},0);
手动实现FragmentTab
<RadioGroup>
<RadioButton>
<RadioButton>
</RadioGroup>
radioGroup.setOnCheckedChangeListener(RadioGroup.OnCheckedChangeListener());
获取ViewPager中的子控件
在adapter中
private View view;
@Override
//在方法会调用多次,第一次在onPageSelected前,所以可以使用该方法来在当前显示的视图进行一些操作
public void setPrimaryItem(ViewGroup container, int position, Object object) {
view= (View) object;
}
public View getPrimaryItem(){
return view;
}
getViewPager().addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
}
@Override
public void onPageSelected(int position) {
//该控件就是当前正在展示的子控件
View primaryItem = bannerAdapter.getPrimaryItem();
}
@Override
public void onPageScrollStateChanged(int state) {
}
});
//也可以
View view =pagerAdapter.getView(viewpager, position);
//在第三方框架rollviewpager中需要
View view =pagerAdapter.getView(viewpager, position%getRealCount());
IntentServiceDemo
作用:
1.在IntentService内有一个工作线程来处理耗时操作,启动IntentService的方式和启动传统的Service相同;
2.当任务执行完后,IntentService会自动停止,而不需要我们手动去控制或stopSelf();
3.可以启动IntentService多次,而每一个耗时操作会以工作队列的方式在IntentService中执行;
public class MyIntentService extends IntentService {
//action
private static final String ACTION_FOO = "com.junx.nfc3.service.action.FOO";
private static final String ACTION_BAZ = "com.junx.nfc3.service.action.BAZ";
//用于获取参数的常量的key
private static final String EXTRA_PARAM1 = "com.junx.nfc3.service.extra.PARAM1";
private static final String EXTRA_PARAM2 = "com.junx.nfc3.service.extra.PARAM2";
public MyIntentService() {
super("MyIntentService");
}
//主要重写该方法即可
@Override
protected void onHandleIntent(Intent intent) {
if (intent != null) {
//先获取action,进行判断
final String action = intent.getAction();
if (ACTION_FOO.equals(action)) {
//获取参数,进行相应操作
final String param1 = intent.getStringExtra(EXTRA_PARAM1);
final String param2 = intent.getStringExtra(EXTRA_PARAM2);
handleActionFoo(param1, param2);
} else if (ACTION_BAZ.equals(action)) {
final String param1 = intent.getStringExtra(EXTRA_PARAM1);
final String param2 = intent.getStringExtra(EXTRA_PARAM2);
handleActionBaz(param1, param2);
}
}
}
//业务逻辑
private void handleActionFoo(String param1, String param2) {
// TODO: Handle action Foo
throw new UnsupportedOperationException("Not yet implemented");
}
private void handleActionBaz(String param1, String param2) {
// TODO: Handle action Baz
throw new UnsupportedOperationException("Not yet implemented");
}
//也可以写一个方法,给予act调用
public static void startSomeTask(Context context,String action,String params1,String params2){
Intent intent = new Intent(context,MyIntentService.class);
intent.setAction(action);
intent.putExtra(EXTRA_PARAM1,params1);
intent.putExtra(EXTRA_PARAM2,params2);
context.startService(intent);
}
}
BindServiceDemo
bindService和startService的区别:
//生命周期
执行startService时,Service会经历onCreate->onStartCommand。当执行stopService时,直接调用onDestroy方法。调用者如果没有stopService,Service会一直在后台运行,下次调用者再起来仍然可以stopService;
执行bindService时,Service会经历onCreate->onBind。这个时候调用者和Service绑定在一起。调用者调用unbindService方法或者调用者Context不存在了(如Activity被finish了),Service就会调用onUnbind->onDestroy;
//多次调用时的区别
多次调用startService,该Service只能被创建一次,即该Service的onCreate方法只会被调用一次。但是每次调用startService,onStartCommand方法都会被调用。Service的onStart方法在API 5时被废弃,替代它的是onStartCommand方法;
第一次执行bindService时,onCreate和onBind方法会被调用,但是多次执行bindService时,onCreate和onBind方法并不会被多次调用,即并不会多次创建服务和绑定服务;
//后台判断最近是否打开了其他应用
public class MyService extends Service{
private LogBinder mLogBinder;
private static final int HAVE_BEEN_DESTROYED=-1;
@Override
public void onCreate() {
mLogBinder = new LogBinder();
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return mLogBinder;
}
//获取最近打开的包的应用的名字
public String getApplicationStatus() {
SystemClock.sleep(1000);
if (x!=0){
return "sdfs";
}
UsageStatsManager mUsageStatsManager = (UsageStatsManager)getApplicationContext().getSystemService(Context.USAGE_STATS_SERVICE);
long time = System.currentTimeMillis();
List<UsageStats> stats ;
stats = mUsageStatsManager.queryUsageStats(UsageStatsManager.INTERVAL_DAILY, time-60*60*1000, time);
if(stats != null) {
TreeMap<Long,UsageStats> mySortedMap = new TreeMap<Long,UsageStats>();
for (UsageStats usageStats : stats) {
mySortedMap.put(usageStats.getLastTimeUsed(),usageStats);
}
if(mySortedMap != null && !mySortedMap.isEmpty()) {
UsageStats usageStats = mySortedMap.get(mySortedMap.lastKey());
usageStats.getPackageName();
String packageName = mySortedMap.get(mySortedMap.lastKey()).getPackageName();
return packageName;
}
}
return null;
}
Thread thread;
int x=0;
public class LogBinder extends Binder{
public void show(final CallBack callBack){
thread= new Thread(new Runnable() {
@Override
public void run() {
//当线程退出时终止循环
F:while(x!=HAVE_BEEN_DESTROYED){
String name = getApplicationStatus();
//如果最近打开的应用的包名不是以下的三个,则退出死循环
if (!name.equals("com.junx.mycircleview")&&!name.equals("com.android.launcher3")
&&!name.equals("com.android.systemui"))
break F;
}
if (x!=HAVE_BEEN_DESTROYED){
Log.v("meee","检测到打开了别的应用");
callBack.onShow("");
}
}
});
thread.start();
}
}
//监听数据回调
public interface CallBack{
void onShow(String text);
}
@Override
public void onDestroy() {
x=HAVE_BEEN_DESTROYED;
}
}
//实现ServiceConnection,用于获取binder对象
public class Myconn implements ServiceConnection {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
((MyService.LogBinder) service).show(new MyService.CallBack() {
@Override
public void onShow(final String text) {
//当检测到打开其他应用后,执行相应操作
mHandler.sendEmptyMessage(233);
}
});
}
@Override
public void onServiceDisconnected(ComponentName name) {
}
}
//绑定操作
intent = new Intent(this, MyService.class);
conn = new Myconn();
bindService(intent, conn, BIND_AUTO_CREATE);
打开有使用权限的设置界面
Intent intent = new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS);
startActivity(intent);
Android Studio自定义类似syso的模版
在Setting中搜索Live Templates
选中android 点击右上方的+号
选择Live Template
输入缩写和完整内容后
记得在中下方选择applicable in Java
在使用标志位的情况下遍历文件夹
File dir = new File(Environment.getExternalStorageDirectory().getPath() + "/Pictures");
if (dir.exists()&&dir.isDirectory()){
long startTime = System.currentTimeMillis();
File[] files = dir.listFiles();
fore:for(File file:files){
if (file.length()>0&&file.getName().endsWith(".jpg")){
Log.d("TAG",getClass()+":\n"+"开始添加:"+file.toString());
Bitmap bitmap = BitmapFactory.decodeFile(file.toString());
//该方法在子线程运行,并在开始时会将Flag置false
processImg(bitmap,Uri.decode(file.toString()));
while(!Flag){
//循环等待
}
}
}
}
从assets中获取图片
private void addPictures() {
AssetManager assetsManager = getAssets();
try {
String[] list = assetsManager.list("");
for (int i = 0; i < list.length; i++) {
String fileName=list[i];
if (!TextUtils.isEmpty(fileName)&&fileName.endsWith(".jpg")){
InputStream inputStream = assetsManager.open(fileName);
FileOutputStream fileOutputStream = new FileOutputStream(Environment.getExternalStorageDirectory().getPath()
+ "/picture/"+fileName);
byte[] b = new byte[1024];
int n=0;
while((n=inputStream.read(b))!=-1){
fileOutputStream.write(b, 0, n);
}
inputStream.close();
fileOutputStream.close();
Logger.d(""+fileName+">>>"+fileOutputStream);
}
}
} catch (IOException e) {
e.printStackTrace();
}
}
Android Studio添加jniLibs目录
在module的gradle中
android {
...
sourceSets{
main{
jniLibs.srcDirs=['libs']
}
}
}
从Uri中获取图片
MediaStore.Images.Media.getBitmap(getApplicationContext().getContentResolver(),Uri.parse(max_id))
File.CreateNewFile();
当该文件的目录不存在时,抛出异常;
当该文件不存在时,创建该文件,返回true
当该文件存在时,返回false
Thread.interrupted()
//在线程受到阻塞时抛出一个中断信号,这样线程就得以退出阻塞的状态.
//如果线程被Object.wait, Thread.join和Thread.sleep三种方法之一阻塞,那么,它将接收到一个中断异常(InterruptedException),从而提早地终结被阻塞状态。
调用线程的interrupt方法,并不能真正中断线程,只是给线程做了中断状态的标志
Thread.interrupted():测试当前线程是否处于中断状态。执行后将中断状态标志为false
Thread.isInterrupte(): 测试线程Thread对象是否已经处于中断状态。但不具有清除功能
弹出框的标准处理
AlertDialog.Builder builder = new AlertDialog.Builder(Videocmp.this);
builder.setTitle("请选择图片!");
builder.setItems(items, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int item) {
boolean result= Utility.checkPermission(Videocmp.this);
if (items[item].equals("拍照")) {
if(result)
cameraIntent();
} else if (items[item].equals("图库")) {
if(result)
galleryIntent();
} else if (items[item].equals("Cancel")) {
dialog.dismiss();
}
}
});
builder.show();
对于时间的操作
//保存时间做变量
Long mTime=System.currentTimeMills();
//前一天
mTime=mTime-(1000*60*60*24);
手动绘制bitmap
Bitmap bmp = Bitmap.createBitmap(150, 150, Bitmap.Config.RGB_565);
Canvas canvas = new Canvas(bmp);
Paint paint = new Paint();
paint.setColor(Color.rgb(random.nextInt(128), random.nextInt(128), random.nextInt(128)));
paint.setTextSize(24);
paint.setFlags(Paint.ANTI_ALIAS_FLAG);
canvas.drawRect(new Rect(0, 0, 150, 150), paint);
paint.setColor(Color.WHITE);
paint.setTextAlign(Paint.Align.CENTER);
canvas.drawText(s, 75, 75, paint);
动态设置listview的高度
//在listview或者gridview外部包装一层Scrollview的时候,需要动态设置设置listview的高度.使其处于内部item的与listview的高度相等
public class FixedViewUtil {
public static void setListViewHeightBasedOnChildren(GridView listView,int col) {
// 获取listview的adapter
ListAdapter listAdapter = listView.getAdapter();
if (listAdapter == null) {
return;
}
// 固定列宽,有多少列
int totalHeight = 0;
// i每次加4,相当于listAdapter.getCount()小于等于4时 循环一次,计算一次item的高度,
// listAdapter.getCount()小于等于8时计算两次高度相加
for (int i = 0; i < listAdapter.getCount(); i += col) {
// 获取listview的每一个item
View listItem = listAdapter.getView(i, null, listView);
listItem.measure(0, 0);
// 获取item的高度和
totalHeight += listItem.getMeasuredHeight();
totalHeight += listView.getVerticalSpacing();
if (i==listAdapter.getCount()-1) {
totalHeight += listView.getVerticalSpacing();
}
}
// 获取listview的布局参数
ViewGroup.LayoutParams params = listView.getLayoutParams();
// 设置高度
params.height = totalHeight;
// 设置margin
((MarginLayoutParams) params).setMargins(10, 10, 10, 10);
// 设置参数
listView.setLayoutParams(params);
}
public static void setListViewHeightBasedOnChildren(ListView lv){
ListAdapter listAdapter = lv.getAdapter();
int listViewHeight = 0;
int adaptCount = listAdapter.getCount();
for(int i=0;i<adaptCount;i++){
View temp = listAdapter.getView(i,null,lv);
temp.measure(0,0);
listViewHeight += temp.getMeasuredHeight();
}
LayoutParams layoutParams = lv.getLayoutParams();
layoutParams.width = LayoutParams.MATCH_PARENT;
layoutParams.height = listViewHeight;
lv.setLayoutParams(layoutParams);
}
}
//也可以自定义gridview,使其占满最大
public class MyGridView extends GridView{
public MyGridView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MyGridView(Context context) {
super(context);
}
public MyGridView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, expandSpec);
}
}
ProgressBar的使用
// 方式一:new Dialog
final ProgressDialog dialog = new ProgressDialog(this);
dialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);// 设置进度条的形式为圆形转动的进度条
dialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);// 设置水平进度条
dialog.show();
// 方式二:使用静态方式创建并显示,这种进度条只能是圆形条,设置title和Message提示内容
ProgressDialog dialog2 = ProgressDialog.show(this, "提示", "正在登陆中");
比较日期或时间的大小
public static boolean firstDayIsBiggerOrEqualsSecond(String str1, String str2) {
boolean isBigger = false;
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date dt1 = null;
Date dt2 = null;
try {
dt1 = sdf.parse(str1);
dt2 = sdf.parse(str2);
} catch (ParseException e) {
e.printStackTrace();
}
if (dt1.getTime() >= dt2.getTime()) {
isBigger = true;
} else if (dt1.getTime() < dt2.getTime()) {
isBigger = false;
}
return isBigger;
}
获取当前的星期
Calendar c = Calendar.getInstance();
c.setTimeZone(TimeZone.getTimeZone("GMT+8:00"));
int mCurrentDay=c.get(Calendar.DAY_OF_WEEK);
解决ListView.setEmptyView()无效的问题
这个View有一个限制, 就是必须要在当前的View hierarchy里, 不然会不起作用.简单来说该view必须和listview在同一层中
public static void setEmptyView(ListView listview, View emptyView) {
FrameLayout emptyLayout;
if (listview.getEmptyView() == null) {
emptyLayout = new FrameLayout(listview.getContext());
emptyLayout.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
emptyView.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
emptyLayout.addView(emptyView);
emptyView.setVisibility(View.VISIBLE);
getParentView((ViewGroup) listview.getParent()).addView(emptyLayout);
listview.setEmptyView(emptyLayout);
} else {
emptyLayout = (FrameLayout) listview.getEmptyView();
emptyLayout.removeAllViews();
emptyLayout.setVisibility(View.VISIBLE);
emptyLayout.addView(emptyView);
}
}
private static ViewGroup getParentView(ViewGroup parent) {
ViewGroup tempVg = parent;
if (parent.getParent() != null && parent.getParent() instanceof ViewGroup) {
tempVg = (ViewGroup) parent.getParent();
getParentView(tempVg);
} else {
return tempVg;
}
return tempVg;
}
获取屏幕信息的方法
//依赖于手机系统,获取到的是系统的屏幕信息;
mDisplayMetrics = getResources().getDisplayMetrics();
//依赖于activity,获取到的是当前页面的屏幕的信息
WindowManager manager = (WindowManager) getSystemService(Context.WINDOW_SERVICE);
manager.getDefaultDisplay().getMetrics(dm);
输入法遮挡了输入框的解决办法
1.
<activity android:name=".MainActivity" android:windowSoftInputMode="stateVisible|adjustPan">
2.在act的oncreate()
//需要在setContentView之前调用
getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_PAN);
setContentView(R.layout.activity_main);
3.
把顶级的layout替换成ScrollView,或者说在顶级的Layout上面再加一层ScrollView的封装。这样就会把软键盘和输入框一起滚动了,软键盘会一直处于底部。
沉浸式状态栏 的实现
api4.4以后,android便支持沉浸式
1.创建res/values-v19/style.xml文件
<resources>
<style name="AppTheme" parent="Theme.AppCompat.NoActionBar">
<!--支持沉浸式效果-->
<item name="android:windowTranslucentStatus">true</item>
</style>
</resources>
2.通过反射获取状态栏的高度
public int getStatusHeight(Context context) {
int statusHeight = -1;
try {
Class clazz = Class.forName("com.android.internal.R$dimen");
Object object = clazz.newInstance();
int height = Integer.parseInt(clazz.getField("status_bar_height")
.get(object).toString());
statusHeight = context.getResources().getDimensionPixelSize(height);
} catch (Exception e) {
e.printStackTrace();
}
return statusHeight;
}
3.在布局文件的最上方设置一个imageview
//判断版本号,动态设置imageview高度和颜色
if (Build.VERSION.SDK_INT>=Build.VERSION_CODES.KITKAT){
int statusHeight = getStatusHeight(this);
ImageView iv2= (ImageView) findViewById(R.id.imageView2);
iv2.getLayoutParams().height=statusHeight;
iv2.setBackgroundColor(Color.RED);
}
保存bitmap到本地
FileOutputStream fileOutputStream = new FileOutputStream(Environment.getExternalStorageDirectory().getPath()
+ "/picture/TinyFile2Bitmap.jpg");
boolean isSuccessful = bitmap.compress(Bitmap.CompressFormat.JPEG, 20, fileOutputStream);
fileOutputStream.close();
设置TextView中字间距
android:letterSpacing="0.25" //以一个字的宽度做比例
在DDMS的复制文件出错
[2017-10-13 09:51:45 - ddms] transfer error: No such file or directory
[2017-10-13 09:51:45] Failed to pull selection: No such file or directory
文件不要使用中文命名
ViewGroup获取宽高的问题
View的机制决定了它无法通过getWidth获取宽高
但可以使用测量
获取测量宽高前需要先测量自身
sv.measure(0,0);
Log.v("meee",getClass()+":\n"+sv.getMeasuredHeight());
从相机预览帧中截取图片
private void getPreViewImage() {
mCamera.setPreviewCallback(new Camera.PreviewCallback(){
@Override
public void onPreviewFrame(byte[] data, Camera camera) {
Camera.Size size = camera.getParameters().getPreviewSize();
try{
YuvImage image = new YuvImage(data, ImageFormat.NV21, size.width, size.height, null);
if(image!=null){
ByteArrayOutputStream stream = new ByteArrayOutputStream();
image.compressToJpeg(new Rect(0, 0, size.width, size.height), 80, stream);
Bitmap bmp = BitmapFactory.decodeByteArray(stream.toByteArray(), 0, stream.size());
int i = bmp.getRowBytes() * bmp.getHeight() / 1024;
iv_student1.setImageBitmap(bmp);
stream.close();
mCamera.setPreviewCallback(null);
}
}catch(Exception ex){
Log.e("Sys","Error:"+ex.getMessage());
}
}
});
}
分割线的问题
因为View中没有重写onmeasure方法,所以不支持wrap_content(效果为match_parent)
所以需要使用imageview做分割线,支持wrap_content
设置statusbar透明
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
setTranslucentStatus(true);
}
tintManager = new SystemBarTintManager(this);
tintManager.setStatusBarTintEnabled(true);
tintManager.setStatusBarTintResource(android.R.color.transparent); //设置上方状态栏透明
}
@TargetApi(19)
private void setTranslucentStatus(boolean on) {
Window win = getWindow();
WindowManager.LayoutParams winParams = win.getAttributes();
final int bits = WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS;
if (on) {
winParams.flags |= bits;
} else {
winParams.flags &= ~bits;
}
win.setAttributes(winParams);
}
FragmentTabHost的点击事件监听
mTabHost.getTabWidget().getChildTabViewAt(0).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
}
});
Gson的使用
被fastjson的混淆坑伤了,决定用Gson:
#导包
compile 'com.google.code.gson:gson:2.8.2'
#转换
Model model = new Gson().fromJson(str, Model.class);
String jsonString = new Gson().toJson(model);//通过反射拿到的,没有get方法也可以
#属性转换
当后台返回的数据是这样的:{"address_detail":"333"}
但android的命名习惯是这样的
public class bean{
private String addressDetail;}
我们可以加上转化标签
public class bean{
@SerializedName("address_detail")
//@SerializedName(value = "emailAddress", alternate = {"email", "email_address"})
private String addressDetail;}
这样就可以解决前后台编码规范不同的问题啦,//但toJson的时候不会转换回来
#泛型的使用
很多时候,接口返回的json是这样的
{"code":"0","message":"success","data":[字段不同]}
如果我们解析字段是这样的
//每个接口都需要重写,很low
public class UserResponse {
public int code;
public String message;
public User data;
}
解决办法:
public class Result<T> {
public int code;
public String message;
public T data;
}
Model<List<ResultBean>> list = gson.fromJson(str
, new TypeToken<Model<List<ResultBean>>>() {//构造器是protect,只能以new xx(){}来创建
}.getType());
String fanxing = gson.toJson(list);//toJson支持泛型
dp转px
public class Utils {
public static int dip2px(Context context, float dp) {
float scale = context.getResources().getDisplayMetrics().density;
return (int) (dp * scale + 0.5f);
}
}
设置当前应用的版本号
一:在配置文件中修改
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.ct.parentipa"
android:versionCode="100107"
android:versionName="1.6.0">
二:在module的gradle中修改,优先级高于配置文件
android {
defaultConfig {
...
versionCode 1
versionName "1.0"
}
获取当前应用和程序的版本号
public static String getAppVersionName(Context context) {
String versionName = "";
int versioncode=0;
try {
// ---get the package info---
PackageManager pm = context.getPackageManager();
PackageInfo pi = pm.getPackageInfo(context.getPackageName(), 0);
versionName = pi.versionName;
versioncode = pi.versionCode;
if (versionName == null || versionName.length() <= 0) {
return "";
}
} catch (Exception e) {
}
return versionName;
}
private String getVersionName() throws Exception
{
// 获取packagemanager的实例
PackageManager packageManager = getPackageManager();
// getPackageName()是你当前类的包名,0代表是获取版本信息
PackageInfo packInfo = packageManager.getPackageInfo(getPackageName(),0);
String version = packInfo.versionName;
return version;
}
MD5工具类
public class MD5 {
/**
* d41d8cd98f00b204e9800998ecf8427e MD5 ("a") =
* 0cc175b9c0f1b6a831c399e269772661 MD5 ("abc") =
* 900150983cd24fb0d6963f7d28e17f72 MD5 ("message digest") =
*/
public static String getMD5(byte[] source) {
String s = null;
char hexDigits[] = { // 用来将字节转换成 16 进制表示的字符组
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };
try {
java.security.MessageDigest md = java.security.MessageDigest.getInstance("MD5");
md.update(source);
byte tmp[] = md.digest(); // MD5 的计算结果是128 位的长整数,
// 用字节表示就是 16 个字符
char str[] = new char[16 * 2]; // 每个字节都用 16 进制表示的话,使用两个字符,
int k = 0; // 表示转换结果中对应的字符位置
for (int i = 0; i < 16; i++) {
byte byte0 = tmp[i];
str[k++] = hexDigits[byte0 >>> 4 & 0xf];
str[k++] = hexDigits[byte0 & 0xf];
}
s = new String(str);
} catch (Exception e) {
e.printStackTrace();
}
return s;
}
static final int S11 = 7;
static final int S12 = 12;
static final int S13 = 17;
static final int S14 = 22;
static final int S21 = 5;
static final int S22 = 9;
static final int S23 = 14;
static final int S24 = 20;
static final int S31 = 4;
static final int S32 = 11;
static final int S33 = 16;
static final int S34 = 23;
static final int S41 = 6;
static final int S42 = 10;
static final int S43 = 15;
static final int S44 = 21;
static final byte[] PADDING = { -128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0 };
private long[] state = new long[4];
private long[] count = new long[2];
private byte[] buffer = new byte[64];
public String digestHexStr;
private byte[] digest = new byte[16];
public String getMD5ofStr(String inbuf) {
md5Init();
md5Update(inbuf.getBytes(), inbuf.length());
md5Final();
digestHexStr = "";
for (int i = 0; i < 16; i++) {
digestHexStr += byteHEX(digest[i]);
}
return digestHexStr;
}
public MD5() {
md5Init();
return;
}
private void md5Init() {
count[0] = 0L;
count[1] = 0L;
state[0] = 0x67452301L;
state[1] = 0xefcdab89L;
state[2] = 0x98badcfeL;
state[3] = 0x10325476L;
return;
}
private long F(long x, long y, long z) {
return (x & y) | ((~x) & z);
}
private long G(long x, long y, long z) {
return (x & z) | (y & (~z));
}
private long H(long x, long y, long z) {
return x ^ y ^ z;
}
private long I(long x, long y, long z) {
return y ^ (x | (~z));
}
private long FF(long a, long b, long c, long d, long x, long s, long ac) {
a += F(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private long GG(long a, long b, long c, long d, long x, long s, long ac) {
a += G(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private long HH(long a, long b, long c, long d, long x, long s, long ac) {
a += H(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private long II(long a, long b, long c, long d, long x, long s, long ac) {
a += I(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private void md5Update(byte[] inbuf, int inputLen) {
int i, index, partLen;
byte[] block = new byte[64];
index = (int) (count[0] >>> 3) & 0x3F;
if ((count[0] += (inputLen << 3)) < (inputLen << 3))
count[1]++;
count[1] += (inputLen >>> 29);
partLen = 64 - index;
if (inputLen >= partLen) {
md5Memcpy(buffer, inbuf, index, 0, partLen);
md5Transform(buffer);
for (i = partLen; i + 63 < inputLen; i += 64) {
md5Memcpy(block, inbuf, 0, i, 64);
md5Transform(block);
}
index = 0;
} else
i = 0;
md5Memcpy(buffer, inbuf, index, i, inputLen - i);
}
private void md5Final() {
byte[] bits = new byte[8];
int index, padLen;
Encode(bits, count, 8);
index = (int) (count[0] >>> 3) & 0x3f;
padLen = (index < 56) ? (56 - index) : (120 - index);
md5Update(PADDING, padLen);
md5Update(bits, 8);
Encode(digest, state, 16);
}
private void md5Memcpy(byte[] output, byte[] input, int outpos, int inpos, int len) {
int i;
for (i = 0; i < len; i++)
output[outpos + i] = input[inpos + i];
}
private void md5Transform(byte block[]) {
long a = state[0], b = state[1], c = state[2], d = state[3];
long[] x = new long[16];
Decode(x, block, 64);
a = FF(a, b, c, d, x[0], S11, 0xd76aa478L); /* 1 */
d = FF(d, a, b, c, x[1], S12, 0xe8c7b756L); /* 2 */
c = FF(c, d, a, b, x[2], S13, 0x242070dbL); /* 3 */
b = FF(b, c, d, a, x[3], S14, 0xc1bdceeeL); /* 4 */
a = FF(a, b, c, d, x[4], S11, 0xf57c0fafL); /* 5 */
d = FF(d, a, b, c, x[5], S12, 0x4787c62aL); /* 6 */
c = FF(c, d, a, b, x[6], S13, 0xa8304613L); /* 7 */
b = FF(b, c, d, a, x[7], S14, 0xfd469501L); /* 8 */
a = FF(a, b, c, d, x[8], S11, 0x698098d8L); /* 9 */
d = FF(d, a, b, c, x[9], S12, 0x8b44f7afL); /* 10 */
c = FF(c, d, a, b, x[10], S13, 0xffff5bb1L); /* 11 */
b = FF(b, c, d, a, x[11], S14, 0x895cd7beL); /* 12 */
a = FF(a, b, c, d, x[12], S11, 0x6b901122L); /* 13 */
d = FF(d, a, b, c, x[13], S12, 0xfd987193L); /* 14 */
c = FF(c, d, a, b, x[14], S13, 0xa679438eL); /* 15 */
b = FF(b, c, d, a, x[15], S14, 0x49b40821L); /* 16 */
a = GG(a, b, c, d, x[1], S21, 0xf61e2562L); /* 17 */
d = GG(d, a, b, c, x[6], S22, 0xc040b340L); /* 18 */
c = GG(c, d, a, b, x[11], S23, 0x265e5a51L); /* 19 */
b = GG(b, c, d, a, x[0], S24, 0xe9b6c7aaL); /* 20 */
a = GG(a, b, c, d, x[5], S21, 0xd62f105dL); /* 21 */
d = GG(d, a, b, c, x[10], S22, 0x2441453L); /* 22 */
c = GG(c, d, a, b, x[15], S23, 0xd8a1e681L); /* 23 */
b = GG(b, c, d, a, x[4], S24, 0xe7d3fbc8L); /* 24 */
a = GG(a, b, c, d, x[9], S21, 0x21e1cde6L); /* 25 */
d = GG(d, a, b, c, x[14], S22, 0xc33707d6L); /* 26 */
c = GG(c, d, a, b, x[3], S23, 0xf4d50d87L); /* 27 */
b = GG(b, c, d, a, x[8], S24, 0x455a14edL); /* 28 */
a = GG(a, b, c, d, x[13], S21, 0xa9e3e905L); /* 29 */
d = GG(d, a, b, c, x[2], S22, 0xfcefa3f8L); /* 30 */
c = GG(c, d, a, b, x[7], S23, 0x676f02d9L); /* 31 */
b = GG(b, c, d, a, x[12], S24, 0x8d2a4c8aL); /* 32 */
a = HH(a, b, c, d, x[5], S31, 0xfffa3942L); /* 33 */
d = HH(d, a, b, c, x[8], S32, 0x8771f681L); /* 34 */
c = HH(c, d, a, b, x[11], S33, 0x6d9d6122L); /* 35 */
b = HH(b, c, d, a, x[14], S34, 0xfde5380cL); /* 36 */
a = HH(a, b, c, d, x[1], S31, 0xa4beea44L); /* 37 */
d = HH(d, a, b, c, x[4], S32, 0x4bdecfa9L); /* 38 */
c = HH(c, d, a, b, x[7], S33, 0xf6bb4b60L); /* 39 */
b = HH(b, c, d, a, x[10], S34, 0xbebfbc70L); /* 40 */
a = HH(a, b, c, d, x[13], S31, 0x289b7ec6L); /* 41 */
d = HH(d, a, b, c, x[0], S32, 0xeaa127faL); /* 42 */
c = HH(c, d, a, b, x[3], S33, 0xd4ef3085L); /* 43 */
b = HH(b, c, d, a, x[6], S34, 0x4881d05L); /* 44 */
a = HH(a, b, c, d, x[9], S31, 0xd9d4d039L); /* 45 */
d = HH(d, a, b, c, x[12], S32, 0xe6db99e5L); /* 46 */
c = HH(c, d, a, b, x[15], S33, 0x1fa27cf8L); /* 47 */
b = HH(b, c, d, a, x[2], S34, 0xc4ac5665L); /* 48 */
a = II(a, b, c, d, x[0], S41, 0xf4292244L); /* 49 */
d = II(d, a, b, c, x[7], S42, 0x432aff97L); /* 50 */
c = II(c, d, a, b, x[14], S43, 0xab9423a7L); /* 51 */
b = II(b, c, d, a, x[5], S44, 0xfc93a039L); /* 52 */
a = II(a, b, c, d, x[12], S41, 0x655b59c3L); /* 53 */
d = II(d, a, b, c, x[3], S42, 0x8f0ccc92L); /* 54 */
c = II(c, d, a, b, x[10], S43, 0xffeff47dL); /* 55 */
b = II(b, c, d, a, x[1], S44, 0x85845dd1L); /* 56 */
a = II(a, b, c, d, x[8], S41, 0x6fa87e4fL); /* 57 */
d = II(d, a, b, c, x[15], S42, 0xfe2ce6e0L); /* 58 */
c = II(c, d, a, b, x[6], S43, 0xa3014314L); /* 59 */
b = II(b, c, d, a, x[13], S44, 0x4e0811a1L); /* 60 */
a = II(a, b, c, d, x[4], S41, 0xf7537e82L); /* 61 */
d = II(d, a, b, c, x[11], S42, 0xbd3af235L); /* 62 */
c = II(c, d, a, b, x[2], S43, 0x2ad7d2bbL); /* 63 */
b = II(b, c, d, a, x[9], S44, 0xeb86d391L); /* 64 */
state[0] += a;
state[1] += b;
state[2] += c;
state[3] += d;
}
private void Encode(byte[] output, long[] input, int len) {
int i, j;
for (i = 0, j = 0; j < len; i++, j += 4) {
output[j] = (byte) (input[i] & 0xffL);
output[j + 1] = (byte) ((input[i] >>> 8) & 0xffL);
output[j + 2] = (byte) ((input[i] >>> 16) & 0xffL);
output[j + 3] = (byte) ((input[i] >>> 24) & 0xffL);
}
}
private void Decode(long[] output, byte[] input, int len) {
int i, j;
for (i = 0, j = 0; j < len; i++, j += 4)
output[i] = b2iu(input[j]) | (b2iu(input[j + 1]) << 8) | (b2iu(input[j + 2]) << 16)
| (b2iu(input[j + 3]) << 24);
return;
}
public static long b2iu(byte b) {
return b < 0 ? b & 0x7F + 128 : b;
}
public static String byteHEX(byte ib) {
char[] Digit = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
char[] ob = new char[2];
ob[0] = Digit[(ib >>> 4) & 0X0F];
ob[1] = Digit[ib & 0X0F];
String s = new String(ob);
return s;
}
public static void main(String argv[]) {
System.out.println(MD5.getMD5("123".getBytes()));
}
}
时间戳和date的互相转换
//data转换为时间戳
public static String dateToStamp(String s) throws ParseException{
String res;
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date date = simpleDateFormat.parse(s);
long ts = date.getTime();
res = String.valueOf(ts);
return res;
}
//时间戳转换为时间
public static String stampToDate(String s){
String res;
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
long lt = new Long(s);
Date date = new Date(lt);
res = simpleDateFormat.format(date);
return res;
}
原生JSON解析
JSONObject jsonObject = new JSONObject(k);
status = jsonObject.getInt("status");
JSONArray jsonArray = jsonObject.getJSONArray("datas");
object = jsonArray.get(i);
请求时session不匹配的问题
每次请求发现过期,但在过期时进行递归回调发现偶尔又能成功
最后发现是登陆时使用的sss.com的域名 请求时是使用的192.域名,
最后造成了这个状况
通过屏幕的宽高来修改dialog的宽高
//设置提示框的宽高
Window dialogWindow = dialog.getWindow();
WindowManager m = getActivity().getWindowManager();
Display d = m.getDefaultDisplay(); // 获取屏幕宽、高用
WindowManager.LayoutParams p = dialogWindow.getAttributes(); // 获取对话框当前的参数值
p.height = (int) (d.getHeight() * 0.3); // 高度设置为屏幕的0.6
p.width = (int) (d.getWidth() * 0.8); // 宽度设置为屏幕的0.65
dialogWindow.setAttributes(p);
EditText和CheckBox的记忆性导致的问题
第一次加载Fragment时,给EditText赋值后,remove后,重新加载这个Fragment,并且给EditText重新赋值,可是页面上显示EditText仍然为原来的值。
原因:
当fragment已存在时,重新加载会执行onViewStateRestored把原有的控件数据重新赋值回来。onViewStateRestored在onActivityCreated(Bundle)后面执行,所以onViewCreated里面的mobileEt被覆盖掉了。
解决办法,重写onViewStateRestored
@Override
public void onViewStateRestored(@Nullable Bundle savedInstanceState) {
super.onViewStateRestored(savedInstanceState);
mobileEt.setText(mobile);
}
Recyclerview上拉加载更多
1.但在线性布局使用很合适,其他很难说
使用RecyclerView.OnScrollListener来监听RecyclerView判断上拉的时候是否滑动到底部,然后实现加载逻辑
2.三方:HeaderViewRecyclerAdapter
Recyclerview中布局的宽高参数设置无效的问题
如果item的根布局使用了linearlayout,framelayout会出现宽高都为wrap_content的问题
解决:
在adapter中,
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
//第一 二种inflate方式都不行
View.inflate(parent.getContext(),R.layout.item_msg_chat,null);
LayoutInflater.from(parent.getContext()).inflate(R.layout.item_msg_chat,null);
//这种才不会有这种奇怪的情况
View view = LayoutInflater.from(parent.getContext())
.inflate(R.layout.item_msg_chat, parent, false);
return new ViewHolder(view);
}
TextView自动换行
<TextView
android:maxWidth="300dp"
android:singleLine="false"
旋转Bitmap的方向
//旋转图片的方向
public static Bitmap rotateBitmapByDegree(Bitmap bm, int degree) {
Bitmap returnBm = null;
Matrix matrix = new Matrix();
matrix.postRotate(degree);
try {
returnBm = Bitmap.createBitmap(bm, 0, 0, bm.getWidth(),
bm.getHeight(), matrix, true);
} catch (OutOfMemoryError e) {
}
if (returnBm == null) {
returnBm = bm;
}
if (bm != returnBm) {
bm.recycle();
}
return returnBm;
}
网络状态的广播
权限:
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
receiver:
public class NetWorkStatusReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (action.equals(ConnectivityManager.CONNECTIVITY_ACTION)) {
Toast.makeText(context, "network changed:"+getConnectedType(context), Toast.LENGTH_LONG).show();
}
}
public int getConnectedType(Context context) {
if (context != null) {
ConnectivityManager mConnectivityManager = (ConnectivityManager) context
.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo mNetworkInfo = mConnectivityManager.getActiveNetworkInfo();
if (mNetworkInfo != null && mNetworkInfo.isAvailable()) {
return mNetworkInfo.getType();
}
}
return -1;
}
}
注册:
//通过广播监控网络状态
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
registerReceiver(new NetWorkStatusReceiver(),intentFilter);
判断当前网络是否可用
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
public boolean isNetworkConnected(Context context) {
if (context != null) {
ConnectivityManager mConnectivityManager = (ConnectivityManager) context
.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo mNetworkInfo = mConnectivityManager.getActiveNetworkInfo();
if (mNetworkInfo != null) {
return mNetworkInfo.isAvailable();
}
}
return false;
}
//获取网络类型
public static final int TYPE_NOT_NET=0;
public static final int TYPE_WIFI = 1;
public static final int TYPE_MOBILE_MMS = 2;
public static final int TYPE_MOBILE_SUPL = 3;
public static final int TYPE_MOBILE_DUN = 4;
public static final int TYPE_MOBILE_HIPRI = 5;
public static final int TYPE_WIMAX = 6;
public static final int TYPE_BLUETOOTH = 7;
public static final int TYPE_DUMMY = 8;
public static final int TYPE_ETHERNET = 9;
public static final int TYPE_MOBILE_FOTA = 10;
public static final int TYPE_MOBILE_IMS = 11;
public static final int TYPE_MOBILE_CBS = 12;
public static final int TYPE_WIFI_P2P = 13;
public static final int TYPE_MOBILE_IA = 14;
public static final int TYPE_MOBILE_EMERGENCY = 15;
public static final int TYPE_PROXY = 16;
public static final int TYPE_VPN = 17;
public int getConnectedType(Context context) {
if (context != null) {
ConnectivityManager mConnectivityManager = (ConnectivityManager) context
.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo mNetworkInfo = mConnectivityManager.getActiveNetworkInfo();
if (mNetworkInfo != null && mNetworkInfo.isAvailable()) {
return mNetworkInfo.getType();
}
}
return -1;
}
Handler清空所有任务
//清空指定任务
mHandler.removeMessages(12);
//清空所有任务
mHandler.removeCallbacksAndMessages(null);
使用FragmentTabHost时获取fragment实例
1.
String tabs[] = new String[]{"家长留言", "上课打卡", "主页", "校园信箱"};
2.
for (int i = 0; i < tabs.length; i++) {
//在这里传入的值就是tag
TabHost.TabSpec tabSpec = mTabHost.newTabSpec(tabs[i]);
tabSpec.setIndicator(getView(i));
mTabHost.addTab(tabSpec, classes[i], null);
}
3.可以通过fragmentmanager的findFragmentByTag(String tag)来获取fragmeng实例
ItemFragment2 fragment = (ItemFragment2) mManager.findFragmentByTag("上课打卡");
自定义相机的时候
/mnt/sdcard/Pictures/20170921-10:46:08.png: open failed: EINVAL (Invalid argument)
保存失败,发现保存时图片中不能带有:号
shape位图
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android" >
<!-- 圆角 -->
<corners
android:radius="9dp"
android:topLeftRadius="2dp"
android:topRightRadius="2dp"
android:bottomLeftRadius="2dp"
android:bottomRightRadius="2dp"/><!-- 设置圆角半径 -->
<!-- 渐变 -->
<gradient
android:startColor="@android:color/white"
android:centerColor="@android:color/black"
android:endColor="@android:color/black"
android:useLevel="true"
android:angle="45"
android:type="radial"
android:centerX="0"
android:centerY="0"
android:gradientRadius="90"/>
<!-- 间隔 -->
<padding
android:left="2dp"
android:top="2dp"
android:right="2dp"
android:bottom="2dp"/><!-- 各方向的间隔 -->
<!-- 大小 -->
<size
android:width="50dp"
android:height="50dp"/><!-- 宽度和高度 -->
<!-- 填充 -->
<solid
android:color="@android:color/white"/><!-- 填充的颜色 -->
<!-- 描边 -->
<stroke
android:width="2dp"
android:color="@android:color/black"
android:dashWidth="1dp"
android:dashGap="2dp"/>
</shape>
填充:设置填充的颜色
间隔:设置四个方向上的间隔
大小:设置大小
圆角:同时设置五个属性,则Radius属性无效
android:Radius="20dp" 设置四个角的半径
android:topLeftRadius="20dp" 设置左上角的半径
android:topRightRadius="20dp" 设置右上角的半径
android:bottomLeftRadius="20dp" 设置右下角的半径
android:bottomRightRadius="20dp" 设置左下角的半径
描边:dashWidth和dashGap属性,只要其中一个设置为0dp,则边框为实现边框
android:width="20dp" 设置边边的宽度
android:color="@android:color/black" 设置边边的颜色
android:dashWidth="2dp" 设置虚线的宽度
android:dashGap="20dp" 设置虚线的间隔宽度
渐变:当设置填充颜色后,无渐变效果。angle的值必须是45的倍数(包括0),仅在type="linear"有效,
Handler发送延时消息
handler.sendEmptyMessageDelayed(MESSAGE_LOGIN,5000);
handler.removeMessages(MESSAGE_LOGIN);
AlertDialog不能弹出输入法
public class SettingPop extends AlertDialog{
protected SettingPop(@NonNull Context context) {
super(context);
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.pop_setting);
//解决alertdialog不能弹出输入法的问题
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM
|WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);
}
}
Android Studio制作9patch图
右键一张png图片,Create 9Patch file
双击生成的9.图,选择拉伸区域
注:9.必须放在drawable目录下,否则报错
后缀必须为.9.png
左上为拉伸区域,右下为文字显示区域
eclipse
添加自定义类库作为依赖
1.将类库导入到adt中
import project
2.添加类库作为依赖
右键project
properties
Android>>library>>add apply
添加兼容包
下载到compat包
复制到lib下...就是这么简单
导入项目乱码文题
右键project
properties
修改编码格式,一般是UTF-8 GBK
Android Studio
Plugin Error
将错误提示滚动到最后
点击Enable support
会弹出是否restart 点击是等待其重启启动后即可
迁移Eclipse项目至Android Studio
从Eclipse中导出
1.将你的ADT插件版本升级到22.0以上。
2.在Eclipse中,选择File-->Export。
3.在弹出的导出窗口中,打开Android的文件夹,选择“Generate Gradle Build Files”。
4.选中你想要导入到Android Studio中的项目,Finish。
PS:导出的项目将会和原来的项目在同一目录,覆盖原来的同时,会新增一个叫build.gradle的文件,导入Android Studio时将首先读取这个文件。
导入到Android Studio
1.在Android Studio 中,首先关掉你当前的打开的项目。
2.在欢迎界面,点击Import Project(注:也是可以直接在菜单选择Import project的)
3.选中你在Eclipse中导出的项目,展开目录,点击build.gradle文件,然后OK
4.在之后的弹出对话框中,会要求你选择Gradle的配置,选中Use gradle wrapper.(注:也可以自定义你本机装的Gradle)
Eclipse生成apk
右键project Android Tools
Export Singed Apk
选择要生成apk的project
输入两次密码
ListView动态高度
wrap_content即可
提示找不到Header类
api23后,google移除了某些类
可以通过导入jar包的方式解决
Android Studio导入Eclipse项目运行乱码
一般是由于编码格式错误引起的
可以在module 的gradle中修改
android {
compileSdkVersion 26
buildToolsVersion "25.0.3"
android {compileOptions.encoding = "gbk"}//添加这一行,强制运行时的编码格式
但设置修改的编码格式是全局的
也可以修改某一个文件的编码格式
在android studio的右下角,可以设置编码格式
CRLF>>relode 是以不同的编码格式重新打开 convert:是以当前的文本更改代码格式为选中的编码格式
Error:Error: This fragment should provide a default constructor (a public constructor with no arguments) (com.ct.parentipa.fragment.MianFragment) [ValidFragment]
为该Fragment提供一个默认的空构造器
添加多个注解
@SuppressLint("ResourceAsColor,ValidFragment")
提示This fragment should provide a default constructor (a public constructor with no arguments) (com.example.TestFragment)
意思是应该使用bundle来传递函数
可以使用两种方法解决
1.修改形参,使用bundle传递数据
2.添加注解
@SuppressLint("ValidFragment")
public class PhotoFragment extends MyBaseFragment {
SimpleDateFormat
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");//2017-08-28
Date date = new Date();//当前时间
String dateStr = format.format(date);
测试数据
08-28 03:00:04.087 11104-11104/com.ct.teacheripa V/meee: school_id=41010004
08-28 03:00:04.087 11104-11104/com.ct.teacheripa V/meee: type=1
08-28 03:00:04.087 11104-11104/com.ct.teacheripa V/meee: date=2017-08-28
08-28 03:00:04.087 11104-11104/com.ct.teacheripa V/meee: login_name=410100045001
08-28 03:00:04.087 11104-11104/com.ct.teacheripa V/meee: class_id=41010004G55C0
08-28 03:00:04.087 11104-11104/com.ct.teacheripa V/meee: school_id=41010004
Android Studio混淆
生成签名的apk
build >generate Signed Apk>输入两次密码
基本混淆模版
在module的build.gradle
android {
......
buildTypes {
release {
//开启混淆
minifyEnabled true
// Zipalign优化
zipAlignEnabled true
// 移除无用的resource文件
shrinkResources true
//第一个是系统默认的基本混淆,第二个是自定义的混淆PLUS
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt'
}
}
}
proguard-rules.pro模版
#############################################
#
# 对于一些基本指令的添加
#
#############################################
# 代码混淆压缩比,在0~7之间,默认为5,一般不做修改
-optimizationpasses 5
# 混合时不使用大小写混合,混合后的类名为小写
-dontusemixedcaseclassnames
# 指定不去忽略非公共库的类
-dontskipnonpubliclibraryclasses
# 这句话能够使我们的项目混淆后产生映射文件
# 包含有类名->混淆后类名的映射关系
-verbose
# 指定不去忽略非公共库的类成员
-dontskipnonpubliclibraryclassmembers
# 不做预校验,preverify是proguard的四个步骤之一,Android不需要preverify,去掉这一步能够加快混淆速度。
-dontpreverify
# 保留Annotation不混淆
-keepattributes *Annotation*,InnerClasses
# 避免混淆泛型
-keepattributes Signature
# 抛出异常时保留代码行号
-keepattributes SourceFile,LineNumberTable
# 指定混淆是采用的算法,后面的参数是一个过滤器
# 这个过滤器是谷歌推荐的算法,一般不做更改
-optimizations !code/simplification/cast,!field/*,!class/merging/*
#############################################
#
# Android开发中一些需要保留的公共部分
#
#############################################
# 保留我们使用的四大组件,自定义的Application等等这些类不被混淆
# 因为这些子类都有可能被外部调用
-keep public class * extends android.app.Activity
-keep public class * extends android.app.Appliction
-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider
-keep public class * extends android.app.backup.BackupAgentHelper
-keep public class * extends android.preference.Preference
-keep public class * extends android.view.View
-keep public class com.android.vending.licensing.ILicensingService
# 保留support下的所有类及其内部类
-keep class android.support.** {*;}
# 保留继承的
-keep public class * extends android.support.v4.**
-keep public class * extends android.support.v7.**
-keep public class * extends android.support.annotation.**
# 保留R下面的资源
-keep class **.R$* {*;}
# 保留本地native方法不被混淆
-keepclasseswithmembernames class * {
native <methods>;
}
# 保留在Activity中的方法参数是view的方法,
# 这样以来我们在layout中写的onClick就不会被影响
-keepclassmembers class * extends android.app.Activity{
public void *(android.view.View);
}
# 保留枚举类不被混淆
-keepclassmembers enum * {
public static **[] values();
public static ** valueOf(java.lang.String);
}
# 保留我们自定义控件(继承自View)不被混淆
-keep public class * extends android.view.View{
*** get*();
void set*(***);
public <init>(android.content.Context);
public <init>(android.content.Context, android.util.AttributeSet);
public <init>(android.content.Context, android.util.AttributeSet, int);
}
# 保留Parcelable序列化类不被混淆
-keep class * implements android.os.Parcelable {
public static final android.os.Parcelable$Creator *;
}
# 保留Serializable序列化的类不被混淆
-keepclassmembers class * implements java.io.Serializable {
static final long serialVersionUID;
private static final java.io.ObjectStreamField[] serialPersistentFields;
!static !transient <fields>;
!private <fields>;
!private <methods>;
private void writeObject(java.io.ObjectOutputStream);
private void readObject(java.io.ObjectInputStream);
java.lang.Object writeReplace();
java.lang.Object readResolve();
}
# 对于带有回调函数的onXXEvent、**On*Listener的,不能被混淆
-keepclassmembers class * {
void *(**On*Event);
void *(**On*Listener);
}
# webView处理,项目中没有使用到webView忽略即可
-keepclassmembers class fqcn.of.javascript.interface.for.webview {
public *;
}
-keepclassmembers class * extends android.webkit.webViewClient {
public void *(android.webkit.WebView, java.lang.String, android.graphics.Bitmap);
public boolean *(android.webkit.WebView, java.lang.String);
}
-keepclassmembers class * extends android.webkit.webViewClient {
public void *(android.webkit.webView, jav.lang.String);
}
对于有些旧版本的项目并没有proguard-rules.pro,stackoverflow中有解答
Looks like your project is missing proguard files. You can add it yourself: put the proguard-rules.txt file into the app directory. It's already added to your build.gradle file so no further actions required.
shareimprove this answer
answered Jul 29 '15 at 9:52
aga
17.3k75079
3
Android Studio older version do not include proguard-rules.pro automatically, so manually adding it works...Thank you !! – Kushal Jul 29 '15 at 13:14
Android Studio关联SVN
只有安装了command line的svn可以关联as(默认不安装)
在安装svn配置选项中选择command line>>will be installed
在as中的setting >>subversion >>general>>use command line>>右边的三个点 >>找到svn.exe
忽略某些代码
Settings->Version Control>>右边的+号
一一般需要忽略.idea文件夹、.gradle文件夹、所有的build文件夹、所有的.iml文件及local.properties文件。
忽略完文件后,我们进行项目同SVN的关联,选择VCS->Import into Version Control->Share Project(Subversion);
这里说明一点,在Import into Version Control下有Import into Subversion和Share Project(Subversion)两个选项;第一个是直接将项目导入到SVN服务器上,但是这样做本地的项目同SVN服务器没有建立起关联,在导入后项目所有的文件都会变成红色,而且在要提交到SVN服务器时会提示项目不是SVN下的工作副本;第二个是将Android Studio当前项目同SVN服务器关联起来,但是并没有将项目导入到SVN服务器上,需要在完成建立连接后再次提交项目到SVN服务器。
svn和as关联简易使用说明
http://www.it165.net/pro/html/201508/51801.html
Fresco的使用
是一款控件级别的图片加载框架
//导包
compile 'com.facebook.fresco:fresco:1.5.0'
//使用方法
使用时,将显示显示图片的ImageView更改为SimpleDraweeView即可
在代码中指定图片的uri
Uri uri=Uri.parse("http://sfsf.com/a.png")
draweeView.setImageURI(uri);
//原理:
三级缓存,除内存外的缓存处理在非ui线程;
一.Bitmap缓存
1.bitmap缓存,5.0版本以后bmp缓存位于java的堆head中
4.x及以前是位于ashmem中,而不是在java的堆内存中,这意味着图片的创建和回收不会引发过多的GC
当app切换到后台时,bitmap缓存会被回收
二.内存缓存
内存缓存中存储了图片的原始压缩格式,从内存缓存中去取出的图片,在显示之前必须先解码,当后台时,内存缓存也会被清空
三.磁盘缓存
存储的也是图片的原始压缩格式,在使用前也要进行解码,即使关机磁盘缓存也不会丢失
优化效果:ram :180m>>80m
有3个线程池:
1.3个线程用于网路下载图片
2.2个线程用于磁盘文件的读写
3.2个线程用于CPU的相关操作,比如图片的解码,转换,以后后台的一些耗时操作
网络请求申请 显示已经过期
使用fiddler发现,该请求不带cookies
原因:
1.使用的是第三方的网路请求框架
如果请求的框架的对象不同,那么便不带cookies
2.后来还发现可能是模拟器的时间没有校准
解决办法:使用单例模式来封装请求框架,校准
fastJson用法
阅读别人的代码的方法
按照生命周期依次读一遍,尽量弄懂每个方法的意思,再阅读点击事件
然后再读一遍
picasso用法
compile 'com.squareup.picasso:picasso:2.5.2'
//展示默认
Picasso.with(this).load(imageUrl).placeholder(R.mipmap.default_image).into(imageView);
//请求错误时候
Picasso.with(this).load(imageUrl+"landptf").error(R.mipmap.default_image).into(imageView);
//图片填充方式
Picasso.with(this).load(imageUrl).fit().into(imageView);
//旋转图片
//以(0,0)为中心顺时针旋转45度
Picasso.with(this).load(imageUrl).rotate(45).into(imageView);
//以(64,64)为中心顺时针旋转45度
Picasso.with(this).load(imageUrl).rotate(45, 64, 64).into(imageView);
Fiddler的用法
1.获取fiddler的端口号
Tool>>Fiddler Options>>connections
2.获取本地的ip地址
alt+r>>cmd>>ipconfig
3.在模拟器中网络 >>详细设置中设置ip和端口号
注意如果fiddler没有打开||本地模拟器改变,那么模拟器是无法上网的,所用用完要把模拟器设置回去
Android Studio 版本控制后文件名称颜色的含义
绿色,已经加入控制暂未提交
红色,未加入版本控制
蓝色,加入,已提交,有改动
白色,加入,已提交,无改动
灰色:版本控制已忽略文件。
自定义代码模版
Setttings>>Live Templates
选中android
就可以自定义自己的代码模版了
Async_http请求网络框架的使用
1.导包
repositories {
mavenCentral()
}
dependencies {
compile 'com.loopj.android:android-async-http:1.4.9'
}
2.post请求使用方法
public class AsyncHttpAsynchttpUtil {
//final单例模式,让请求带上cookies
static final AsyncHttpClient client = new AsyncHttpClient();
public static void loadSchoolBuspisition(String url,
String value, final OnLodaSchoolbusPisition loadListener){
//设置post的请求参数
RequestParams params = new RequestParams();
params.put("key", value);
//使用client的post()发起请求,并使用接口进行回调
client.post(url, params, new AsyncHttpResponseHandler() {
@Override
public void onFailure(int arg0, Header[] arg1, byte[] arg2,
Throwable arg3) {
//arg0是响应码
}
@Override
public void onSuccess(int arg0, Header[] arg1, byte[] arg2) {
//arg2是数据流
String result = new String(arg2);
//通过接口进行回调
loadListener.OnFinish(list, status);
}
});
}
}
ZoomImageView(手势缩放图片框架)
github:
https://github.com/sephiroth74/ImageViewZoom
1.导包
gradle:
compile 'it.sephiroth.android.library.imagezoom:imagezoom:+'
2.使用方法
setImageBitmap( final Bitmap bitmap, Matrix matrix );
setImageBitmap( final Bitmap bitmap, Matrix matrix, float minZoom, float maxZoom );
如果你想加载一张新的bmp(the same from another ImageView ),你可以使用
Matrix matrix = mImageView1.getDisplayMatrix();
mImageView2.setImageBitmap( bitmap, matrix);
Matrix(将图片放大到占满屏幕)
//获取屏幕的宽高
DisplayMetrics dm = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getMetrics(dm);
widths = dm.widthPixels;
heights = dm.heightPixels;
//获取图片的参数
bitmap = BitmapFactory.decodeByteArray(datas, 0, datas.length, options);
Bitmap BitmapOrg = bitmap;
int width = BitmapOrg.getWidth();
int height = BitmapOrg.getHeight();
//目标宽高为屏幕的宽高
int newWidth = widths;
int newHeight = heights;
//当前宽高和屏幕宽高的比例
float scaleWidth = ((float) newWidth) / width;
float scaleHeight = ((float) newHeight) / height;
//取较小值
float bls = scaleWidth < scaleHeight ? scaleWidth : scaleHeight;
Matrix matrix = new Matrix();
matrix.postScale(bls, bls);
bitmap = Bitmap.createBitmap(BitmapOrg, 0, 0, width, height, matrix, true);
图片的边界压缩
private Bitmap yasuo(String filePath) {
BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
// 获取这个图片的宽和高
BitmapFactory.decodeFile(filePath, options); //此时返回bm为空
//设置缩放比
options.inSampleSize = 2;
//重新读入图片,注意这次要把options.inJustDecodeBounds 设为 false哦
options.inJustDecodeBounds = false;
Bitmap bitmap = BitmapFactory.decodeFile(filePath, options);
return bitmap;
}
保存图片至本地和本地图库
//保存图片至本地
public static void saveImageToGallery(Context context, Bitmap bmp) {
if (bmp==null){
Toast.makeText(context,"保存图片失败",Toast.LENGTH_SHORT).show();
return;
}
// 首先保存图片
File appDir = new File(Environment.getExternalStorageDirectory(), "Boohee");
if (!appDir.exists()) {
appDir.mkdir();
}
String fileName = System.currentTimeMillis() + ".jpg";
File file = new File(appDir, fileName);
try {
FileOutputStream fos = new FileOutputStream(file);
bmp.compress(CompressFormat.JPEG, 100, fos);
fos.flush();
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
// 其次把文件插入到系统图库
try {
MediaStore.Images.Media.insertImage(context.getContentResolver(),
file.getAbsolutePath(), fileName, null);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
//解决在部分机器缓存更新不及时问题
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
Intent mediaScanIntent = new Intent(
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
Uri contentUri = Uri.fromFile(file); //out is your output file
mediaScanIntent.setData(contentUri);
context.sendBroadcast(mediaScanIntent);
} else {
context.sendBroadcast(new Intent(
Intent.ACTION_MEDIA_MOUNTED,
Uri.parse("file://" + Environment.getExternalStorageDirectory())));
}
Toast.makeText(context, "保存成功", 1).show();
}
LinearLayout线性布局的时候异常
marginTop和marginBottom的问题
当两个相冲突的时候,不确定哪个发生作用
特别注意两个控件的margin相冲的时候
Bitmap的压缩
图片质量压缩:
OutputStream outputStream = new ByteArrayOutputStream();
//质量为0-100之间
BitmapFactory.decodeResource(res, R.drawable.studentdefault).compress(Bitmap.CompressFormat.JPEG,100,outputStream);
byte[] bytes = outputStream.toString().getBytes();
bitmap = BitmapFactory.decodeByteArray(bytes,0,bytes.length);
viewHolder.logo.setImageBitmap(bitmap);
//仅仅是改变图片的质量,但所占内存大小不变,改变的是该图片的文件占用大小
采样压缩:
BitmapFactory.Options options = new BitmapFactory.Options();
//采样率为2,间隔两个像素才采集一次像素,宽高都为原来的一半
options.inSampleSize = 2;
bitmap=BitmapFactory.decodeResource(res, R.drawable.studentdefault);
参数矩阵压缩:(这个效果好)
bitmap= BitmapFactory.decodeResource(res,R.drawable.studentdefault);
Matrix matrix = new Matrix();
matrix.setScale(0.3f,0.3f); bitmap=Bitmap.createBitmap(bitmap,0,0,bitmap.getWidth(),bitmap.getHeight(),matrix,true);
边界压缩:
private Bitmap getImage() {
BitmapFactory.Options newOpts = new BitmapFactory.Options();
newOpts.inJustDecodeBounds = false;
newOpts.inSampleSize = 2;//设置缩放比例
//重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了
Bitmap bitmap=BitmapFactory.decodeResource(getResources(), R.drawable.flash, newOpts);
return bitmap;//压缩好比例大小后再进行质量压缩
}
压缩到自定义的大小
public Bitmap createBitmapThumbnail(Bitmap bitMap,int newWidth,int newHeight) {
int width = bitMap.getWidth();
int height = bitMap.getHeight();
// 计算缩放比例
float scaleWidth = ((float) newWidth) / width;
float scaleHeight = ((float) newHeight) / height;
// 取得想要缩放的matrix参数
Matrix matrix = new Matrix();
matrix.postScale(scaleWidth, scaleHeight);
// 得到新的图片
Bitmap newBitMap = Bitmap.createBitmap(bitMap, 0, 0, width, height,
matrix, true);
return newBitMap;
}
HashMap高效率遍历
//只需要遍历一次
Iterator iterator = map.entrySet().iterator();
while(iterator.hasNext()){
Map.Entry entry = (Map.Entry) iterator.next();
String key = (String) entry.getKey();
String value = (String) entry.getValue();
}
//效率比较低,需要遍历两次
Iterator iterator = map.keySet().iterator();
while(iterator.hasNext()){
Object key = iterator.next();
Object o = map.get(key);
}
ViewPager+Fragment遇到的一个oom异常
有个坑,如果fragment的oncreatview的中inflater生成的view与父控件绑定的话,会oom
AndroidStudio创建位图
androidstudio怎么创建位图
点击module的右键>>创建xml文件
类型选择为drawable
颜色的位图:
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_selected="true" android:color="#ffFf0000"/>
<item android:state_selected="false" android:color="#ff00ff00"/>
</selector>
类的数组
与基本类型的数组不同
Class[] classes ={xx.class,x.class};
Caused by: android.view.InflateException: Binary XML file line #57: Error inflating class android.support.design.widget.FloatingActionButton
发现不使用backgroundTint便不会报错
在xml里面得用app:backgroundTint="@color/red"才有效果,不能用android:backgroundTint.
前面要加上命名空间xmlns:app="http://schemas.android.com/apk/res-auto"
设置无标题
requestWindowFeature(Window.FEATURE_NO_TITLE);
Sdk版本适配
android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.LOLLIPOP
!!! JUnit version 3.8 or later expected:
添加两个依赖
// Required -- JUnit 4 framework
testCompile 'junit:junit:4.12'
// Optional -- Mockito framework
testCompile 'org.mockito:mockito-core:1.10.19'
java.lang.RuntimeException: Method setUp in android.test.AndroidTestCase not mocked
然后又显示code-1 error
//单元测试显示code -1的error的解决办法
testOptions {
unitTests.returnDefaultValues = true
}
测试用例的编写
//创建AndroidTestCase的子类
public class TestClass extends AndroidTestCase {
public int add(int x, int y){
return x+y;
}
//测试的方法必须以add作前缀
public void testAdd(){
int x=1;
int y=2;
int result = add(x, y);
//在这里进行断言
assertEquals(4,result);
}
}
建议ignore
Android Studio 中建议过滤的文件:
- .idea 文件夹
- .gradle 文件夹
- 所有的 build 文件夹
- 所有的 .iml 文件
- local.properties 文件
Cursor的使用方法
//通过cursor将所有数据传入缩略图数组中
private void getThumbnailColumnData(Cursor cur)
{
if (cur.moveToFirst())
{
int _id;
int image_id;
String image_path;
int _idColumn = cur.getColumnIndex(Thumbnails._ID);
int image_idColumn = cur.getColumnIndex(Thumbnails.IMAGE_ID);
int dataColumn = cur.getColumnIndex(Thumbnails.DATA);
do
{
_id = cur.getInt(_idColumn);
image_id = cur.getInt(image_idColumn);
image_path = cur.getString(dataColumn);
thumbnailList.put("" + image_id, image_path);
} while (cur.moveToNext());
}
}
json的原生解析
JSONObject jsonObject = new JSONObject(k);
status = jsonObject.getInt("status");
Math.pow(2.0D, i);
//次方,第一个参数是底数,第二个参数是次方数
options.inJustDecodeBounds
options.inJustDecodeBounds = true 表示只读图片,不加载到内存中,设置这个参数为true,就不会给图片分配内存空间,但是可以获取到图片的大小等属性; 设置为false, 就是要加载这个图片.
设置inJustDecodeBounds为true后,decodeFile并不分配空间,但可计算出原始图片的长度和宽度
options.inSampleSize
//设置缩小倍数,8的话图片就为原来大小的1/8
软引用的使用
private HashMap<String, SoftReference<Bitmap>> imageCache = new HashMap<String, SoftReference<Bitmap>>();
//保存
public void put(String path, Bitmap bmp)
{
if (!TextUtils.isEmpty(path) && bmp != null)
{
imageCache.put(path, new SoftReference<Bitmap>(bmp));
}
}
//使用
public void get(String path){
SoftReference<Bitmap> reference = imageCache.get(path);
Bitmap bmp = reference.get();
if(bmp!=null){
//todo
}
}
判断sd卡的挂载状态
if (Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
}
保存图片到本地
public class FileUtils {
//保存的路径
public static String SDPATH = Environment.getExternalStorageDirectory()
+ "/formats/";
public static void saveBitmap(Bitmap bm, String picName) {
try {
//如果空文件不存在
if (!isFileExist("")) {
File tempf = createSDDir("");
}
File f =getFilePath(SDPATH, picName + ".JPEG");
//如果文件已经存在,则删除
if (f.exists()) {
f.delete();
}
//保存bmp文件到本地文件中
FileOutputStream out = new FileOutputStream(f);
bm.compress(Bitmap.CompressFormat.JPEG,100, out);
out.flush();
out.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
//传入文件名,返回路径名
public static File createSDDir(String dirName) throws IOException {
File dir = new File(SDPATH + dirName);
/* //判断sd卡挂载状态,但是都是空的
if (Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
}*/
return dir;
}
//判断文件是否存在
public static boolean isFileExist(String fileName) {
File file = new File(SDPATH + fileName);
file.isFile();
return file.exists();
}
//删除文件
public static void delFile(String fileName) {
File file = new File(SDPATH + fileName);
if (file.isFile()) {
file.delete();
}
file.exists();
}
//删除文件夹
public static void deleteDir() {
File dir = new File(SDPATH);
if (dir == null || !dir.exists() || !dir.isDirectory())
return;
//通过递归删除所有的文件和文件夹
for (File file : dir.listFiles()) {
if (file.isFile())
file.delete();
else if (file.isDirectory())
deleteDir();
}
dir.delete();// 删除目录本身
}
//判断文件是否存在
public static boolean fileIsExists(String path) {
try {
File f = new File(path);
if (!f.exists()) {
return false;
}
} catch (Exception e) {
return false;
}
return true;
}
//通过路径和文件名,返回文件
public static File getFilePath(String filePath, String fileName) {
File file = null;
makeRootDirectory(filePath);
try {
file = new File(filePath + fileName);
} catch (Exception e) {
e.printStackTrace();
}
return file;
}
//判断路径是否存在,如果不存在就创建该路径
public static void makeRootDirectory(String filePath) {
File file = null;
try {
file = new File(filePath);
if (!file.exists()) {
file.mkdir();
}
} catch (Exception e) {
}
}
}
检查身份证是否有效的工具类
public class IDCardAnddianhua {
Context context;
/*********************************** 身份证验证开始 ****************************************/
/**
* 身份证号码验证 1、号码的结构 公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为:六位数字地址码,
* 八位数字出生日期码,三位数字顺序码和一位数字校验码。 2、地址码(前六位数)
* 表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。 3、出生日期码(第七位至十四位)
* 表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。 4、顺序码(第十五位至十七位)
* 表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号, 顺序码的奇数分配给男性,偶数分配给女性。 5、校验码(第十八位数)
* (1)十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, ... , 16 ,先对前17位数字的权求和
* Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
* (2)计算模 Y = mod(S, 11) (3)通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0 X 9 8 7 6 5 4 3 2
*/
public IDCardAnddianhua(Context context) {
super();
this.context = context;
}
/**
* 功能:判断身份证是否有效
* @param IDStr
* 身份证号
* @return 有效:返回"" 无效:返回String信息
* @throws ParseException
*/
@SuppressWarnings("unchecked")
public boolean IDCardValidate(String IDStr) throws ParseException {
String errorInfo = "";// 记录错误信息
String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4",
"3", "2" };
String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7",
"9", "10", "5", "8", "4", "2" };
String Ai = "";
// ================ 号码的长度 15位或18位 ================
if (IDStr.length() != 15 && IDStr.length() != 18) {
Toast.makeText(context, "身份证号码长度应该为15位或18位", 1).show();
errorInfo = "身份证号码长度应该为15位或18位。";
return false;
}
// =======================(end)========================
// ================ 数字 除最后以为都为数字 ================
if (IDStr.length() == 18) {
Ai = IDStr.substring(0, 17);
} else if (IDStr.length() == 15) {
Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
}
if (isNumeric(Ai) == false) {
Toast.makeText(context, "身份证15位号码都应为数字 ; 18位号码除最后一位外,都应为数字", 1).show();
return false;
}
// =======================(end)========================
// ================ 出生年月是否有效 ================
String strYear = Ai.substring(6, 10);// 年份
String strMonth = Ai.substring(10, 12);// 月份
String strDay = Ai.substring(12, 14);// 月份
if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) {
Toast.makeText(context, "身份证生日无效", 1).show();
return false;
}
GregorianCalendar gc = new GregorianCalendar();
SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
try {
if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150
|| (gc.getTime().getTime() - s.parse(
strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {
Toast.makeText(context, "身份证生日不在有效范围", 1).show();
return false;
}
} catch (NumberFormatException e) {
e.printStackTrace();
} catch (java.text.ParseException e) {
e.printStackTrace();
}
if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
Toast.makeText(context, "身份证月份无效", 1).show();
return false;
}
if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
Toast.makeText(context, "身份证月份无效", 1).show();
return false;
}
// =====================(end)=====================
// ================ 地区码时候有效 ================
Hashtable h = GetAreaCode();
if (h.get(Ai.substring(0, 2)) == null) {
// errorInfo = "身份证地区编码错误。";
Toast.makeText(context, "身份证地区编码错误", 1).show();
return false;
}
// ==============================================
// ================ 判断最后一位的值 ================
int TotalmulAiWi = 0;
for (int i = 0; i < 17; i++) {
TotalmulAiWi = TotalmulAiWi
+ Integer.parseInt(String.valueOf(Ai.charAt(i)))
* Integer.parseInt(Wi[i]);
}
int modValue = TotalmulAiWi % 11;
String strVerifyCode = ValCodeArr[modValue];
Ai = Ai + strVerifyCode;
if (IDStr.length() == 18) {
if (Ai.equals(IDStr) == false) {
Toast.makeText(context, "身份证无效,不是合法的身份证号码", 1).show();
return false;
}
} else {
return true;
}
// =====================(end)=====================
return true;
}
/**
* 功能:设置地区编码
* @return Hashtable 对象
*/
@SuppressWarnings("unchecked")
private static Hashtable GetAreaCode() {
Hashtable hashtable = new Hashtable();
hashtable.put("11", "北京");
hashtable.put("12", "天津");
hashtable.put("13", "河北");
hashtable.put("14", "山西");
hashtable.put("15", "内蒙古");
hashtable.put("21", "辽宁");
hashtable.put("22", "吉林");
hashtable.put("23", "黑龙江");
hashtable.put("31", "上海");
hashtable.put("32", "江苏");
hashtable.put("33", "浙江");
hashtable.put("34", "安徽");
hashtable.put("35", "福建");
hashtable.put("36", "江西");
hashtable.put("37", "山东");
hashtable.put("41", "河南");
hashtable.put("42", "湖北");
hashtable.put("43", "湖南");
hashtable.put("44", "广东");
hashtable.put("45", "广西");
hashtable.put("46", "海南");
hashtable.put("50", "重庆");
hashtable.put("51", "四川");
hashtable.put("52", "贵州");
hashtable.put("53", "云南");
hashtable.put("54", "西藏");
hashtable.put("61", "陕西");
hashtable.put("62", "甘肃");
hashtable.put("63", "青海");
hashtable.put("64", "宁夏");
hashtable.put("65", "新疆");
hashtable.put("71", "台湾");
hashtable.put("81", "香港");
hashtable.put("82", "澳门");
hashtable.put("91", "国外");
return hashtable;
}
/**
* 功能:判断字符串是否为数字
*
* @param str
* @return
*/
private static boolean isNumeric(String str) {
Pattern pattern = Pattern.compile("[0-9]*");
Matcher isNum = pattern.matcher(str);
if (isNum.matches()) {
return true;
} else {
return false;
}
}
/**
* 功能:判断字符串是否为日期格式
* @param str
* @return
*/
public static boolean isDate(String strDate) {
Pattern pattern = Pattern
.compile("^((\\d{2}(([02468][048])|([13579][26]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])))))|(\\d{2}(([02468][1235679])|([13579][01345789]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|(1[0-9])|(2[0-8]))))))(\\s(((0?[0-9])|([1-2][0-3]))\\:([0-5]?[0-9])((\\s)|(\\:([0-5]?[0-9])))))?$");
Matcher m = pattern.matcher(strDate);
if (m.matches()) {
return true;
} else {
return false;
}
}
}
Semaphore的使用
public class SemaphoreTest {
public static void main(String[] args) {
// 线程池
ExecutorService exec = Executors.newCachedThreadPool();
// 只能5个线程同时访问
final Semaphore semp = new Semaphore(5);
// 模拟20个客户端访问
for (int index = 0; index < 20; index++) {
final int NO = index;
Runnable run = new Runnable() {
public void run() {
try {
// 获取许可
semp.acquire();
Thread.sleep((long) (Math.random() * 10000));
// 访问完后,释放 ,如果屏蔽下面的语句,则在控制台只能打印5条记录,之后线程一直阻塞
semp.release();
} catch (InterruptedException e) {
}
}
};
exec.execute(run);
}
// 退出线程池
exec.shutdown();
}
}
LruCache的使用
// 获取应用的最大可用内存
int maxMemory = (int) Runtime.getRuntime().maxMemory();
int cacheMemory = maxMemory / 8;
//将LruCache的最大容量设置为应用最大容量的1/8,
//它像是一个链表一样,当容量超过最大容量的时候,将最先添加的容量删除直至小于最大容量
//底层也是用hashmap实现的,没有软引用
LruCache<String, Bitmap> mLruCache = new LruCache<String, Bitmap>(cacheMemory) {
@Override
//返回每个item的大小
protected int sizeOf(String key, Bitmap value) {
return value.getRowBytes() * value.getHeight();
}
};
子线程和looper的使用
// 后台轮询线程
mPoolThread = new Thread() {
@Override
public void run() {
Looper.prepare();
//todo
Looper.loop();
}
};
mPoolThread.start();
从网络下载图片并保存
public static boolean downloadImgByUrl(String urlStr, File file) {
FileOutputStream fos = null;
InputStream is = null;
try {
URL url = new URL(urlStr);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
is = conn.getInputStream();
fos = new FileOutputStream(file);
byte[] buf = new byte[512];
int len = 0;
while ((len = is.read(buf)) != -1) {
fos.write(buf, 0, len);
}
fos.flush();
return true;
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (is != null)
is.close();
} catch (IOException e) {
}
try {
if (fos != null)
fos.close();
} catch (IOException e) {
}
}
return false;
}
获取ImagrView的宽高
public static ImageSize getImageViewSize(ImageView imageView) {
ImageSize imageSize = new ImageSize();
DisplayMetrics displayMetrics = imageView.getContext().getResources()
.getDisplayMetrics();
LayoutParams lp = imageView.getLayoutParams();
int width = imageView.getWidth();// 获取imageview的实际宽度
if (width <= 0) {
// 获取imageview在layout中声明的宽度
width = lp.width;
}
if (width <= 0) {
// 检查最大值
//width = imageView.getMaxWidth();
width = getImageViewFieldValue(imageView, "mMaxWidth");
}
if (width <= 0) {
//获取iv的矩阵的宽度
width = displayMetrics.widthPixels;
}
int height = imageView.getHeight();// 获取imageview的实际高度
if (height <= 0) {
// 获取imageview在layout中声明的宽度
height = lp.height;
}
if (height <= 0) {
// 检查最大值
height = getImageViewFieldValue(imageView, "mMaxHeight");
}
if (height <= 0) {
//获取iv的矩阵宽度
height = displayMetrics.heightPixels;
}
imageSize.width = width;
imageSize.height = height;
return imageSize;
}
通过空间的宽高进行图片的压缩
bitmap = BitmapFactory.decodeStream(is, null, opts);
int width = bitmap.getWidth();
int height = bitmap.getHeight();
int newWidth = imageViewSize.width;
int newHeight = imageViewSize.height;
float scaleWidth = ((float) newWidth) / width;
float scaleHeight = ((float) newHeight) / height;
float bls = scaleWidth > scaleHeight ? scaleWidth : scaleHeight;
Matrix matrix = new Matrix();
matrix.postScale(bls, bls);
bitmap = Bitmap.createBitmap(bitmap, 0, 0, width, height, matrix, false);
通过传入的参数,获取需要的缩放比
public static int caculateInSampleSize(Options options, int reqWidth,
int reqHeight) {
int width = options.outWidth;
int height = options.outHeight;
int inSampleSize = 1;
if (width > reqWidth || height > reqHeight) {
int widthRadio = Math.round(width * 1.0f / reqWidth);
int heightRadio = Math.round(height * 1.0f / reqHeight);
inSampleSize = Math.max(widthRadio, heightRadio);
}
return inSampleSize;
}
进行缩放压缩
protected Bitmap decodeSampledBitmapFromPath(String path, int width,
int height) {
// 获得图片的宽和高,并不把图片加载到内存中
BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeFile(path, options);
options.inSampleSize = caculateInSampleSize(options,
width, height);
// 使用获得到的InSampleSize再次解析图片
options.inJustDecodeBounds = false;
Bitmap bitmap = BitmapFactory.decodeFile(path, options);
return bitmap;
}
通过反射获取imageview的属性值
private static int getImageViewFieldValue(Object object, String fieldName) {
int value = 0;
try {
Field field = ImageView.class.getDeclaredField(fieldName);
field.setAccessible(true);
int fieldValue = field.getInt(object);
if (fieldValue > 0 && fieldValue < Integer.MAX_VALUE) {
value = fieldValue;
}
} catch (Exception e) {
}
return value;
}
获取图片的缓存地址
public File getDiskCacheDir(Context context, String uniqueName) {
String cachePath;
if (Environment.MEDIA_MOUNTED.equals(Environment
.getExternalStorageState())) {
cachePath = context.getExternalCacheDir().getPath();
} else {
cachePath = context.getCacheDir().getPath();
}
return new File(cachePath + File.separator + uniqueName);
}
将Byte[]转换成2位16进制
public String bytes2hex02(byte[] bytes) {
StringBuilder sb = new StringBuilder();
String tmp = null;
for (byte b : bytes) {
// 将每个字节与0xFF进行与运算,然后转化为10进制,然后借助于Integer再转化为16进制
tmp = Integer.toHexString(0xFF & b);
if (tmp.length() == 1)// 每个字节8为,转为16进制标志,2个16进制位
{
tmp = "0" + tmp;
}
sb.append(tmp);
}
return sb.toString();
}
图片的三级缓存
public class loaderBitmap {
//内存缓存
static LruCache<String, Bitmap> lruCache = null;
static {
//LruCache的大小设置为4m
int maxsize = 4 * 1024 * 1024;
lruCache = new LruCache<String, Bitmap>(maxsize) {
@Override
protected int sizeOf(String key, Bitmap value) {
return value.getRowBytes() * value.getHeight();
}
};
}
/**
* @param context
* @param url
* @return
*/
public static Bitmap setBitmapFromCache(Context context, String url) {
if (TextUtils.isEmpty(url)) {
return null;
}
Bitmap bitmap = null;
//先从内存缓存中查找图片资源
bitmap = getBitmapFromMemotyCache(url);
if (bitmap != null) {
return bitmap;
}
//再从应用的缓存目录中查找图片
bitmap = getBitmapFormFileCache(context, url);
if (bitmap != null) {
return bitmap;
}
//再从网络上去异步请求
bitmap = getBitmapAsyncLoad(url, context);
if (bitmap != null) {
return bitmap;
}
return null;
}
//从缓存文件中取数据
private static Bitmap getBitmapFormFileCache(Context context, String url) {
Bitmap bitmap = null;
String fileName = url.substring(url.lastIndexOf("/") + 1);
//获得应用的缓存目录
File cacheDir = null;
try {
cacheDir = context.getCacheDir();
} catch (Exception e) {
e.printStackTrace();
}
if (cacheDir != null) {
File[] files = cacheDir.listFiles();
for (int i = 0; i < files.length; i++) {
if (files[i].getName().equals(fileName)) {
bitmap = BitmapFactory.decodeFile(files[i].getAbsolutePath());
return bitmap;
}
}
}
return null;
}
//从lrucache中取bmp
private static Bitmap getBitmapFromMemotyCache(String url) {
Bitmap bitmap = null;
if (url != null) {
bitmap = lruCache.get(url);
}
return bitmap;
}
//从网络中请求数据
private static Bitmap getBitmapAsyncLoad(String url,
Context context) {
MyAsyncTask task = new MyAsyncTask(context);
task.execute(url);
return null;
}
//异步任务
static class MyAsyncTask extends AsyncTask<String, Void, Bitmap> {
Context context;
public MyAsyncTask(Context context) {
this.context = context;
}
@Override
protected Bitmap doInBackground(String... params) {
Bitmap bitmap = null;
String path = params[0];
try {
URL url = new URL(path);
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
connection.setConnectTimeout(5000);
connection.setRequestMethod("GET");
connection.setDoInput(true);
if (connection.getResponseCode() == 200) {
InputStream inputStream = connection.getInputStream();
//把返回的图片资源做按比例缩放的处理
bitmap = CompressBitmap(inputStream);
//存入缓存
if (bitmap != null) {
lruCache.put(path, bitmap);
bitmapCacheFile(bitmap, path, context);
}
}
} catch (Exception e) {
// TODO: handle exception
}
return null;
}
@Override
protected void onPostExecute(Bitmap result) {
super.onPostExecute(result);
}
//保存到本地缓存
private void bitmapCacheFile(Bitmap bitmap, String path,
Context context2) throws FileNotFoundException {
File cacheFile = context2.getCacheDir();
// TODO Auto-generated method stub
if (!cacheFile.exists()) {
cacheFile.mkdirs();
}
String fileName = path.substring(path.lastIndexOf("/") + 1);
File file = new File(cacheFile, fileName);
OutputStream stream = new FileOutputStream(file);
bitmap.compress(CompressFormat.JPEG, 100, stream);
}
//对图片进行压缩
private Bitmap CompressBitmap(InputStream inputStream) {
byte[] datas = StreamUtil.getBytesFromStream(inputStream);
BitmapFactory.Options options = new BitmapFactory.Options();
//只获取图片的参数
options.inJustDecodeBounds = true;
BitmapFactory.decodeByteArray(datas, 0, datas.length, options);
//边界的宽高
int outWidth = options.outWidth;
int outHeight = options.outHeight;
int targetWidth = 70;
int targetHeight = 70;
int wbl = outWidth / targetWidth;
int hbl = outHeight / targetHeight;
int bl = wbl > hbl ? wbl : hbl;
if (bl < 0) {
bl = 1;
}
options.inSampleSize = bl;
options.inJustDecodeBounds = false;
Bitmap bitmap = BitmapFactory.decodeByteArray(datas, 0, datas.length, options);
return bitmap;
}
}
}
MD5工具类
public class MD5 {
/**
* MD5的算法在RFC1321 中定 在RFC 1321中,给出了Test suite用来验你的实现是否正确: MD5 ("") =
* d41d8cd98f00b204e9800998ecf8427e MD5 ("a") =
* 0cc175b9c0f1b6a831c399e269772661 MD5 ("abc") =
* 900150983cd24fb0d6963f7d28e17f72 MD5 ("message digest") =
* f96b697d7cb7938d525a2f31aaf161d0 MD5 ("abcdefghijklmnopqrstuvwxyz") =
* c3fcd3d76192e4007dfb496cca67e13b
*/
//传入一个字节数组,返回一个md5的字符串
public static String getMD5(byte[] source) {
String s = null;
//16进制的数
char hexDigits[] = {
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };
try {
java.security.MessageDigest md = java.security.MessageDigest.getInstance("MD5");
md.update(source);
byte tmp[] = md.digest();
//MD5的计算结果是128证数,也就是16个字节
//每个字节等于2个16进制数字
char str[] = new char[16 * 2];
// 表示转换结果中对应的字符位置
int k = 0;
for (int i = 0; i < 16; i++) {
byte byte0 = tmp[i]; // 取第 i 个字
// >>>为逻辑右移,将符号位置一起移动
// 取字节中高4 位的数字转换,
str[k++] = hexDigits[byte0 >>> 4 & 0xf];
// 取字节中低4 位的数字转换
str[k++] = hexDigits[byte0 & 0xf];
}
// 换后的结果转换为字符串
s = new String(str);
} catch (Exception e) {
e.printStackTrace();
}
return s;
}
/*
* 下面这些S11-S44实际上是 4*4的矩阵,在原始的C实现中是 #define 实现的, 这里把他们实现成为static
* final是表示了只读,切能在同一个进程空间内的多 Instance间共
*/
static final int S11 = 7;
static final int S12 = 12;
static final int S13 = 17;
static final int S14 = 22;
static final int S21 = 5;
static final int S22 = 9;
static final int S23 = 14;
static final int S24 = 20;
static final int S31 = 4;
static final int S32 = 11;
static final int S33 = 16;
static final int S34 = 23;
static final int S41 = 6;
static final int S42 = 10;
static final int S43 = 15;
static final int S44 = 21;
static final byte[] PADDING = { -128, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0 };
/*
* 下面的三个成员是MD5计算过程中用到的3个核心数据,在原始的C实现 被定义到MD5_CTX结构
*/
private long[] state = new long[4]; // state (ABCD)
private long[] count = new long[2]; // number of bits, modulo 2^64 (lsb
// first)
private byte[] buffer = new byte[64]; // input buffer
/*
* digestHexStr是MD5的唯一一个公共成员,是最新一次计算结果的 16进制ASCII表示.
*/
public String digestHexStr;
/*
* digest,是最新一次计算结果的2进制内部表示,表 128bit的MD5 .
*/
private byte[] digest = new byte[16];
/*
* getMD5ofStr是类MD5 主要的公共方法,入口参数是你想要进行MD5变换的字符串
* 返回的是变换完的结果,这个结果是从公共成员digestHexStr取得的.
*/
public String getMD5ofStr(String inbuf) {
md5Init();
md5Update(inbuf.getBytes(), inbuf.length());
md5Final();
digestHexStr = "";
for (int i = 0; i < 16; i++) {
digestHexStr += byteHEX(digest[i]);
}
return digestHexStr;
}
// 这是MD5这个类的标准结构函数,JavaBean 求有 个public的并且没有参数的构 ?函
public MD5() {
md5Init();
return;
}
/* md5Init是个初始化函数,初始化核心变量,装入标准的幻 */
private void md5Init() {
count[0] = 0L;
count[1] = 0L;
// 魔法数字:初始化常量
state[0] = 0x67452301L;
state[1] = 0xefcdab89L;
state[2] = 0x98badcfeL;
state[3] = 0x10325476L;
return;
}
/*
* F, G, H ,I 4个基本的MD5函数,在原始的MD5的C实现中,由于他们
* 单的位运算,可能出于效率的考虑把他们实现成了宏,在java中,我们把他 实现成了private方法,名字保持了原来C中的
*/
private long F(long x, long y, long z) {
return (x & y) | ((~x) & z);
}
private long G(long x, long y, long z) {
return (x & z) | (y & (~z));
}
private long H(long x, long y, long z) {
return x ^ y ^ z;
}
private long I(long x, long y, long z) {
return y ^ (x | (~z));
}
/*
* FF,GG,HH和II将调用F,G,H,I进行近一步变 FF, GG, HH, and II transformations for
* rounds 1, 2, 3, and 4. Rotation is separate from addition to prevent
* recomputation.
*/
private long FF(long a, long b, long c, long d, long x, long s, long ac) {
a += F(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private long GG(long a, long b, long c, long d, long x, long s, long ac) {
a += G(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private long HH(long a, long b, long c, long d, long x, long s, long ac) {
a += H(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
private long II(long a, long b, long c, long d, long x, long s, long ac) {
a += I(b, c, d) + x + ac;
a = ((int) a << s) | ((int) a >>> (32 - s));
a += b;
return a;
}
/*
* md5Update是MD5的主计算过程,inbuf是要变换的字节串,inputlen是长度,这个
* 函数由getMD5ofStr调用,调用之前需要调用md5init,因此把他设计成private
*/
private void md5Update(byte[] inbuf, int inputLen) {
int i, index, partLen;
byte[] block = new byte[64];
index = (int) (count[0] >>> 3) & 0x3F;
if ((count[0] += (inputLen << 3)) < (inputLen << 3))
count[1]++;
count[1] += (inputLen >>> 29);
partLen = 64 - index;
// Transform as many times as possible.
if (inputLen >= partLen) {
md5Memcpy(buffer, inbuf, index, 0, partLen);
md5Transform(buffer);
for (i = partLen; i + 63 < inputLen; i += 64) {
md5Memcpy(block, inbuf, 0, i, 64);
md5Transform(block);
}
index = 0;
} else
i = 0;
md5Memcpy(buffer, inbuf, index, i, inputLen - i);
}
/*
* md5Final整理和填写输出结
*/
private void md5Final() {
byte[] bits = new byte[8];
int index, padLen;
// /* Save number of bits */
Encode(bits, count, 8);
// /* Pad out to 56 mod 64.
index = (int) (count[0] >>> 3) & 0x3f;
padLen = (index < 56) ? (56 - index) : (120 - index);
md5Update(PADDING, padLen);
// /* Append length (before padding) */
md5Update(bits, 8);
// /* Store state in digest */
Encode(digest, state, 16);
}
/*
* md5Memcpy是个内部使用的byte数组的块拷贝函数,从input的inpos 始把len长度 字节拷贝到output的outpos位置
*/
private void md5Memcpy(byte[] output, byte[] input, int outpos, int inpos, int len) {
int i;
for (i = 0; i < len; i++)
output[outpos + i] = input[inpos + i];
}
/*
* md5Transform是MD5核心变换程式,有md5Update调用,block是分块的原始字节
*/
private void md5Transform(byte block[]) {
long a = state[0], b = state[1], c = state[2], d = state[3];
long[] x = new long[16];
Decode(x, block, 64);
/* Round 1 */
a = FF(a, b, c, d, x[0], S11, 0xd76aa478L); /* 1 */
d = FF(d, a, b, c, x[1], S12, 0xe8c7b756L); /* 2 */
c = FF(c, d, a, b, x[2], S13, 0x242070dbL); /* 3 */
b = FF(b, c, d, a, x[3], S14, 0xc1bdceeeL); /* 4 */
a = FF(a, b, c, d, x[4], S11, 0xf57c0fafL); /* 5 */
d = FF(d, a, b, c, x[5], S12, 0x4787c62aL); /* 6 */
c = FF(c, d, a, b, x[6], S13, 0xa8304613L); /* 7 */
b = FF(b, c, d, a, x[7], S14, 0xfd469501L); /* 8 */
a = FF(a, b, c, d, x[8], S11, 0x698098d8L); /* 9 */
d = FF(d, a, b, c, x[9], S12, 0x8b44f7afL); /* 10 */
c = FF(c, d, a, b, x[10], S13, 0xffff5bb1L); /* 11 */
b = FF(b, c, d, a, x[11], S14, 0x895cd7beL); /* 12 */
a = FF(a, b, c, d, x[12], S11, 0x6b901122L); /* 13 */
d = FF(d, a, b, c, x[13], S12, 0xfd987193L); /* 14 */
c = FF(c, d, a, b, x[14], S13, 0xa679438eL); /* 15 */
b = FF(b, c, d, a, x[15], S14, 0x49b40821L); /* 16 */
/* Round 2 */
a = GG(a, b, c, d, x[1], S21, 0xf61e2562L); /* 17 */
d = GG(d, a, b, c, x[6], S22, 0xc040b340L); /* 18 */
c = GG(c, d, a, b, x[11], S23, 0x265e5a51L); /* 19 */
b = GG(b, c, d, a, x[0], S24, 0xe9b6c7aaL); /* 20 */
a = GG(a, b, c, d, x[5], S21, 0xd62f105dL); /* 21 */
d = GG(d, a, b, c, x[10], S22, 0x2441453L); /* 22 */
c = GG(c, d, a, b, x[15], S23, 0xd8a1e681L); /* 23 */
b = GG(b, c, d, a, x[4], S24, 0xe7d3fbc8L); /* 24 */
a = GG(a, b, c, d, x[9], S21, 0x21e1cde6L); /* 25 */
d = GG(d, a, b, c, x[14], S22, 0xc33707d6L); /* 26 */
c = GG(c, d, a, b, x[3], S23, 0xf4d50d87L); /* 27 */
b = GG(b, c, d, a, x[8], S24, 0x455a14edL); /* 28 */
a = GG(a, b, c, d, x[13], S21, 0xa9e3e905L); /* 29 */
d = GG(d, a, b, c, x[2], S22, 0xfcefa3f8L); /* 30 */
c = GG(c, d, a, b, x[7], S23, 0x676f02d9L); /* 31 */
b = GG(b, c, d, a, x[12], S24, 0x8d2a4c8aL); /* 32 */
/* Round 3 */
a = HH(a, b, c, d, x[5], S31, 0xfffa3942L); /* 33 */
d = HH(d, a, b, c, x[8], S32, 0x8771f681L); /* 34 */
c = HH(c, d, a, b, x[11], S33, 0x6d9d6122L); /* 35 */
b = HH(b, c, d, a, x[14], S34, 0xfde5380cL); /* 36 */
a = HH(a, b, c, d, x[1], S31, 0xa4beea44L); /* 37 */
d = HH(d, a, b, c, x[4], S32, 0x4bdecfa9L); /* 38 */
c = HH(c, d, a, b, x[7], S33, 0xf6bb4b60L); /* 39 */
b = HH(b, c, d, a, x[10], S34, 0xbebfbc70L); /* 40 */
a = HH(a, b, c, d, x[13], S31, 0x289b7ec6L); /* 41 */
d = HH(d, a, b, c, x[0], S32, 0xeaa127faL); /* 42 */
c = HH(c, d, a, b, x[3], S33, 0xd4ef3085L); /* 43 */
b = HH(b, c, d, a, x[6], S34, 0x4881d05L); /* 44 */
a = HH(a, b, c, d, x[9], S31, 0xd9d4d039L); /* 45 */
d = HH(d, a, b, c, x[12], S32, 0xe6db99e5L); /* 46 */
c = HH(c, d, a, b, x[15], S33, 0x1fa27cf8L); /* 47 */
b = HH(b, c, d, a, x[2], S34, 0xc4ac5665L); /* 48 */
/* Round 4 */
a = II(a, b, c, d, x[0], S41, 0xf4292244L); /* 49 */
d = II(d, a, b, c, x[7], S42, 0x432aff97L); /* 50 */
c = II(c, d, a, b, x[14], S43, 0xab9423a7L); /* 51 */
b = II(b, c, d, a, x[5], S44, 0xfc93a039L); /* 52 */
a = II(a, b, c, d, x[12], S41, 0x655b59c3L); /* 53 */
d = II(d, a, b, c, x[3], S42, 0x8f0ccc92L); /* 54 */
c = II(c, d, a, b, x[10], S43, 0xffeff47dL); /* 55 */
b = II(b, c, d, a, x[1], S44, 0x85845dd1L); /* 56 */
a = II(a, b, c, d, x[8], S41, 0x6fa87e4fL); /* 57 */
d = II(d, a, b, c, x[15], S42, 0xfe2ce6e0L); /* 58 */
c = II(c, d, a, b, x[6], S43, 0xa3014314L); /* 59 */
b = II(b, c, d, a, x[13], S44, 0x4e0811a1L); /* 60 */
a = II(a, b, c, d, x[4], S41, 0xf7537e82L); /* 61 */
d = II(d, a, b, c, x[11], S42, 0xbd3af235L); /* 62 */
c = II(c, d, a, b, x[2], S43, 0x2ad7d2bbL); /* 63 */
b = II(b, c, d, a, x[9], S44, 0xeb86d391L); /* 64 */
state[0] += a;
state[1] += b;
state[2] += c;
state[3] += d;
}
/*
* Encode把long数组按顺序拆成byte数组,因为java的long类型 64bit的, 只拆 32bit,以适应原始C实现的用
*/
private void Encode(byte[] output, long[] input, int len) {
int i, j;
for (i = 0, j = 0; j < len; i++, j += 4) {
output[j] = (byte) (input[i] & 0xffL);
output[j + 1] = (byte) ((input[i] >>> 8) & 0xffL);
output[j + 2] = (byte) ((input[i] >>> 16) & 0xffL);
output[j + 3] = (byte) ((input[i] >>> 24) & 0xffL);
}
}
/*
* Decode把byte数组按顺序合成成long数组,因为java的long类型 64bit的,
* 只合成低32bit,高32bit清零,以适应原始C实现的用
*/
private void Decode(long[] output, byte[] input, int len) {
int i, j;
for (i = 0, j = 0; j < len; i++, j += 4)
output[i] = b2iu(input[j]) | (b2iu(input[j + 1]) << 8) | (b2iu(input[j + 2]) << 16)
| (b2iu(input[j + 3]) << 24);
return;
}
/*
* 把byte按照不虑虑正负号的原则的"升位"程式,因为java没有unsigned运算
*/
public static long b2iu(byte b) {
return b < 0 ? b & 0x7F + 128 : b;
}
//用来把一个byte类型的数转换成十六进制的ASCII表示
public static String byteHEX(byte ib) {
char[] Digit = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
char[] ob = new char[2];
ob[0] = Digit[(ib >>> 4) & 0X0F];
ob[1] = Digit[ib & 0X0F];
String s = new String(ob);
return s;
}
public static void main(String argv[]) {
System.out.println(MD5.getMD5("123".getBytes()));
}
}
StreamUtils
public class StreamUtil {
//把inputStrem转换为String
public static byte[] getBytesFromStream(InputStream is){
byte[] datas=null;
try {
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len = 0;
while ((len = is.read(buffer)) != -1) {
outputStream.write(buffer,0,len);
}
datas=outputStream.toByteArray();
outputStream.close();
} catch (Exception e) {
e.printStackTrace();
}finally{
if(is!=null){
try {
is.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return datas;
}
}
自定义控件:可缩放的imageview
public class ZoomImageView extends ImageView implements OnScaleGestureListener,
OnTouchListener, ViewTreeObserver.OnGlobalLayoutListener
{
private static final String TAG = ZoomImageView.class.getSimpleName();
public static final float SCALE_MAX = 4.0f;
private static final float SCALE_MID = 2.0f;
/**
* 初始化时的缩放比例,如果图片宽或高大于屏幕,此值将小于0
*/
private float initScale = 1.0f;
private boolean once = true;
/**
* 用于存放矩阵的9个值
*/
private final float[] matrixValues = new float[9];
/**
* 缩放的手势检测
*/
private ScaleGestureDetector mScaleGestureDetector = null;
private final Matrix mScaleMatrix = new Matrix();
/**
* 用于双击检测
*/
private GestureDetector mGestureDetector;
private boolean isAutoScale;
private int mTouchSlop;
private float mLastX;
private float mLastY;
private boolean isCanDrag;
private int lastPointerCount;
private boolean isCheckTopAndBottom = true;
private boolean isCheckLeftAndRight = true;
public ZoomImageView(Context context)
{
this(context, null);
}
public ZoomImageView(Context context, AttributeSet attrs)
{
super(context, attrs);
super.setScaleType(ScaleType.MATRIX);
//手势事件的监听:双击
mGestureDetector = new GestureDetector(context,
new SimpleOnGestureListener()
{
@Override
public boolean onDoubleTap(MotionEvent e)
{
//如果是自动缩放,则终止该方法
if (isAutoScale == true)
return true;
float x = e.getX();
float y = e.getY();
//获取缩放比例,对此进行判断,并设置到自动缩放的任务
if (getScale() < SCALE_MID)
{
ZoomImageView.this.postDelayed(
new AutoScaleRunnable(SCALE_MID, x, y), 16);
isAutoScale = true;
} else if (getScale() >= SCALE_MID
&& getScale() < SCALE_MAX)
{
ZoomImageView.this.postDelayed(
new AutoScaleRunnable(SCALE_MAX, x, y), 16);
isAutoScale = true;
} else
{
ZoomImageView.this.postDelayed(
new AutoScaleRunnable(initScale, x, y), 16);
isAutoScale = true;
}
return true;
}
});
//缩放手势的监听
mScaleGestureDetector = new ScaleGestureDetector(context, this);
this.setOnTouchListener(this);
}
/**
* 自动缩放的任务
*
* @author zhy
*
*/
private class AutoScaleRunnable implements Runnable
{
static final float BIGGER = 1.07f;
static final float SMALLER = 0.93f;
private float mTargetScale;
private float tmpScale;
/**
* 缩放的中心
*/
private float x;
private float y;
/**
* 传入目标缩放值,
* 根据目标值与当前值,判断应该放大还是缩小
* @param targetScale
*/
public AutoScaleRunnable(float targetScale, float x, float y)
{
this.mTargetScale = targetScale;
this.x = x;
this.y = y;
if (getScale() < mTargetScale)
{
tmpScale = BIGGER;
} else
{
tmpScale = SMALLER;
}
}
@Override
public void run()
{
// 进行缩放
mScaleMatrix.postScale(tmpScale, tmpScale, x, y);
checkBorderAndCenterWhenScale();
setImageMatrix(mScaleMatrix);
final float currentScale = getScale();
// 如果值在合法范围内,继续缩放
if (((tmpScale > 1f) && (currentScale < mTargetScale))
|| ((tmpScale < 1f) && (mTargetScale < currentScale)))
{
ZoomImageView.this.postDelayed(this, 16);
} else
// 设置为目标的缩放比例
{
final float deltaScale = mTargetScale / currentScale;
mScaleMatrix.postScale(deltaScale, deltaScale, x, y);
checkBorderAndCenterWhenScale();
setImageMatrix(mScaleMatrix);
isAutoScale = false;
}
}
}
//缩放手势的操作
@SuppressLint("NewApi")
@Override
public boolean onScale(ScaleGestureDetector detector)
{
float scale = getScale();
float scaleFactor = detector.getScaleFactor();
if (getDrawable() == null)
return true;
/**
* 缩放的范围控制
*/
if ((scale < SCALE_MAX && scaleFactor > 1.0f)
|| (scale > initScale && scaleFactor < 1.0f))
{
/**
* 最大值最小值判断
*/
if (scaleFactor * scale < initScale)
{
scaleFactor = initScale / scale;
}
if (scaleFactor * scale > SCALE_MAX)
{
scaleFactor = SCALE_MAX / scale;
}
/**
* 设置缩放比例
*/
mScaleMatrix.postScale(scaleFactor, scaleFactor,
detector.getFocusX(), detector.getFocusY());
checkBorderAndCenterWhenScale();
setImageMatrix(mScaleMatrix);
}
return true;
}
/**
* 在缩放时,进行图片显示范围的控制
*/
private void checkBorderAndCenterWhenScale()
{
RectF rect = getMatrixRectF();
float deltaX = 0;
float deltaY = 0;
int width = getWidth();
int height = getHeight();
// 如果宽或高大于屏幕,则控制范围
if (rect.width() >= width)
{
if (rect.left > 0)
{
deltaX = -rect.left;
}
if (rect.right < width)
{
deltaX = width - rect.right;
}
}
if (rect.height() >= height)
{
if (rect.top > 0)
{
deltaY = -rect.top;
}
if (rect.bottom < height)
{
deltaY = height - rect.bottom;
}
}
// 如果宽或高小于屏幕,则让其居中
if (rect.width() < width)
{
deltaX = width * 0.5f - rect.right + 0.5f * rect.width();
}
if (rect.height() < height)
{
deltaY = height * 0.5f - rect.bottom + 0.5f * rect.height();
}
mScaleMatrix.postTranslate(deltaX, deltaY);
}
/**
* 根据当前图片的Matrix获得图片的范围
*
* @return
*/
private RectF getMatrixRectF()
{
Matrix matrix = mScaleMatrix;
RectF rect = new RectF();
Drawable d = getDrawable();
if (null != d)
{
rect.set(0, 0, d.getIntrinsicWidth(), d.getIntrinsicHeight());
matrix.mapRect(rect);
}
return rect;
}
@Override
public boolean onScaleBegin(ScaleGestureDetector detector)
{
return true;
}
@Override
public void onScaleEnd(ScaleGestureDetector detector)
{
}
//触摸事件的监听
@Override
public boolean onTouch(View v, MotionEvent event)
{
if (mGestureDetector.onTouchEvent(event))
return true;
mScaleGestureDetector.onTouchEvent(event);
float x = 0, y = 0;
// 拿到触摸点的个数
final int pointerCount = event.getPointerCount();
// 得到多个触摸点的x与y均值
for (int i = 0; i < pointerCount; i++)
{
x += event.getX(i);
y += event.getY(i);
}
x = x / pointerCount;
y = y / pointerCount;
/**
* 每当触摸点发生变化时,重置mLasX , mLastY
*/
if (pointerCount != lastPointerCount)
{
isCanDrag = false;
mLastX = x;
mLastY = y;
}
//保存最后一次的触摸点个数
lastPointerCount = pointerCount;
RectF rectF = getMatrixRectF();
//解决触摸时间的冲突
switch (event.getAction())
{
//对比图片的matrix和屏幕的宽高 >>确定事件是否拦截
case MotionEvent.ACTION_DOWN:
//当图片的宽度大于屏幕的空间的宽度的时候
if (rectF.width() > getWidth() || rectF.height() > getHeight())
{
getParent().requestDisallowInterceptTouchEvent(true);
}
break;
case MotionEvent.ACTION_MOVE:
if (rectF.width() > getWidth() || rectF.height() > getHeight())
{
getParent().requestDisallowInterceptTouchEvent(true);
}
float dx = x - mLastX;
float dy = y - mLastY;
if (!isCanDrag)
{
isCanDrag = isCanDrag(dx, dy);
}
if (isCanDrag)
{
if (getDrawable() != null)
{
isCheckLeftAndRight = isCheckTopAndBottom = true;
// 如果宽度小于屏幕宽度,则禁止左右移动
if (rectF.width() < getWidth())
{
dx = 0;
isCheckLeftAndRight = false;
}
// 如果高度小雨屏幕高度,则禁止上下移动
if (rectF.height() < getHeight())
{
dy = 0;
isCheckTopAndBottom = false;
}
mScaleMatrix.postTranslate(dx, dy);
checkMatrixBounds();
setImageMatrix(mScaleMatrix);
}
}
mLastX = x;
mLastY = y;
break;
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
lastPointerCount = 0;
break;
}
return true;
}
/**
* 获得当前的缩放比例
*
* @return
*/
public final float getScale()
{
mScaleMatrix.getValues(matrixValues);
return matrixValues[Matrix.MSCALE_X];
}
@Override
protected void onAttachedToWindow()
{
super.onAttachedToWindow();
getViewTreeObserver().addOnGlobalLayoutListener(this);
}
@SuppressWarnings("deprecation")
@Override
protected void onDetachedFromWindow()
{
super.onDetachedFromWindow();
getViewTreeObserver().removeGlobalOnLayoutListener(this);
}
@Override
public void onGlobalLayout()
{
if (once)
{
Drawable d = getDrawable();
if (d == null)
return;
Log.e(TAG, d.getIntrinsicWidth() + " , " + d.getIntrinsicHeight());
int width = getWidth();
int height = getHeight();
// 拿到图片的宽和高
int dw = d.getIntrinsicWidth();
int dh = d.getIntrinsicHeight();
float scale = 1.0f;
// 如果图片的宽或者高大于屏幕,则缩放至屏幕的宽或者高
if (dw > width && dh <= height)
{
scale = width * 1.0f / dw;
}
if (dh > height && dw <= width)
{
scale = height * 1.0f / dh;
}
// 如果宽和高都大于屏幕,则让其按按比例适应屏幕大小
if (dw > width && dh > height)
{
scale = Math.min(width * 1.0f / dw, height * 1.0f / dh);
}
initScale = scale;
Log.e(TAG, "initScale = " + initScale);
mScaleMatrix.postTranslate((width - dw) / 2, (height - dh) / 2);
mScaleMatrix.postScale(scale, scale, getWidth() / 2,
getHeight() / 2);
// 图片移动至屏幕中心
setImageMatrix(mScaleMatrix);
once = false;
}
}
/**
* 移动时,进行边界判断,主要判断宽或高大于屏幕的
*/
private void checkMatrixBounds()
{
RectF rect = getMatrixRectF();
float deltaX = 0, deltaY = 0;
final float viewWidth = getWidth();
final float viewHeight = getHeight();
// 判断移动或缩放后,图片显示是否超出屏幕边界
if (rect.top > 0 && isCheckTopAndBottom)
{
deltaY = -rect.top;
}
if (rect.bottom < viewHeight && isCheckTopAndBottom)
{
deltaY = viewHeight - rect.bottom;
}
if (rect.left > 0 && isCheckLeftAndRight)
{
deltaX = -rect.left;
}
if (rect.right < viewWidth && isCheckLeftAndRight)
{
deltaX = viewWidth - rect.right;
}
mScaleMatrix.postTranslate(deltaX, deltaY);
}
/**
* 是否是推动行为
*
* @param dx
* @param dy
* @return
*/
private boolean isCanDrag(float dx, float dy)
{
return Math.sqrt((dx * dx) + (dy * dy)) >= mTouchSlop;
}
}
事件由子控件响应
getParent().requestDisallowInterceptTouchEvent(true);
删除module
在setting中打开project struct
点击要删除的module
点击左上方的-号
ok
以project打开项目
右键需要删除的module
Android Studio中添加so库
在android开发板中使用M13模块的nfc读卡器的时候,添加so文件失败,怎么想也不明白,
后来在网上找了一篇有用的蚊帐;
Module的根目录中建立libs目录,将so文件复制进去
在module的gradle
android {
.......
task nativeLibsToJar(type: Zip, description: "create a jar archive of the native libs") {
destinationDir file("$projectDir/libs")
baseName "Native_Libs2"
extension "jar"
from fileTree(dir: "libs", include: "**/*.so")
into "lib"
}
tasks.withType(JavaCompile) {
compileTask -> compileTask.dependsOn(nativeLibsToJar)
}
}
平时用的添加so库方法:
1.创建src/main/jniLibs文件夹
复制so文件到该文件中
2.在module的gradle中
android {
.......
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
3.build/makeproject
切换android视图,就可以看先so文件已经变成了jar文件了
Debug
在需要检查的代码开头和结尾打上断点
右键 run as debug
将十六进制的数据解析成字符串(包括中文)
反复写了一万遍,转换结果都是乱码
网上查到原来可能是android studio中UTF-8失效的问题
//好用的工具类
public static String hex2String(String s){
byte[] baKeyword = new byte[s.length()/2];
for(int i = 0; i < baKeyword.length; i++){
try{
baKeyword[i] = (byte)(0xff & Integer.parseInt(s.substring(i*2, i*2+2),16));
}catch(Exception e){
e.printStackTrace();
}
}
try{
s = new String(baKeyword, "GB2312");
}catch (Exception e1){
e1.printStackTrace();
}
return s;
}
集合的比较器的使用
Collection.sort(list,comparator);
Comparator<Double> comparator=new Comparator<>{
public int compare(Double dl,double d2){
return p1>=p2 ?1:-1;
}
}
Style的使用
<style name="xx">
<item name="textsize">12sp<>
<item name=textColor>...
</style>
2.在布局中引用样式
<Tv
Style="@style/xx">
注意和android:textStyle的区别,用错了会崩溃
将app设置为大内存
配置文件中:
<application android:largelHeap="true"
sdk版本适配
if(android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.LOLLIPOP)
延时的message
//发送延时的message给handler mHandler.sendEmptyMessageDelayed(MESSAGE_SCROLL_TO,DELAYED_TIME);
对gravity的理解
gravity失效,因为gravity作用域内容,layoutgravity才是作用于控件在父控件中的位置的
设置app的方向
<activity android:name=".MainActivity" android:screenOrientation="portrait">
屏蔽虚拟按键
decorView = getWindow().getDecorView();
int uiOptions =
//隐藏导航栏
View.SYSTEM_UI_FLAG_HIDE_NAVIGATION
//设置全屏:隐藏状态栏
| View.SYSTEM_UI_FLAG_FULLSCREEN
| View.SYSTEM_UI_FLAG_LAYOUT_STABLE
//设置layout是否全屏(与状态栏位置重叠)
| View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
//真正意义上的全屏
| View.SYSTEM_UI_FLAG_IMMERSIVE;
decorView.setSystemUiVisibility(uiOptions);
decorView.setVisibility(View.GONE);
隐藏下拉菜单
Caused by: java.lang.SecurityException: StatusBarManagerService: Neither user 10056 nor current process has android.permission.STATUS_BAR.
因为需要两个权限,而其中statubar的权限需要系统签名,获取系统签名的方法自行百度
<uses-permission android:name="android.permission.STATUS_BAR" />
<uses-permission android:name="android.permission.EXPAND_STATUS_BAR" />
public static final int DISABLE_EXPAND = 0x00010000;//4.2以上的整形标识
public static final int DISABLE_EXPAND_LOW = 0x00000001;//4.2以下的整形标识
public static final int DISABLE_NONE = 0x00000000;//取消StatusBar所有disable属性
//在onWindowFocusChanged()中屏蔽下拉效果
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
banStatusBar();
}
//取消下拉的屏蔽
private void unBanStatusBar() {
Object service = getSystemService("statusbar");
try {
Class<?> statusBarManager = Class.forName
("android.app.StatusBarManager");
Method expand = statusBarManager.getMethod("disable", int.class);
expand.invoke(service, DISABLE_NONE);
} catch (Exception e) {
}
}
//屏蔽下拉
private void banStatusBar() {
int currentApiVersion = android.os.Build.VERSION.SDK_INT;
if (currentApiVersion <= 16) {
setStatusBarDisable(DISABLE_EXPAND_LOW);
} else {
setStatusBarDisable(DISABLE_EXPAND);
}
}
private void setStatusBarDisable(int disable_status) {
Object service = getSystemService("statusbar");
try {
Class<?> statusBarManager = Class.forName
("android.app.StatusBarManager");
Method expand = statusBarManager.getMethod("disable", int.class);
expand.invoke(service, disable_status);
} catch (Exception e) {
unBanStatusBar();
e.printStackTrace();
}
}
判断当前线程是不是主线程
public boolean isMainThread() {
return Looper.getMainLooper() == Looper.myLooper();
}
public boolean isMainThread() {
return Looper.getMainLooper().getThread().getId() == Thread.currentThread().getId();
}
Okhttp的execute()enqueue()
execute是执行的意思,execute()是立即执行,同步执行
enqueue是加入队列的意思,enqueue()是将任务加入线程池的队列中,异步执行