架构组成

在JetBrains Intellij IDEA的基础上,Android Studio 提供:

  • 基于Gradle的构建支持
  • Android 专属的重构和快速修复
  • 提示工具以捕获性能、可用性、版本兼容性等问题
  • 支持ProGuard 和应用签名
  • 基于模板的向导来生成常用的 Android 应用设计和组件
  • 功能强大的布局编辑器,可以让你拖拉 UI 控件并进行效果预览

功能介绍

1、优化小贴士:在主体中打开你的应用,点击小贴士,会得到这样的建议:为你的应用开发平板电脑版本。

2、应用翻译服务:允许开发者直接在开发主体中获得专业的翻译。上传你的需求,选择翻译,其会显示翻译方和价格,并在一周内发回译本。

3、推荐跟踪:允许开发者找出最有效的广告

4、营收曲线图:向开发者展示其应用营收,以国家进行划分

5、试用版测试和阶段性展示:开发者可以对应用进行测试,然后向测试用户推出,测试结果不会对外公布。当一个版本的测试结束,开发者可以向特定比例用户推出 。

Android Studio这款开发工具被首次公布,这也是为了方便开发者基于Android开发。

一个小APP开发实战“笔记本”

一、需求分析

1.业务需求分析:

分析潜在客户需要什么产品或服务

2.架构分析:

分析所做的产品或服务需要什么功能及实现⽅式并画出功能结构图,记事本案例如下

3.数据库类设计分析:

数据库设计是项⽬开发中⾮常关键的⼀个环节。同样在记事本案例中也⾄关重要,我们通过数据库表(Note)进⾏增删改查操作,记事本的数据表如下所⽰

4.界⾯需求分析:

友好的界⾯在移动平台开发中⾮常重要,也是⽤户使⽤软件的先决条件。记事本案例分为3个界⾯,分别为记事本界⾯、添加界⾯和修改界⾯
记事本界⾯包含添加按钮和记录列表,点击后会跳转到添加界⾯,界⾯标题为添加记录,也可在该界⾯清除和保存编辑的内容。

点击记事本界⾯中的Item时,会跳转到修改记录界⾯,界⾯标题为修改记录,在该界⾯中可以查看和修改已保存的记录内容,也可清除和保存编辑的
内容

二、本案例步骤

1.创建项⽬后将Activity名称设置为NotepadActivity,

布局⽂件为activity_notepad并将所需素材add.png(添加按钮),save_note.png(保存按钮),delete.png(删除按钮),back.png(返回按钮)导⼊drawable⽂件夹中,如下

上⾯添加按钮右边还有个返回按钮图⽚是⽩⾊的

2.编写activity_notepad.xml布局⽂件,

放置⼀个TextView控件⽤于显⽰界⾯标题,⼀个ListView控件⽤于显⽰记录列表,⼀个ImageView控件⽤于显⽰添加按钮的图⽚

3.修改清单⽂件

项⽬创建后所有界⾯都有⼀个默认的标题栏,该标题栏不太美观,因此需要在清单⽂件(AndroidManifest.xml)中的标签中修改android:theme属性

android:theme="@style/Theme.AppCompat.NoActionBar"

4.搭建记事本界⾯Item布局

在res/layout⽂件夹中,创建⼀个布局⽂件notepad_item_layout.xml,在其中放置两个TextView控件,分别⽤来显⽰记录的部分内容与保存记录的时间

?xml version="1.0" encoding="utf-8"?>

5.封装记录信息实体类

由于记事本中的每个记录都会有记录内容和保存记录的时间属性,因此需要创建⼀个NotepadBean⽤于存放这些属性。选中你所创建的项⽬的包,右击选择New>Package,创建⼀个bean包,在该包中创建⼀NotepadBean类,该类中定义记录信息的所有属性

package com.example.notepad.bean;public class NotepadBean {private String id;private String notepadContent;private String notepadTime;public String getId(){returnid;}public void setId(String id){this.id=id;}public String getNotepadContent(){return notepadContent;}public void setNotepadContent(String notepadContent){this.notepadContent=notepadContent;}public String getNotepadTime(){return notepadTime;}public void setNotepadTime(String notepadTime){this.notepadTime=notepadTime;}}

注意包名

6.编写记事本界⾯列表适配器

由于记事本界⾯的记录列表是使⽤ListView控件来展⽰的,因此需要创建⼀个数据适配器NotepadAdapter对ListView控件进⾏数据适配。步骤如下
选中你所创建的项⽬的包,右击选择New>Package,创建⼀个adapter包,在adapter包中创建⼀个NotepadAdapter类继承⾃BaseAdapter类,并重写getCount(),getItem(),getItemId(),getView()⽅法,这些⽅法⽤于获取Item总数、对应Item对象、Item对象的Id、对应的Item视图,在NotepadAdapter类中创建⼀个ViewHolder类,在该类中初始化Item界⾯中的控件,具体代码如下

