Android手势锁实现

最终效果如下

整体思路

a、自定义了一个RelativeLayout(GestureLockViewGroup)在里面会根据传入的每行的个数,生成多个GestureLockView(就是上面一个个小圈圈),然后会自动进行布局,里面的宽度,间距,内圆的直径,箭头的大小神马的都是百分比实现的,所以大胆的设置你喜欢的个数,只要你没有密集恐惧症~

b、GestureLockView有三个状态,没有手指触碰、手指触碰、和手指抬起,会根据这三个状态绘制不同的效果,以及抬起时的小箭头需要旋转的角度,会根据用户选择的GestureLockView,进行计算,在GestureLockViewGroup为每个GestureLockView设置

c、GestureLockViewGroup主要就是判断用户ACTION_MOVE,ACTION_DOWN , ACTION_UP时改变选中的GestureLockView的状态,并且记录下来,提供一定的回调。

自定义属性

<?xml version="1.0" encoding="utf-8"?>
<resources><attr name="color_no_finger_inner_circle" format="color" /><attr name="color_no_finger_outer_circle" format="color" /><attr name="color_finger_on" format="color" /><attr name="color_finger_up" format="color" /><attr name="count" format="integer" /><attr name="tryTimes" format="integer" /><declare-styleable name="GestureLockViewGroup"><attr name="color_no_finger_inner_circle" /><attr name="color_no_finger_outer_circle" /><attr name="color_finger_on" /><attr name="color_finger_up" /><attr name="count" /><attr name="tryTimes" /></declare-styleable></resources>

GestureLockView

package com.zhy.zhy_gesturelockview.view;import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Paint.Style;
import android.graphics.Path;
import android.view.View;public class GestureLockView extends View
{private static final String TAG = "GestureLockView";/*** GestureLockView的三种状态*/enum Mode{STATUS_NO_FINGER, STATUS_FINGER_ON, STATUS_FINGER_UP;}/*** GestureLockView的当前状态*/private Mode mCurrentStatus = Mode.STATUS_NO_FINGER;/*** 宽度*/private int mWidth;/*** 高度*/private int mHeight;/*** 外圆半径*/private int mRadius;/*** 画笔的宽度*/private int mStrokeWidth = 2;/*** 圆心坐标*/private int mCenterX;private int mCenterY;private Paint mPaint;/*** 箭头(小三角最长边的一半长度 = mArrawRate * mWidth / 2 )*/private float mArrowRate = 0.333f;private int mArrowDegree = -1;private Path mArrowPath;/*** 内圆的半径 = mInnerCircleRadiusRate * mRadus* */private float mInnerCircleRadiusRate = 0.3F;/*** 四个颜色,可由用户自定义,初始化时由GestureLockViewGroup传入*/private int mColorNoFingerInner;private int mColorNoFingerOutter;private int mColorFingerOn;private int mColorFingerUp;public GestureLockView(Context context , int colorNoFingerInner , int colorNoFingerOutter , int colorFingerOn , int colorFingerUp ){super(context);this.mColorNoFingerInner = colorNoFingerInner;this.mColorNoFingerOutter = colorNoFingerOutter;this.mColorFingerOn = colorFingerOn;this.mColorFingerUp = colorFingerUp;mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);mArrowPath = new Path();}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec){super.onMeasure(widthMeasureSpec, heightMeasureSpec);mWidth = MeasureSpec.getSize(widthMeasureSpec);mHeight = MeasureSpec.getSize(heightMeasureSpec);// 取长和宽中的小值mWidth = mWidth < mHeight ? mWidth : mHeight;mRadius = mCenterX = mCenterY = mWidth / 2;mRadius -= mStrokeWidth / 2;// 绘制三角形,初始时是个默认箭头朝上的一个等腰三角形,用户绘制结束后,根据由两个GestureLockView决定需要旋转多少度float mArrowLength = mWidth / 2 * mArrowRate;mArrowPath.moveTo(mWidth / 2, mStrokeWidth + 2);mArrowPath.lineTo(mWidth / 2 - mArrowLength, mStrokeWidth + 2+ mArrowLength);mArrowPath.lineTo(mWidth / 2 + mArrowLength, mStrokeWidth + 2+ mArrowLength);mArrowPath.close();mArrowPath.setFillType(Path.FillType.WINDING);}@Overrideprotected void onDraw(Canvas canvas){switch (mCurrentStatus){case STATUS_FINGER_ON:// 绘制外圆mPaint.setStyle(Style.STROKE);mPaint.setColor(mColorFingerOn);mPaint.setStrokeWidth(2);canvas.drawCircle(mCenterX, mCenterY, mRadius, mPaint);// 绘制内圆mPaint.setStyle(Style.FILL);canvas.drawCircle(mCenterX, mCenterY, mRadius* mInnerCircleRadiusRate, mPaint);break;case STATUS_FINGER_UP:// 绘制外圆mPaint.setColor(mColorFingerUp);mPaint.setStyle(Style.STROKE);mPaint.setStrokeWidth(2);canvas.drawCircle(mCenterX, mCenterY, mRadius, mPaint);// 绘制内圆mPaint.setStyle(Style.FILL);canvas.drawCircle(mCenterX, mCenterY, mRadius* mInnerCircleRadiusRate, mPaint);drawArrow(canvas);break;case STATUS_NO_FINGER:// 绘制外圆mPaint.setStyle(Style.FILL);mPaint.setColor(mColorNoFingerOutter);canvas.drawCircle(mCenterX, mCenterY, mRadius, mPaint);// 绘制内圆mPaint.setColor(mColorNoFingerInner);canvas.drawCircle(mCenterX, mCenterY, mRadius* mInnerCircleRadiusRate, mPaint);break;}}/*** 绘制箭头* @param canvas*/private void drawArrow(Canvas canvas){if (mArrowDegree != -1){mPaint.setStyle(Paint.Style.FILL);canvas.save();canvas.rotate(mArrowDegree, mCenterX, mCenterY);canvas.drawPath(mArrowPath, mPaint);canvas.restore();}}/*** 设置当前模式并重绘界面* * @param mode*/public void setMode(Mode mode){this.mCurrentStatus = mode;invalidate();}public void setArrowDegree(int degree){this.mArrowDegree = degree;}public int getArrowDegree(){return this.mArrowDegree;}
}

