Android之调用微信登陆、分享、支付

转载:http://blog.csdn.net/lowprofile_coding/article/details/48086381

前言:用了微信sdk各种痛苦,感觉比qq sdk调用麻烦多了,回调过于麻烦,还必须要在指定包名下的actvity进行回调,所以我在这里写一篇博客,有这个需求的朋友可以借鉴一下,以后自己别的项目有用到也有个找资料的地方.

一.微信登陆分三个步骤:

  1).微信授权登陆
  2).根据授权登陆code 获取该用户token
  3).根据token获取用户资料
  4).接收微信的请求及返回值 如果你的程序需要接收微信发送的请求,或者接收发送到微信请求的响应结果,需要下面3步操作:

  a. 在你的包名相应目录下新建一个wxapi目录,并在该wxapi目录下新增一个WXEntryActivity类,该类继承自Activity(例如应用程序的包名为net.sourceforge.simcpux,

        则新添加的类如下图所示)


并在manifest文件里面加上exported属性,设置为true,例如:

 b. 实现IWXAPIEventHandler接口,微信发送的请求将回调到onReq方法,发送到微信请求的响应结果将回调到onResp方法
 c. 在WXEntryActivity中将接收到的intent及实现了IWXAPIEventHandler接口的对象传递给IWXAPI接口的handleIntent方法,示例如下图:
             

微信官网登陆教程:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419317851&token=&lang=zh_CN

微信官网接入指南:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=1417751808&token=&lang=zh_CN


二.微信分享直接调用sdk就行,回调跟登陆回调的类是一样的,根据BaseResp的类型来区分是登陆还是分享。

三.微信支付

    1).发送一个支付请求

    2).接收微信支付的返回值(跟接收微信登陆.分享的返回值类似,我就不写详细操作步骤了) 

        官网参考地址:https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=8_5

四.贴上代码进行讲解

    我把微信登陆,分享,支付都封装到了一个类里面了,你们可以参考这个类.封装了6个方法,我对几个需要的方法介绍一下

    1).构造方法:初始化对象的时候,顺便初始化微信对象,把app_id注册到微信

    2).login()  发起一个登陆的请求  在微信登陆监听Actviity中获取code

    3).getAccessToken(String code)  当你从监听Activity中获取了code之后就可以通过这个方法获取微信访问token了

    4).getWeiXinUserInfo(final WeiXinToken obj)  获取微信用户信息   传入一个WeiXinToken对象,这个对象是第三步的返回值

    5).share(final boolean friendsCircle,final VideoB videoB)   分享给朋友或者朋友圈  如果你有分享图片,图片过大的时候一定要经过压缩,微信官网说明图片不能大

         于32kb

    6).isWXAppInstalled()  检查微信是否安装

    7).wxPay(final BaseActivity activity,String order_id,String payType)  微信支付   我们项目微信支付的一些参数保存在服务器上,所以我这边还请求了自己的

        服务器,如果你们是放在本地,直接copy回调函数里面的代码即可.在微信支付监听Actviity中获取支付的状态码

PayReq类属性对应含义请参考微信官方文档:https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=9_12