package com.example.notepad.adapter;import android.content.Context;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.TextView;import com.example.notepad.R;import com.example.notepad.bean.NotepadBean;import java.util.List;public class NotepadAdapter extends BaseAdapter {private LayoutInflater layoutInflater;private List list;public NotepadAdapter(Context context, Listlist){this.layoutInflater=LayoutInflater.from(context);this.list=list;}@Overridepublic int getCount(){return list==null?0:list.size();}@Overridepublic Object getItem(int position){return list.get(position);}@Overridepublic long getItemId(int position){return position;}@Overridepublic View getView(int position,View convertView, ViewGroup parent){ViewHolder viewHolder;if(convertView==null){convertView=layoutInflater.inflate(R.layout.notepad_item_layout,null);viewHolder=new ViewHolder(convertView);convertView.setTag(viewHolder);}else{viewHolder=(ViewHolder)convertView.getTag();}NotepadBean noteInfo=(NotepadBean)getItem(position);viewHolder.tvNoteoadContent.setText(noteInfo.getNotepadContent());viewHolder.tvNotepadTime.setText(noteInfo.getNotepadTime());return convertView;}class ViewHolder{TextView tvNoteoadContent;TextView tvNotepadTime;public ViewHolder(View view){tvNoteoadContent=(TextView)view.findViewById(R.id.item_content);tvNotepadTime=(TextView)view.findViewById(R.id.item_time);}}}

注意包名

创建数据库