GestureLockViewGroup

package com.zhy.zhy_gesturelockview.view;import java.util.ArrayList;
import java.util.List;import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.Point;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.RelativeLayout;import com.zhy.zhy_gesturelockview.R;
import com.zhy.zhy_gesturelockview.view.GestureLockView.Mode;/*** 整体包含n*n个GestureLockView,每个GestureLockView间间隔mMarginBetweenLockView,* 最外层的GestureLockView与容器存在mMarginBetweenLockView的外边距* * 关于GestureLockView的边长(n*n): n * mGestureLockViewWidth + ( n + 1 ) ** mMarginBetweenLockView = mWidth ; 得:mGestureLockViewWidth = 4 * mWidth / ( 5* * mCount + 1 ) 注:mMarginBetweenLockView = mGestureLockViewWidth * 0.25 ;* * @author zhy* */
public class GestureLockViewGroup extends RelativeLayout
{private static final String TAG = "GestureLockViewGroup";/*** 保存所有的GestureLockView*/private GestureLockView[] mGestureLockViews;/*** 每个边上的GestureLockView的个数*/private int mCount = 3;/*** 存储答案*/private int[] mAnswer = { 0, 1, 2, 5, 8 };/*** 保存用户选中的GestureLockView的id*/private List<Integer> mChoose = new ArrayList<Integer>();private Paint mPaint;/*** 每个GestureLockView中间的间距 设置为:mGestureLockViewWidth * 25%*/private int mMarginBetweenLockView = 30;/*** GestureLockView的边长 4 * mWidth / ( 5 * mCount + 1 )*/private int mGestureLockViewWidth;/*** GestureLockView无手指触摸的状态下内圆的颜色*/private int mNoFingerInnerCircleColor = 0xFF939090;/*** GestureLockView无手指触摸的状态下外圆的颜色*/private int mNoFingerOuterCircleColor = 0xFFE0DBDB;/*** GestureLockView手指触摸的状态下内圆和外圆的颜色*/private int mFingerOnColor = 0xFF378FC9;/*** GestureLockView手指抬起的状态下内圆和外圆的颜色*/private int mFingerUpColor = 0xFFFF0000;/*** 宽度*/private int mWidth;/*** 高度*/private int mHeight;private Path mPath;/*** 指引线的开始位置x*/private int mLastPathX;/*** 指引线的开始位置y*/private int mLastPathY;/*** 指引下的结束位置*/private Point mTmpTarget = new Point();/*** 最大尝试次数*/private int mTryTimes = 4;/*** 回调接口*/private OnGestureLockViewListener mOnGestureLockViewListener;public GestureLockViewGroup(Context context, AttributeSet attrs){this(context, attrs, 0);}public GestureLockViewGroup(Context context, AttributeSet attrs,int defStyle){super(context, attrs, defStyle);/*** 获得所有自定义的参数的值*/TypedArray a = context.getTheme().obtainStyledAttributes(attrs,R.styleable.GestureLockViewGroup, defStyle, 0);int n = a.getIndexCount();for (int i = 0; i < n; i++){int attr = a.getIndex(i);switch (attr){case R.styleable.GestureLockViewGroup_color_no_finger_inner_circle:mNoFingerInnerCircleColor = a.getColor(attr,mNoFingerInnerCircleColor);break;case R.styleable.GestureLockViewGroup_color_no_finger_outer_circle:mNoFingerOuterCircleColor = a.getColor(attr,mNoFingerOuterCircleColor);break;case R.styleable.GestureLockViewGroup_color_finger_on:mFingerOnColor = a.getColor(attr, mFingerOnColor);break;case R.styleable.GestureLockViewGroup_color_finger_up:mFingerUpColor = a.getColor(attr, mFingerUpColor);break;case R.styleable.GestureLockViewGroup_count:mCount = a.getInt(attr, 3);break;case R.styleable.GestureLockViewGroup_tryTimes:mTryTimes = a.getInt(attr, 5);default:break;}}a.recycle();// 初始化画笔mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);mPaint.setStyle(Paint.Style.STROKE);// mPaint.setStrokeWidth(20);mPaint.setStrokeCap(Paint.Cap.ROUND);mPaint.setStrokeJoin(Paint.Join.ROUND);// mPaint.setColor(Color.parseColor("#aaffffff"));mPath = new Path();}@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec){super.onMeasure(widthMeasureSpec, heightMeasureSpec);mWidth = MeasureSpec.getSize(widthMeasureSpec);mHeight = MeasureSpec.getSize(heightMeasureSpec);// Log.e(TAG, mWidth + "");// Log.e(TAG, mHeight + "");mHeight = mWidth = mWidth < mHeight ? mWidth : mHeight;// setMeasuredDimension(mWidth, mHeight);// 初始化mGestureLockViewsif (mGestureLockViews == null){mGestureLockViews = new GestureLockView[mCount * mCount];// 计算每个GestureLockView的宽度mGestureLockViewWidth = (int) (4 * mWidth * 1.0f / (5 * mCount + 1));//计算每个GestureLockView的间距mMarginBetweenLockView = (int) (mGestureLockViewWidth * 0.25);// 设置画笔的宽度为GestureLockView的内圆直径稍微小点(不喜欢的话,随便设)mPaint.setStrokeWidth(mGestureLockViewWidth * 0.29f);for (int i = 0; i < mGestureLockViews.length; i++){//初始化每个GestureLockViewmGestureLockViews[i] = new GestureLockView(getContext(),mNoFingerInnerCircleColor, mNoFingerOuterCircleColor,mFingerOnColor, mFingerUpColor);mGestureLockViews[i].setId(i + 1);//设置参数,主要是定位GestureLockView间的位置RelativeLayout.LayoutParams lockerParams = new RelativeLayout.LayoutParams(mGestureLockViewWidth, mGestureLockViewWidth);// 不是每行的第一个,则设置位置为前一个的右边if (i % mCount != 0){lockerParams.addRule(RelativeLayout.RIGHT_OF,mGestureLockViews[i - 1].getId());}// 从第二行开始,设置为上一行同一位置View的下面if (i > mCount - 1){lockerParams.addRule(RelativeLayout.BELOW,mGestureLockViews[i - mCount].getId());}//设置右下左上的边距int rightMargin = mMarginBetweenLockView;int bottomMargin = mMarginBetweenLockView;int leftMagin = 0;int topMargin = 0;/*** 每个View都有右外边距和底外边距 第一行的有上外边距 第一列的有左外边距*/if (i >= 0 && i < mCount)// 第一行{topMargin = mMarginBetweenLockView;}if (i % mCount == 0)// 第一列{leftMagin = mMarginBetweenLockView;}lockerParams.setMargins(leftMagin, topMargin, rightMargin,bottomMargin);mGestureLockViews[i].setMode(Mode.STATUS_NO_FINGER);addView(mGestureLockViews[i], lockerParams);}Log.e(TAG, "mWidth = " + mWidth + " ,  mGestureViewWidth = "+ mGestureLockViewWidth + " , mMarginBetweenLockView = "+ mMarginBetweenLockView);}}@Overridepublic boolean onTouchEvent(MotionEvent event){int action = event.getAction();int x = (int) event.getX();int y = (int) event.getY();switch (action){case MotionEvent.ACTION_DOWN:// 重置reset();break;case MotionEvent.ACTION_MOVE:mPaint.setColor(mFingerOnColor);mPaint.setAlpha(50);GestureLockView child = getChildIdByPos(x, y);if (child != null){int cId = child.getId();if (!mChoose.contains(cId)){mChoose.add(cId);child.setMode(Mode.STATUS_FINGER_ON);if (mOnGestureLockViewListener != null)mOnGestureLockViewListener.onBlockSelected(cId);// 设置指引线的起点mLastPathX = child.getLeft() / 2 + child.getRight() / 2;mLastPathY = child.getTop() / 2 + child.getBottom() / 2;if (mChoose.size() == 1)// 当前添加为第一个{mPath.moveTo(mLastPathX, mLastPathY);} else// 非第一个,将两者使用线连上{mPath.lineTo(mLastPathX, mLastPathY);}}}// 指引线的终点mTmpTarget.x = x;mTmpTarget.y = y;break;case MotionEvent.ACTION_UP:mPaint.setColor(mFingerUpColor);mPaint.setAlpha(50);this.mTryTimes--;// 回调是否成功if (mOnGestureLockViewListener != null && mChoose.size() > 0){mOnGestureLockViewListener.onGestureEvent(checkAnswer());if (this.mTryTimes == 0){mOnGestureLockViewListener.onUnmatchedExceedBoundary();}}Log.e(TAG, "mUnMatchExceedBoundary = " + mTryTimes);Log.e(TAG, "mChoose = " + mChoose);// 将终点设置位置为起点,即取消指引线mTmpTarget.x = mLastPathX;mTmpTarget.y = mLastPathY;// 改变子元素的状态为UPchangeItemMode();// 计算每个元素中箭头需要旋转的角度for (int i = 0; i + 1 < mChoose.size(); i++){int childId = mChoose.get(i);int nextChildId = mChoose.get(i + 1);GestureLockView startChild = (GestureLockView) findViewById(childId);GestureLockView nextChild = (GestureLockView) findViewById(nextChildId);int dx = nextChild.getLeft() - startChild.getLeft();int dy = nextChild.getTop() - startChild.getTop();// 计算角度int angle = (int) Math.toDegrees(Math.atan2(dy, dx)) + 90;startChild.setArrowDegree(angle);}//reset();break;}invalidate();return true;}private void changeItemMode(){for (GestureLockView gestureLockView : mGestureLockViews){if (mChoose.contains(gestureLockView.getId())){gestureLockView.setMode(Mode.STATUS_FINGER_UP);}}}/*** * 做一些必要的重置*/private void reset(){mChoose.clear();mPath.reset();for (GestureLockView gestureLockView : mGestureLockViews){gestureLockView.setMode(Mode.STATUS_NO_FINGER);gestureLockView.setArrowDegree(-1);}}/*** 检查用户绘制的手势是否正确* @return*/private boolean checkAnswer(){if (mAnswer.length != mChoose.size())return false;for (int i = 0; i < mAnswer.length; i++){if (mAnswer[i] != mChoose.get(i))return false;}return true;}/*** 检查当前左边是否在child中* @param child* @param x* @param y* @return*/private boolean checkPositionInChild(View child, int x, int y){//设置了内边距,即x,y必须落入下GestureLockView的内部中间的小区域中,可以通过调整padding使得x,y落入范围不变大,或者不设置paddingint padding = (int) (mGestureLockViewWidth * 0.15);if (x >= child.getLeft() + padding && x <= child.getRight() - padding&& y >= child.getTop() + padding&& y <= child.getBottom() - padding){return true;}return false;}/*** 通过x,y获得落入的GestureLockView* @param x* @param y* @return*/private GestureLockView getChildIdByPos(int x, int y){for (GestureLockView gestureLockView : mGestureLockViews){if (checkPositionInChild(gestureLockView, x, y)){return gestureLockView;}}return null;}/*** 设置回调接口* * @param listener*/public void setOnGestureLockViewListener(OnGestureLockViewListener listener){this.mOnGestureLockViewListener = listener;}/*** 对外公布设置答案的方法* * @param answer*/public void setAnswer(int[] answer){this.mAnswer = answer;}/*** 设置最大实验次数* * @param boundary*/public void setUnMatchExceedBoundary(int boundary){this.mTryTimes = boundary;}@Overridepublic void dispatchDraw(Canvas canvas){super.dispatchDraw(canvas);//绘制GestureLockView间的连线if (mPath != null){canvas.drawPath(mPath, mPaint);}//绘制指引线if (mChoose.size() > 0){if (mLastPathX != 0 && mLastPathY != 0)canvas.drawLine(mLastPathX, mLastPathY, mTmpTarget.x,mTmpTarget.y, mPaint);}}public interface OnGestureLockViewListener{/*** 单独选中元素的Id* * @param position*/public void onBlockSelected(int cId);/*** 是否匹配* * @param matched*/public void onGestureEvent(boolean matched);/*** 超过尝试次数*/public void onUnmatchedExceedBoundary();}
}

布局文件

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"xmlns:zhy="http://schemas.android.com/apk/res/com.zhy.zhy_gesturelockview"android:layout_width="match_parent"android:layout_height="match_parent" ><com.zhy.zhy_gesturelockview.view.GestureLockViewGroup
        android:id="@+id/id_gestureLockViewGroup"android:layout_width="match_parent"android:layout_height="match_parent"android:background="#F2F2F7"android:gravity="center_vertical"zhy:count="3"zhy:tryTimes="5" />
<!--   zhy:color_no_finger_inner_circle="#ff085D58"zhy:color_no_finger_outer_circle="#ff08F0E0"zhy:color_finger_on="#FF1734BF" --></RelativeLayout>

调用

package com.zhy.zhy_gesturelockview;import android.app.Activity;
import android.os.Bundle;
import android.widget.Toast;import com.zhy.zhy_gesturelockview.view.GestureLockViewGroup;
import com.zhy.zhy_gesturelockview.view.GestureLockViewGroup.OnGestureLockViewListener;public class MainActivity extends Activity
{private GestureLockViewGroup mGestureLockViewGroup;@Overrideprotected void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mGestureLockViewGroup = (GestureLockViewGroup) findViewById(R.id.id_gestureLockViewGroup);mGestureLockViewGroup.setAnswer(new int[] { 1, 2, 3 });mGestureLockViewGroup.setOnGestureLockViewListener(new OnGestureLockViewListener(){@Overridepublic void onUnmatchedExceedBoundary(){Toast.makeText(MainActivity.this, "错误5次...",Toast.LENGTH_SHORT).show();mGestureLockViewGroup.setUnMatchExceedBoundary(5);}@Overridepublic void onGestureEvent(boolean matched){Toast.makeText(MainActivity.this, matched+"",Toast.LENGTH_SHORT).show();}@Overridepublic void onBlockSelected(int cId){}});}}