/*** 微信分享,登陆,支付* @author ansen * @create time 2015-08-29*/
public class WeiXinPresenter extends Presenter{public static final int IMAGE_SIZE=32768;//微信分享图片大小限制public static final String APP_ID = "";//应用唯一标识,在微信开放平台提交应用审核通过后获得public static final String SECRET="";//应用密钥AppSecret,在微信开放平台提交应用审核通过后获得private  IWXAPI wxAPI;private  IView iView;private  IUserController userController;@Overridepublic IView getIView() {return iView;}public WeiXinPresenter(Context context){if(context!=null && context instanceof  IView)iView =(IView) context;if(wxAPI==null){wxAPI = WXAPIFactory.createWXAPI(context,APP_ID,true);wxAPI.registerApp(APP_ID);}if(null==userController)userController=ControllerFactory.getUserController();}/*** 微信登陆(三个步骤)* 1.微信授权登陆* 2.根据授权登陆code 获取该用户token* 3.根据token获取用户资料* @param activity*/public void login(){SendAuth.Req req = new SendAuth.Req();req.scope = "snsapi_userinfo";req.state = String.valueOf(System.currentTimeMillis());wxAPI.sendReq(req);}/*** 获取微信访问token*/public void getAccessToken(String code){if(!userController.isLogin()){//没有登陆的情况用第三方登陆userController.getWeiXinAccessToken(APP_ID,SECRET,code,new RequestDataCallback<WeiXinToken>(){@Overridepublic void dataCallback(WeiXinToken obj){if(obj!=null){if(obj.getErrcode()==0){if(MLog.debug)iView.showToast("授权用户唯一标识:"+obj.getOpenid());getWeiXinUserInfo(obj);}else{iView.showToast(obj.getErrmsg());}}else{}}});}else{//用户已登陆}}/*** 获取微信用户信息*/private void getWeiXinUserInfo(final WeiXinToken obj){userController.getWeiXinUserInfo(obj.getAccess_token(), obj.getOpenid(), new RequestDataCallback<RegisterB>() {@Overridepublic void dataCallback(RegisterB registerB){registerB.setAccess_token(obj.getAccess_token());registerB.setToken_expire_at(obj.getExpires_in());if(registerB.getErrcode()==0){registerB.setThird_type_name(Constants.WEI_XIN);thirdLogin(registerB);}else{iView.showToast(registerB.getErrmsg());}}});}/*** 调用我们自己的服务器进行登录* @param registerB*/private void thirdLogin(RegisterB registerB){userController.thirdAuth(registerB,new RequestDataCallback<UserP>(){@Overridepublic void dataCallback(UserP user){if(checkCallbackData(user, true)){if(user.getError()==user.ErrorNone){iView.showToast(R.string.login_success);getAppController().sendLoginChangeIntent();userController.saveLoginUser(user,FileUtil.getFilePath());((ILoginView)iView).toMain();}else{iView.showToast(user.getError_reason());}}}});}/*** 微信分享* @param friendsCircle  是否分享到朋友圈*/public void share(final boolean friendsCircle,final  VideoB videoB){new LoadPicThread(videoB.getCover_url(),new Handler(){@Overridepublic void handleMessage(Message msg) {byte[] bytes=(byte[]) msg.obj;if(bytes.length>IMAGE_SIZE){iView.showToast(R.string.image_no_big);return;}System.out.println("图片长度:"+bytes.length);WXVideoObject videoObject = new WXVideoObject();//视频类型videoObject.videoUrl = videoB.getShare_url() + Constants.WEI_XIN + "&share_from="+com.kaka.utils.Constants.ANDROID;// 视频播放urlWXMediaMessage wxMessage = new WXMediaMessage(videoObject);wxMessage.title = videoB.getContent();wxMessage.thumbData = bytes;SendMessageToWX.Req req = new SendMessageToWX.Req();//transaction字段用于唯一标识一个请求req.transaction = String.valueOf(videoB.getId() + System.currentTimeMillis());req.message = wxMessage;req.scene = friendsCircle ? SendMessageToWX.Req.WXSceneTimeline : SendMessageToWX.Req.WXSceneSession;wxAPI.sendReq(req);}}).start();}private class LoadPicThread extends Thread{private String url;private Handler handler;public LoadPicThread(String url,Handler handler){this.url=url;this.handler=handler;}@Overridepublic void run(){try {URL picurl = new URL(url); HttpURLConnection conn = (HttpURLConnection)picurl.openConnection(); // 获得连接 conn.setConnectTimeout(6000);//设置超时 conn.setDoInput(true); conn.setUseCaches(false);//不缓存 conn.connect();Bitmap bmp=BitmapFactory.decodeStream(conn.getInputStream());ByteArrayOutputStream output = new ByteArrayOutputStream();bmp.compress(Bitmap.CompressFormat.JPEG, 100, output);int options = 100;while (output.toByteArray().length > IMAGE_SIZE && options != 10) { output.reset();  // 清空baosbmp.compress(Bitmap.CompressFormat.JPEG, options, output);// 这里压缩options%,把压缩后的数据存放到baos中  options -= 10;}bmp.recycle();byte[] result = output.toByteArray();output.close();Message message=handler.obtainMessage();message.obj=result;message.sendToTarget();} catch (Exception e) {e.printStackTrace();}}}//检查微信是否安装public boolean isWXAppInstalled(){return wxAPI.isWXAppInstalled();}public void wxPay(final BaseActivity activity,String order_id,String payType){activity.showProgress("");ControllerFactory.getWalletsController().getPayments(order_id, payType, new RequestDataCallback<PaymentsP>() {@Overridepublic void dataCallback(PaymentsP obj) {if(checkCallbackData(obj, true)){if(obj.getError()==obj.ErrorNone){PayReq req = new PayReq();//待修改req.appId = obj.getAppid();req.nonceStr=obj.getNoncestr();req.packageValue=obj.getPackage_value();req.sign=obj.getSign();req.partnerId=obj.getPartnerid();req.prepayId=obj.getPrepayid();req.timeStamp=obj.getTimestamp();wxAPI.registerApp(obj.getAppid());wxAPI.sendReq(req);MLog.i("ansen", "开始进行微信支付..");iView.showToast("开始进行微信支付..");}}else{iView.showToast(obj.getError_reason());}activity.hideProgress();}});}
}
微信登陆以及分享 请求跟返回值的接收   我这边登陆.分享的状态都是发送广播出去,然后结束当前的Activity.