在记事本案例中增删改查记录的数据都是通过操作数据库完成的。因此需要创建数据库类SQLiteHelper与数据库的⼯具类DBUtils,步骤如下
选中你所创建的项⽬的包,右击选择New>Package,创建⼀个utils包,在utils包中创建⼀个DBUtils类,在该类中定义数据库的名称、表名、数据库版本、数据库表中的列名以及获取当前⽇期等信息

 package com.example.notepad.utils;importjava.text.SimpleDateFormat;import java.util.Date;public class DBUtils {public static final String DATABASE_NAME="Notepad";public static final String DATAVASE_TABLE="Note";public static final int DATEBASE_VERION=1;public static final String NOTEPAD_ID="id";public static final String NOTEPAD_CONTENT="content";public static final String NOTEPAD_TIME="notetime";public static final String getTime(){SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy年MM⽉dd⽇HH:mm:ss");Date date=new Date(System.currentTimeMillis());return simpleDateFormat.format(date);}}

注意包名
选中你所创建的项⽬的包,右击选择New>Package,创建⼀个database包,在database包中创建⼀个SQLiteHelper类继承⾃SQLiteOpenHelper类

package com.example.notepad.database;import android.content.ContentValues;import android.content.Context;import android.database.Cursor;import android.database.sqlite.SQLiteDatabase;import android.database.sqlite.SQLiteOpenHelper;import com.example.notepad.bean.NotepadBean;import com.example.notepad.utils.DBUtils;import java.util.ArrayList;import java.util.List;public class SQLiteHelper extends SQLiteOpenHelper {private SQLiteDatabase sqLiteDatabase;public SQLiteHelper(Context context){super(context,DBUtils.DATABASE_NAME,null,DBUtils.DATEBASE_VERION);sqLiteDatabase=this.getWritableDatabase();}@Overridepublic void onCreate(SQLiteDatabase db){db.execSQL("create table "+DBUtils.DATAVASE_TABLE+"( "+DBUtils.NOTEPAD_ID+" integer primary key autoincrement ,"+DBUtils.NOTEPAD_CONTENT+" text , "+DBUtils.NOTEPAD_TIME+" text ) ");}@Overridepublicvoid onUpgrade(SQLiteDatabase db,int oldVersion,int newVersion){}public boolean insertData(String userContent,String userTime){ContentValues contentValues=new ContentValues();contentValues.put(DBUtils.NOTEPAD_CONTENT,userContent);contentValues.put(DBUtils.NOTEPAD_TIME,userTime);return sqLiteDatabase.insert(DBUtils.DATAVASE_TABLE,null,contentValues)>0;}public boolean deteleData(String id){String sql=DBUtils.NOTEPAD_ID+"=?";String[] contentValuesArray=new String[]{String.valueOf(id)};return sqLiteDatabase.delete(DBUtils.DATAVASE_TABLE,sql,contentValuesArray)>0;}public boolean updateData(String id,String content,String userYear){ContentValues contentValues=new ContentValues();contentValues.put(DBUtils.NOTEPAD_CONTENT,content);contentValues.put(DBUtils.NOTEPAD_TIME,userYear);String sql=DBUtils.NOTEPAD_ID+"=?";String[] strings=new String[]{id};return sqLiteDatabase.update(DBUtils.DATAVASE_TABLE,contentValues,sql,strings)>0;}public Listquery(){List list=new ArrayList();Cursor cursor=sqLiteDatabase.query(DBUtils.DATAVASE_TABLE,null,null,null,null,null,DBUtils.NOTEPAD_ID+" desc");if(cursor!=null){while(cursor.moveToNext()){NotepadBean noteInfo=new NotepadBean();String id=String.valueOf(cursor.getInt(cursor.getColumnIndex(DBUtils.NOTEPAD_ID)));String content=cursor.getString(cursor.getColumnIndex(DBUtils.NOTEPAD_CONTENT));String time=cursor.getString(cursor.getColumnIndex(DBUtils.NOTEPAD_TIME));noteInfo.setId(id);noteInfo.setNotepadContent(content);noteInfo.setNotepadTime(time);list.add(noteInfo);}cursor.close();}return list;}}

注意包名

7.实现记事本界⾯的显⽰功能

在NotepadActivity中通过创建⼀个showQueryData()⽅法查询数据库中存放的记录信息,并将该信息显⽰到记录列表中,同时在NotepadActivity中还实现了添加按钮的点击事件

package com.example.notepad;import android.content.DialogInterface;import android.content.Intent;import android.support.v7.app.AlertDialog;import android.support.v7.app.AppCompatActivity;import android.os.Bundle;import android.view.View;import android.widget.AdapterView;import android.widget.ImageView;import android.widget.ListView;import android.widget.Toast;import com.example.notepad.adapter.NotepadAdapter;import com.example.notepad.bean.NotepadBean;import com.example.notepad.database.SQLiteHelper;import java.util.List;public class NotepadActivity extends AppCompatActivity {ListView listView;List list;SQLiteHelper mSQLiteHelper;NotepadAdapter adapter;@Overrideprotected void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_notepad);listView=(ListView)findViewById(R.id.listview);ImageView add=(ImageView)findViewById(R.id.add);add.setOnClickListener(new View.OnClickListener(){@Overridepublic void onClick(View v){Intent intent=new Intent(NotepadActivity.this,RecordActivity.class);startActivityForResult(intent,1);}});initData();}protected void initData(){mSQLiteHelper=new SQLiteHelper(this);showQueryData();listView.setOnItemClickListener(new AdapterView.OnItemClickListener(){@Overridepublic void onItemClick(AdapterView parent, View view, int position, long id){NotepadBean notepadBean=list.get(position);Intent intent=new Intent(NotepadActivity.this,RecordActivity.class);intent.putExtra("id",notepadBean.getId());intent.putExtra("time",notepadBean.getNotepadTime());intent.putExtra("content",notepadBean.getNotepadContent());NotepadActivity.this.startActivityForResult(intent,1);}});listView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener(){@Overridepublic boolean onItemLongClick(AdapterView parent, View view, final int position, long id){AlertDialog dialog;AlertDialog.Builder builder=new AlertDialog.Builder(NotepadActivity.this).setMessage("是否删除此记录").setPositiveButton("确定", new DialogInterAlertDialog.Builder builder=new AlertDialog.Builder(NotepadActivity.this).setMessage("是否删除此记录").setPositiveButton("确定", new DialogInterface.OnClickListener(){@Overridepublic void onClick(DialogInterface dialog, int which){NotepadBean notepadBean=list.get(position);if(mSQLiteHelper.deteleData(notepadBean.getId())){list.remove(position);adapter.notifyDataSetChanged();Toast.makeText(NotepadActivity.this,"删除成功",Toast.LENGTH_SHORT).show();}}}).setNegativeButton("取消", new DialogInterface.OnClickListener(){@Overridepublic void onClick(DialogInterface dialog, int which){dialog.dismiss();}});dialog=builder.create();dialog.show();returntrue;}});}private void showQueryData(){if(list!=null){list.clear();}list=mSQLiteHelper.query();adapter=new NotepadAdapter(this,list);listView.setAdapter(adapter);}@Overrideprotectedvoid onActivityResult(int requestCode,int resultCode,Intent data){super.onActivityResult(requestCode,resultCode,data);if(requestCode==1&&resultCode==2){showQueryData();;}}}

注意包名

8.搭建添加记录和修改记录界⾯的布局