ondraw() 和dispatchdraw()的区别

本文中用到了dispatchdraw,
绘制VIew本身的内容,通过调用View.onDraw(canvas)函数实现

绘制自己的孩子通过dispatchDraw(canvas)实现

参考链接

ondraw() 和dispatchdraw()的区别 - scorplopan的专栏 - 博客频道 - CSDN.NET

在计算箭头应该转化的角度时用到了atan2

atan2 返回的是方位角,也可以理解为计算复数 x+yi 的辐角

参考链接

atan2_百度百科

本文主要参考链接

Android 手势锁的实现 让自己的应用更加安全吧 - Hongyang - 博客频道 - CSDN.NET

源代码下载

源代码下载

完成

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/494134.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

智能连接:5G与人工智能、物联网等技术的超级融合

来源&#xff1a;资本实验室随着新技术的成熟&#xff0c;新型的、先进的应用将来自5G、人工智能&#xff08;AI&#xff09;和物联网&#xff08;IoT&#xff09;的融合。这种融合将创造出一个智能连接的世界&#xff0c;对所有个人、行业、社会和经济产生积极影响。从现在到2…

一个绚丽的loading动效分析与实现!

最终效果如下 从效果上看&#xff0c;我们需要考虑以下几个问题&#xff1a; 1.叶子的随机产生&#xff1b; 2.叶子随着一条正余弦曲线移动&#xff1b; 3.叶子在移动的时候旋转&#xff0c;旋转方向随机&#xff0c;正时针或逆时针&#xff1b; 4.叶子遇到进度条&#xff…