/*** 微信登陆分享回调Activity* @author ansen* @create time 2015-05-25*/
public class WXEntryActivity extends Activity implements IWXAPIEventHandler{private IWXAPI wxAPI;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);if(MLog.debug)System.out.println("WXEntryActivity onCreate");wxAPI = WXAPIFactory.createWXAPI(this,WeiXinPresenter.APP_ID,true);wxAPI.registerApp(WeiXinPresenter.APP_ID);wxAPI.handleIntent(getIntent(), this);}@Overrideprotected void onNewIntent(Intent intent){super.onNewIntent(intent);wxAPI.handleIntent(getIntent(),this);if(MLog.debug)System.out.println("WXEntryActivity onNewIntent");}@Overridepublic void onReq(BaseReq arg0) {if(MLog.debug)System.out.println("WXEntryActivity onReq:"+arg0);if(MLog.debug)Toast.makeText(this, "onReq 方法运行", 0).show();}@Overridepublic void onResp(BaseResp resp){MLog.d("ansen", "onResp.....");if(MLog.debug)Toast.makeText(this,"onResp 方法运行", 0).show();if(resp.getType()==ConstantsAPI.COMMAND_SENDMESSAGE_TO_WX){//分享switch (resp.errCode){case BaseResp.ErrCode.ERR_OK:if(MLog.debug)Toast.makeText(WXEntryActivity.this, "分享成功!", Toast.LENGTH_SHORT).show();break;case BaseResp.ErrCode.ERR_USER_CANCEL:
//                Toast.makeText(WXEntryActivity.this, "分享取消!", Toast.LENGTH_SHORT).show();break;case BaseResp.ErrCode.ERR_AUTH_DENIED:break;}Intent intent = new Intent();intent.setAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_SHARE);LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this);lbm.sendBroadcast(intent);}else if(resp.getType()==ConstantsAPI.COMMAND_SENDAUTH){//登陆发送广播SendAuth.Resp authResp = (Resp) resp;String code = authResp.code;Intent intent = new Intent();intent.setAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN);intent.putExtra("errCode", authResp.errCode);if (authResp.errCode == BaseResp.ErrCode.ERR_OK){//用户同意intent.putExtra("code", code);}if(MLog.debug)Toast.makeText(this, "WXEntryActivity 发送登陆广播!!!!", 0).show();if (android.os.Build.VERSION.SDK_INT >= 12) {intent.setFlags(32);//3.1以后的版本需要设置Intent.FLAG_INCLUDE_STOPPED_PACKAGES}LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this);lbm.sendBroadcast(intent);}finish();}
}
微信支付 请求跟返回值的接收   微信支付也是发送广播,如果你们还有需求判断支付成功或者失败,可以在广播的intent中进行传参

/*** 微信支付回调Activity* @author ansen* @create time 2015-08-29*/
public class WXPayEntryActivity extends Activity  implements IWXAPIEventHandler{private IWXAPI wxAPI;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);wxAPI = WXAPIFactory.createWXAPI(this, WeiXinPresenter.APP_ID);wxAPI.handleIntent(getIntent(), this);}@Overrideprotected void onNewIntent(Intent intent){super.onNewIntent(intent);setIntent(intent);wxAPI.handleIntent(intent, this);}@Overridepublic void onReq(BaseReq arg0) {}@Overridepublic void onResp(BaseResp resp) {MLog.i("微信支付回调..", "ansen onResp");if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX){//微信支付回调if(resp.errCode==BaseResp.ErrCode.ERR_OK){//微信支付成功Intent intent = new Intent();intent.setAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_PAY);LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this);lbm.sendBroadcast(intent);//成功
//				Toast.makeText(this,R.string.wxpay_success, 0).show();}else{
//				Toast.makeText(this,R.string.wxpay_success, 0).show();}}finish();}}
强调一点,一定要注意 接收微信的请求及返回值 的包名跟类名,包名是应用程序的包名+".wxapi"  类名必须是微信指定的类名   并且这两个Activity一定要在AndroidManifest.xml中注册,上传一张是我做的app中包名跟类名的截图


如何在activity中调用微信登陆

1).登陆广播监听内部类  如果接收到了广播就去获取微信token

	private class WXEntryReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent){MLog.i("WXEntryReceiver", "接收微信登陆广播");if(MLog.debug)showToast("接收微信登陆广播");if(intent.getAction().equals(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN)){int errCode = intent.getExtras().getInt("errCode");if(MLog.debug)System.out.println("获取错误码:"+errCode);if(errCode==BaseResp.ErrCode.ERR_USER_CANCEL||errCode==BaseResp.ErrCode.ERR_AUTH_DENIED){requestDataFinish();}else{String code = intent.getExtras().getString("code");xinTestPresenter.getAccessToken(code);}}}}