当点击记事本界⾯的添加按钮时,会跳转到添加记录界⾯,当点击记事本界⾯列表中的Item时,会跳转到修改记录界⾯。由于这两个界⾯上的控件与功能基本相同,因此可以使⽤同⼀个Activity和同⼀个布局⽂件显⽰这两个界⾯,步骤如下
选中你所创建的项⽬的包,创建⼀个名为RecordActivity的Activity并将布局⽂件指名为activity_record,在activity_record.xml布局⽂件中,放置两个TextView控件,分别⽤于显⽰界⾯标题和记录时间,⼀个EditText控件⽤于显⽰输⼊框,三个ImageView控件分别⽤于显⽰后退键图标、删除图标以及保存按钮图标

1.实现添加记录界⾯的功能

由于添加记录界⾯的清除和保存按钮需要实现点击事件,因此将RecordActivity实现View.OnClickListener接⼝并重写onClick()⽅法,在该⽅法中实现将编写的记录添加到数据库中的功能

package com.example.notepad;import android.content.Intent;import android.support.v7.app.AppCompatActivity;import android.os.Bundle;import android.view.View;import android.widget.EditText;import android.widget.ImageView;import android.widget.ImageView;import android.widget.TextView;import android.widget.Toast;import com.example.notepad.database.SQLiteHelper;import com.example.notepad.utils.DBUtils;public class RecordActivity extends AppCompatActivity implements View.OnClickListener {ImageView note_back;TextView note_time;EditText content;ImageView deldete;ImageView note_save;SQLiteHelper mSQLiteHelper;TextView noteName;String id;@Overrideprotected void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_record);note_back =(ImageView) findViewById(R.id.note_back);note_time =(TextView) findViewById(R.id.tv_time);content =(EditText) findViewById(R.id.note_content);deldete =(ImageView) findViewById(R.id.delete);note_save =(ImageView) findViewById(R.id.note_save);noteName =(TextView) findViewById(R.id.note_name);note_back.setOnClickListener(this);deldete.setOnClickListener(this);note_save.setOnClickListener(this);initData();}protected void initData(){mSQLiteHelper = new SQLiteHelper(this);noteName.setText("添加记录");Intent intent = getIntent();if(intent != null){id= intent.getStringExtra("id");if(id != null){noteName.setText("修改记录");content.setText(intent.getStringExtra("content"));note_time.setText(intent.getStringExtra("time"));note_time.setVisibility(View.VISIBLE);}}}@Overridepublic void onClick(View v){switch (v.getId()){case R.id.note_back:finish();break;case R.id.delete:content.setText("");break;case R.id.note_save:String noteContent = content.getText().toString().trim();if(id != null){if(noteContent.length()> 0){if(mSQLiteHelper.updateData(id, noteContent, DBUtils.getTime())){showToast("修改成功");setResult(2);finish();}else{}else{showToast("修改失败");}}else{showToast("修改内容不能为空");}}else{if(noteContent.length()>0){if(mSQLiteHelper.insertData(noteContent,DBUtils.getTime())){showToast("保存成功");setResult(2);finish();}else{showToast("保存失败");}}else {showToast("修改内容不能为空");}}break;}}public void showToast(String message){Toast.makeText(RecordActivity.this,message,Toast.LENGTH_SHORT).show();}}

注意包名

2.实现修改记录界⾯的功能

修改记录界⾯主要包含查看记录和修改记录的功能实现查看记录功能记事本界⾯列表的每个Item最多只显⽰2⾏记录信息,如果想要查看更多的记录内容,则需要点击Item,进⼊到修改记录界⾯进⾏查看,因此点击Item时,需要将Item对应的记录信息传递到修改记录界⾯进⾏显⽰,即为NotepadActivity的initData()⽅法,具体代码已在步骤(7)中写出.

在RecordActivity的initData()⽅法中需要接收记事本界⾯传递过来的记录数据并将数据显⽰到界⾯上,具体代码已在当前步骤中写出实现修改记录功能在RecordActivity中的onClick()⽅法中,找到保存按钮的点击事件,在该事件中根据判断传递过来的id是否为null来判断处理的是添加记录界⾯的保存功能还是修改记录界⾯的保存功能,如果id不为null,则处理修改记录界⾯的保存功能,具体代码已在当前步骤中写出.

9.删除记事本中的记录

当需要删除记事本列表中的记录时,需要长按列表中的Item,此时会弹出⼀个对话框提⽰是否删除Item的对应的记录,因此在NotepadActivity中的initData()⽅法中写出了删除记录的逻辑代码,具体代码已在步骤(7)中写出

10记事本案例已完成

总结

Android进阶技术学习,可以前往这里;使用Android studio制作一个简单的APP记事本就完成了。这对于新手来说可以上手实操。