20岁的谷歌,和它“最成功”的大败笔

来源&#xff1a;大数据文摘编译&#xff1a;张驰、JIN、涂世文、钱天培谷歌20岁了&#xff01;20年中&#xff0c;谷歌打造了无数或成功或流产的产品&#xff0c;其中&#xff0c;这一名为“谷歌光纤”计划的失败或许是它最“成功”的“大败笔”。2010年&#xff0c;谷歌宣布了…

自定义viewgroup实现ArcMenu

最终效果如下 实现思路 通过效果图&#xff0c;会有几个问题&#xff1a; a、动画效果如何实现 可以看出动画是从顶点外外发射的&#xff0c;可能有人说&#xff0c;那还不简单&#xff0c;默认元素都在定点位置&#xff0c;然后TraslateAnimation就好了&#xff1b;这样忽略…

也谈谈Atiyah关于黎曼猜想的证明

来源&#xff1a;潇轩社作者&#xff1a;叶扬波 著名数学家&#xff0c;美国爱荷华大学教授。作为数论学家&#xff0c;他在中国大陆出版有《迹公式与模形式》等专著。以下是他谈Atiyah关于黎曼猜想的证明的文章&#xff0c;观点专业而且独到&#xff0c;转载此文&#xff0c;…

大型Javascript应用架构的模式(译文)