2).定义成员变量

    private WXEntryReceiver wxEntryReceiver=null;
3).在oncreate中注册广播

    //微信登陆广播wxEntryReceiver= new WXEntryReceiver();LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this);IntentFilter filter = new IntentFilter();filter.addAction(APIDefineConst.BROADCAST_ACTION_WEIXIN_TOKEN);lbm.registerReceiver(wxEntryReceiver,filter);
4).调用微信登陆

    WeiXinPresenter xinTestPresenter=new WeiXinPresenter(this);xinTestPresenter.login();

在Activity中调用微信分享跟调用微信支付的代码我就不贴出来了,我这篇博客只是给大家一个参考的地方,遇到问题还是建议第一时间看官方文档.


说说我在做微信登陆碰到的问题

1.微信登陆、分享、支付    回调的activity    包名跟类名一定要严格按照要求去写

2.接收回调的是activity  一定要在AndroidManifest.xml中注册

3.WeiXinPresenter中有两个常量   APP_ID跟SECRET  要去微信申请的时候才有的.你们copy代码的时候要给这两个常量赋值

4.可能访问网络神马的还需要一些权限   记得在AndroidManifest.xml添加权限

5.调用微信的登陆、分享、支付   你的安装包一定要有签名,签名信息一定要跟你在微信官网上申请时签名信息一致

6.微信没有客服支持。。。。。如果出了问题看官方demo   或者 官方API

7.微信sdk经常升级,如果你开发的时候有最新的就用最新的吧.....

我在csdn上上传了一个微信sdk的jar包,有需要的可以去下载:http://download.csdn.net/detail/lowprofile_coding/9061367













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

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

相关文章

两年发表14篇论文,其中10篇一作,这是她的科研进阶攻略

全世界只有3.14 % 的人关注了爆炸吧知识本文来源&#xff1a;浙江大学两年发表14篇论文&#xff0c;其中一作10篇&#xff0c;包括4篇Top SCI&#xff0c;2篇SCI和4篇EI&#xff1b;持有2项发明专利&#xff0c;出版1部英文专著&#xff0c;斩获2020年度学生学术十大新成果奖第…

生活在任务栏的猫, CPU使用率越高它就跑的越快

生活在任务栏的猫, CPU使用率越高它就跑的越快Runcat 是一个桌面软件, 这只猫会显示在您的任务栏上面, 它会一直奔跑, 它的运行速度取决于CPU的使用率, 支持 Windows 和 Mac 平台。您还可以用它查看系统资源使用率, 包括CPU使用率,内存,电池状态,网络传输速度等。如果这只猫一直…

在Ant的javac中指定源文件编码方式,以避免警告: 编码 GBK 的不可映射字符的错误...

为什么80%的码农都做不了架构师&#xff1f;>>> * 该错误会造成源文件中的字符串出现混乱&#xff0c;从而影响indexOf()之类函数的正常功能&#xff0e; <javac srcdir"${common.src.dir}" destdir"${build.temp.common.classes.dir}" de…

现在要吃软饭的,都这么明目张胆了吗?

1 加我一个&#xff1f;▼2 为什么电梯里会有两只光着的右脚&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼这样&#xff1f;3 老爹的脸更黑了▼4 但也不影响吧&#xff1f;都不及格▼5 妈&#xff0c;你说的一点都没错&#xff01;&#xff08;素材来源网络&…

Android之app引导页(背景图片切换加各个页面动画效果)

转载&#xff1a;http://blog.csdn.net/lowprofile_coding/article/details/48037095 先看效果图: 1.显示三个页面的Activity 用view pager去加载三个fragment实现,控制点点点的切换,监听view pager的切换,控制fragment动画的开始跟结束,重写了view pager,实现了背景图片的移…

VSCode SSH 连接提示: spawn UNKNOWN

随笔记录 目录 1. 背景介绍 2. 确认问题 : ssh -V 3. 解决问题 3.1 确认本地 ssh.exe 路径 3.2 修改vscode Remote.ssh:Path 3.2.1 设置 Reomte.ssh:Path - 方法一 3.2.2 设置 Reomte.ssh:Path - 方法二 1. 背景介绍 windows 系统vscode ssh remote CentOS7&#xff…

手把手教你学Dapr - 9. 可观测性

介绍通过Tracing(跟踪)、Metrics(指标)、Logs(日志)和Health(运行状况)监控应用程序。分布式跟踪Dapr 使用 Zipkin 协议进行分布式跟踪 和 Metrics 收集。由于 Zipkin 协议的普遍性&#xff0c;许多后端都是开箱即用的&#xff0c;例如 Stackdriver、Zipkin、New Relic 等。结合…

这些魔术用的是物理原理?有啥诀窍?

全世界只有3.14 % 的人关注了爆炸吧知识许多成功的魔术&#xff0c;都运用了物理知识。我们在观看魔术表演时&#xff0c;也可以学到相当多的知识&#xff0c;有时用一般的原理就可以表演一个十分精彩的魔术。所以我们希望揭开魔术的神秘面纱&#xff0c;探究其中的物理知识。一…

安装Wamp时出现无法启动此程序,因为计算机中丢失MSVCR110.dll的解决方法

可能有的朋友在运行某软件时&#xff0c;会出现了“无法启动此程序,因为计算机中丢失 MSVCR110.dll。尝试重新安装该程序以解决此问题。”的提示,遇到这样的情况该怎么办呢&#xff1f;不用着急&#xff0c;下面小编就为大家带来解决方法&#xff0c;遇到同样问题却不知道如何解…

OAuth 2.0 扩展协议之 PKCE

前言阅读本文前需要了解 OAuth 2.0 授权协议的相关内容&#xff0c; 可以参考我的上一篇文章 OAuth 2.0 的探险之旅[1]。PKCE 全称是 Proof Key for Code Exchange&#xff0c; 在2015年发布&#xff0c; 它是 OAuth 2.0 核心的一个扩展协议&#xff0c; 所以可以和现有的授权模…

欧洲杯直播助PPTV日均流量登顶视频行业首位