附上翻译好的word文件 http://files.cnblogs.com/lizhug/Patterns_For_Large-Scale_JavaScript_Application_Architecture.zip 作者&#xff1a;Addy Osmani 技术评审&#xff1a;Andree Hansson 翻译&#xff1a;李珠刚 珠刚参上 今天我们将要探讨一系列用于大型Javascri…

Animation Property Animation 使用

本篇主要讲Animation 和 Property Animation的使用&#xff0c;最后会讲QQ管家桌面火箭作为例子&#xff1a; 在Android中开发动效有两套框架可以使用&#xff0c;分别为 Animation 和 Property Animation&#xff1b; 相对来说&#xff0c;Animator比Animation要强大太多&…

华为云力推“普惠AI”,EI智能体正在落地行业

来源&#xff1a;北京物联网智能技术应用协会云计算和人工智能是什么关系&#xff1f;一般认为&#xff0c;云计算是人工智能的基础之一。而华为云则认为&#xff0c;要推动人工智能的落地&#xff0c;拥有“算法、算力和数据”还是不足够的&#xff0c;还需要“行业智慧”&…

Android中mesure过程详解

我们在编写layout的xml文件时会碰到layout_width和layout_height两个属性&#xff0c;对于这两个属性我们有三种选择&#xff1a;赋值成具体的数值&#xff0c;match_parent或者wrap_content&#xff0c;而measure过程就是用来处理match_parent或者wrap_content&#xff0c;假如…