随着意大利溃败&#xff0c;西班牙最终捧杯&#xff0c;四年一度火爆的欧洲杯赛事也就此落下帷幕。令人欣喜的是&#xff0c;在本届欧洲杯比赛的直播过程中&#xff0c;由于比赛安排再深夜&#xff0c;互联网应用取代传统电视台成为人们观看赛事的主要渠道&#xff0c;各大视频…

.NET 6新特性试用 | record struct

前言在以前的文章中&#xff0c;我们介绍过record类型&#xff0c;它具有不变性(《为什么应该用record来定义DTO》)和值相等性(《为什么应该用record来定义DTO&#xff08;续&#xff09;》)。record是引用类型。而在.NET 6中&#xff0c;我们可以使用record struct定义值类型。…

原来我们看到的世界地图竟这样震撼!多年的地理白学了...

▲ 点击查看几乎每个家庭都会有两张地图&#xff1a;一张世界地图&#xff0c;一张中国地图。薄薄的两张纸&#xff0c;蕴藏着让每个人学会“看世界”的磅礴力量。哈佛上一任校长&#xff0c;也是300多年来唯一一位女校长德鲁吉尔平福斯特&#xff08;Drew Gilpin Faust&#x…

Exchange Powershell查看用户最后登陆邮箱时间

在Exchange日常管理中&#xff0c;我们可能经常会遇到这样的问题&#xff0c;就是怎么来知道一个用户最后的登录时间&#xff1f;这个问题在使用Exchange powershell就能很好的解决了。 用管理员身份运行Exchange management powershell 查看某一个邮箱数据库的统计信息&#x…

易企秀制作的步骤

2019独角兽企业重金招聘Python工程师标准>>> 1、选图很关键 &#xff08;图片干净 整洁&#xff0c;不同方位展示 &#xff0c;符合主题&#xff09;。 2、配上说明性文字 简明扼要 3、选择合适的模板和背景音乐。 4、及时沟通与调整。 转载于:https://my.oschina.n…

C# WPF MVVM模式Prism框架下事件发布与订阅

01—前言处理同模块不同窗体之间的通信和不同模块之间不同窗体的通信&#xff0c;Prism提供了一种事件机制&#xff0c;可以在应用程序中低耦合的模块之间进行通信&#xff0c;该机制基于事件聚合器服务&#xff0c;允许发布者和订阅者之间通过事件进行通讯&#xff0c;且彼此之…

这几部高分学科纪录片,助力孩子涨姿势拓视野~

全世界只有3.14 % 的人关注了爆炸吧知识▌导读本文为同学们整理了几部高分经典学科纪录片&#xff0c;对应文学、数学、经济学、地理、化学。这不仅是课堂学习的补充与延伸&#xff0c;更是开拓视野、激发学习内驱力的绝佳利器。建议收藏&#xff01;&#xff08;关注视频号少年…

Linux 学习_在Linux下面安装tomcat

要在linux下面安装tomcat&#xff0c;首先我们需要做一些准备工作.... 下载tomcat&#xff1a; 下载地址&#xff1a;http://tomcat.apache.org/download-60.cgi 下载&#xff1a;tar.gz 如图&#xff1a; 说明&#xff1a; WinISO安装版&#xff1a;下载地址&#xff1a;http…

Codeforces 474C Captain Marmot 给定4个点和各自旋转中心 问旋转成正方形的次数

题目链接&#xff1a;点击打开链接 题意&#xff1a; 给定T表示case数 以下4行是一个case 每行2个点&#xff0c;u v 每次u能够绕着v逆时针转90 问最少操作多少次使得4个u构成一个正方形。 思路&#xff1a; 枚举判可行 #include <iostream> #include <cmath> #inc…

欧洲的小国家究竟有多袖珍?

全世界只有3.14 % 的人关注了爆炸吧知识你走遍祖国的每个角落了吗&#xff1f;相信绝大多数人的回答是“NO”但是如果你生活在很小的国家里&#xff0c;也许一根烟&#xff0c;或者一顿午餐的时间&#xff0c;你就可以从这个国家的一端走到另一端。下面就一起看看世界上最小的国…