黑科技揭秘 | 阿里云“天空物联网”连接范围如何达到700平方公里

来源&#xff1a;阿里云还记得前不久在2018杭州云栖大会上&#xff0c;由飞在天上的飞艇、地下基站共同搭建的阿里云“天空物联网”吗&#xff1f;它实现从地面40000米高空到地下20米的覆盖&#xff0c;并持续为大会服务。体现了持续加码的阿里巴巴物联网战略。在飞艇背后&…

JAVA-用栈机制实现单词逆序排列

就是IO那一段还没学到。 之前的PUCH,POP,STRING和CHAR的关系搞得懂了。 学到一个定位STRING当中CHAR的转换函数。 char String.charAt(x) 1 import java.io.*;2 3 4 class stackString5 {6 private int maxSize;7 private char[] stackX;8 private int top;9 …

Android自定义实现FlowLayout

实现FlowLayout 何为FlowLayout&#xff0c;如果对Java的Swing比较熟悉的话一定不会陌生&#xff0c;就是控件根据ViewGroup的宽&#xff0c;自动的往右添加&#xff0c;如果当前行剩余空间不足&#xff0c;则自动添加到下一行。有点所有的控件都往左飘的感觉&#xff0c;第一…

【重磅】马斯克遇终极麻烦:被起诉欺诈罪 或丢掉CEO职位 特斯拉暴跌约13%

参考&#xff1a;CNBC、Bloomberg编译&#xff1a;网易智能编辑&#xff1a;丁广胜参与&#xff1a;小小美国当地时间周四&#xff0c;法庭的文件显示特斯拉电动汽车公司首席执行官伊隆马斯克(Elon Musk)已被美国证券交易委员会(SEC)以欺诈罪起诉。与特斯拉关系密切的消息人士透…

《麻省理工科技评论》:2018年18大科技趋势,2017年7大失败技术

来源&#xff1a;科技周摘要&#xff1a;2018 年伊始&#xff0c;许多科技大势仍在继续&#xff0c;正如比尔盖茨所说&#xff0c;“大多数人高估了某种技术的短期价值&#xff0c;低估了其长期价值。”同样&#xff0c;大多数的年度预测会高估了一年内一些事件发生的可能性&am…

Android实现支持缩放平移图片

本文主要用到了以下知识点 MatrixGestureDetector 能够捕捉到长按、双击ScaleGestureDetector 用于检测缩放的手势 自由的缩放 需求&#xff1a;当图片加载时&#xff0c;将图片在屏幕中居中&#xff1b;图片宽或高大于屏幕的&#xff0c;缩小至屏幕大小&#xff1b;自由对图…

放麦子

题意&#xff1a; 国际象棋&#xff0c;一共64个方格&#xff0c;第一个格子里放一粒麦子&#xff0c;第二个放2粒&#xff0c;第三个放4粒&#xff0c;第四个放8粒。。。。。&#xff08;后面的数字是前面的两倍&#xff09; 求放满64个格子&#xff0c;一共需要多少粒麦子。 …

Material Design风格登录注册

本文实现了以下功能 完整的代码和样例托管在Github当接口锁定时&#xff0c;防止后退按钮显示在登录Activity 上。自定义 ProgressDialog来显示加载的状态。符合材料设计规范。悬浮标签&#xff08;floating labels&#xff09;&#xff08;来自设计支持库&#xff09;用户表单…

英特尔反驳质疑:芯片供应充足、10nm量产没问题

来源&#xff1a;华尔街见闻摘要&#xff1a;英特尔称2018年会将资本支出增加10亿美元&#xff0c;至总额创纪录的150亿美元&#xff1b;CEO称&#xff0c;个人电脑需求意外回升&#xff0c;但有足够供应满足市场&#xff0c;有望达成全年营收目标&#xff0c;股价涨近4%。竞争…

RecyclerView拖拽排序和滑动删除实现

效果图 如何实现 那么是如何实现的呢&#xff1f;主要就要使用到ItemTouchHelper &#xff0c;ItemTouchHelper 一个帮助开发人员处理拖拽和滑动删除的实现类&#xff0c;它能够让你非常容易实现侧滑删除、拖拽的功能。 实现的代码非常简单我们只需要两步&#xff1a; 实例化…

马斯克刚刚宣布辞去特斯拉董事会职务,仍然担任CEO

来源&#xff1a;大数据文摘编译&#xff1a;蒋宝尚据悉&#xff0c;当地时间周六&#xff0c;马斯克辞去特斯拉董事会主席一职&#xff0c;并且支付2000万美元罚款。以表示对美国证券交易委员会(SEC)指控的回应。SEC的指控来源于马斯克8月7日的一篇推文。推文中&#xff0c;他…